当前位置:首页 > Java技术 > 正文内容

solr 模拟数据库like查询(不使用分词)

canca6年前 (2020-04-01)Java技术834

IK分词个别拆分的不够完美,另外个别业务逻辑是需要替代数据库的like查询。所以本篇文章是介绍如何在solr中使用类似数据库的like查询

本片文章是介绍如何在solr中使用类似数据库的like操作。

首先我们抛弃text_ik。IK分词,因为使用的是like操作,所以这块不能在使用分词了。

我们需要在在conf目录下找到managed-schema配置文件。

增加节点如下:

<fieldType name="string_ci" class="solr.TextField"  sortMissingLast="true" omitNorms="true">
    <analyzer>
        <tokenizer class="solr.KeywordTokenizerFactory"/>
        <filter class="solr.LowerCaseFilterFactory" />
    </analyzer>
</fieldType>

另外配置一下查询字段,如下所示:

<field name="name" type="string_ci" indexed="true"  stored="true"/>

这种简单的配置即可完成like操作。

另外上边的配置增加 LowerCaseFilterFactory 。也就是忽略大小写的意思。

现在博主贴出一个query测试案例:

public static Page<TScanqrSeller> querySellerFromSolrByName(Integer pageNumber, Integer pageSize, Kv cond) {
        Page<TScanqrSeller> page = new Page<TScanqrSeller>();
        if(StringUtils.isNotBlank(cond.getStr("name"))){
            String name = StringKit.escapeQueryChars(cond.getStr("name"));
            cond.set("name", name);
        }
         
        SolrQuery query = new SolrQuery();
        query.setHighlight(false);
        query.addHighlightField("name");
        query.setHighlightSimplePre("<mark>");
        query.setHighlightSimplePost("</mark>");
        query.setStart((pageNumber-1)*pageSize);
        query.setRows(pageSize);
        String sql = Const.BLANK;
        if(StringUtils.isNotBlank(cond.getStr("name"))) {
            sql += " AND name:*"+cond.getStr("name")+"*";
        }
        if(StringUtils.isNotBlank(cond.getStr("companyId"))) {
            sql += " AND company_id:"+cond.getStr("companyId");
        }
        sql = sql.substring(5, sql.length());
        query.set("q", sql);
         
        List<TScanqrSeller> articles = new ArrayList<TScanqrSeller>();
        SolrClient solrClient = createSolrServer(SolrUrlEnum.SOLR_SELLER.getValue());
        try {
            QueryResponse response = solrClient.query(query);
            SolrDocumentList docList = response.getResults();
            Iterator<SolrDocument> it = docList.iterator();
            while (it.hasNext()) {
                SolrDocument doc = it.next();
                String id = doc.getFieldValue("id").toString();
                String name = doc.getFieldValue("name")==null?null:doc.getFieldValue("name").toString();
                Long key = doc.getFieldValue("key")==null?0L:Long.valueOf(doc.getFieldValue("key").toString());
                String shortName = doc.getFieldValue("short_name")==null?null:doc.getFieldValue("short_name").toString();
                String companyId = doc.getFieldValue("company_id")==null?null:doc.getFieldValue("company_id").toString();
                  
                TScanqrSeller seller = new TScanqrSeller();
                seller.set("id", id);
                seller.put("name_text", name);
                seller.set("key", key);
                seller.set("short_name", shortName);
                seller.set("company_id", companyId);
                 
                if(StringUtils.isNotBlank(cond.getStr("name"))) {
                    Pattern p = Pattern.compile(cond.getStr("name"), Pattern.CASE_INSENSITIVE);  
                    Matcher m = p.matcher(name);  
                    while(m.find()) {
                        for(int i=0; i<=m.groupCount(); i++){
                            String oldValue = m.group(i);
                            name = name.replaceFirst(StringKit.escapeQueryChars(oldValue), "<mark>"+oldValue+"</mark>");
                        }
                    }
                    seller.set("name", name);
                }
                 
//              name = name.replaceAll("(?i)"+reqparam, "<mark>"+"(?i)"+reqparam+"</mark>");
                articles.add(seller);
            }
            Integer totalRow = Integer.valueOf(docList.getNumFound()+"");
            Integer totalPage = (int) (totalRow % pageSize == 0 ? totalRow / pageSize : Math.ceil((double)totalRow / (double)pageSize)) ;
            page = new Page<TScanqrSeller>(articles, pageNumber, pageSize, totalPage, totalRow);
        } catch (SolrServerException e) {
            e.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
        } finally{
            try {
                solrClient.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
        return page;
    }

上边有一点需要谨记,多条件查询的时候,连接符AND,要使用大写才能生效。


扫描二维码推送至手机访问。

版权声明:本文由Ant.Master's Blog发布,如需转载请注明出处。

本文链接:https://iant.work/post/709.html

标签: Java技术
分享给朋友:

“solr 模拟数据库like查询(不使用分词)” 的相关文章

对象序列化与反序列化

    序列化,并不是JAVA独有的。因此,在这里我用比较通俗的话说了。序列化就是把一个对象转换成有规则的二进制流。而反序列化就是把有规则的二进制数据重整成一个对象。其好处不难看见:1.可以把一个对象保存在一个文件里。例如,下载软件。当您关闭了软件,下次再打开...

Socket与ServerSocket的问题

//服务器端:import java.io.BufferedReader;import java.io.IOException;import java.io.InputStreamReader;import java.io.PrintStream;import java.net.ServerSock...

Java语言的反射机制

    由于项目的需要,在项目中要实现即插即用的方式,也就是说可以动态地加载包,不用设置CLASSPATH路径。当项目发布时,不可能要用户来设置环境变量吧!因此,就要用到JAVA的反射机制了。昨天,我是在研究JAVA的JNI技术。由于没有时间,所以过几天才写JNI。今天...

JSP与Servlet的对应关系

以前在QQzone写下的文章现在贴到这里来了... 最近比较忙啊!现在抽身写一篇文章。是关于JSP与Servlet的对应关系的。希望对大家有所帮助。其实我也是刚刚学的......-------Servlet--------------JSP----------1.ServletContext&nbs...

JAVA获得一个文件夹大小

在JAVA里没有现成的方法获取一个文件夹的大小,那么我们可以用递归的方法,获取文件夹的大小。    import  java.util.*;  import  java.io.*;  class  GetFileSi...

IM技术(1)

    做项目了,NetCL今天开工了,在这些日子里,我会将自己研究的内容写下来。做个记录,以下是我在网上搜到的。关于管理用户状态的解决方案,当然,我都有一个方案。不过对客户端的任务有点重吧,我方法是客户端从服务器端获到一个用户在线状态后,接着就与服务器无关了。好友离线...

发表评论

访客

◎欢迎参与讨论,请在这里发表您的看法和观点。