利用java反射机制编写solr通用的java客户端

一、前言

  通过上一篇的讲解,我们知道了dynamicFiled字段,它是动态的,不需要显示的声明。而且一些常用的基本类型solr已经默认给我们创建好了。

  例如:*_i,*_is,等。

  如果我们要使用动态字段,字段的命名就需要符合上述规则。solr为我们提供了大量的动态字段:

  

二、实体类的编写

  在这里,我们以创建商品的索引为例,创建实体如下:

@Getter@Setter
public class Product {
  //商品id,而且是必有字段
private String id;
//商品名称,是字符串类型,所以我们以"_s"结尾
private String proName_s;
  //商品架构 是double型,所以以"_d"结尾
private Double price_d;
  //商品分类 是字符串类型,而且一个商品可以有多个分类,是多值,所以我们用“_ss”结尾
private List<String> tag_ss; }

三、solrj编写java通用客户端

  我们主要是通过java的反射机制和泛型来编写:

package com.urwork.tools.solr;

import com.urwork.tools.page.Page;
import org.apache.commons.collections.CollectionUtils;
import org.apache.solr.client.solrj.SolrQuery;
import org.apache.solr.client.solrj.impl.CloudSolrClient;
import org.apache.solr.client.solrj.response.QueryResponse;
import org.apache.solr.common.SolrDocument;
import org.apache.solr.common.SolrDocumentList;
import org.apache.solr.common.SolrInputDocument;

import java.io.IOException;
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.List;

/**
 * Created by xxx on xxxx/xx/xx.
 */
public class SolrCloudClient {
    private CloudSolrClient server=null;

    /**
     * 构造函数
     * @param zkAddr        zk地址:192.168.2.233:2181,192.168.2.234:2181,192.168.2.235:2181
     * @param collection    collection名字: company
     */
    private SolrCloudClient(String zkAddr,String collection) {
        server = new CloudSolrClient.Builder().withZkHost(zkAddr).build();
        server.setDefaultCollection(collection);
    }

    /**
     * 删除集合中的所有数据
     * @throws Exception
     */
    public void deleteAll() throws Exception {
        server.deleteByQuery("*:*");
        server.commit();
    }

    /**
     * 根据id删除集合中的数据
     * @param id
     * @throws Exception
     */
    public void deleteById(String id) throws Exception {
        server.deleteById(id);
        server.commit();
    }

    /**
     * 根据ids删除集合中的数据
     * @param ids
     * @throws Exception
     */
    public void deleteByIds(List<String> ids) throws Exception {
        server.deleteById(ids);
        server.commit();
    }

    /**
     * 批量更新索引
     * @param docs
     * @param <T>
     * @throws Exception
     */
    public <T> void addList(List<T> docs) throws Exception {
        if (CollectionUtils.isEmpty(docs)){
            return;
        }
        List<SolrInputDocument> list = new ArrayList<>();
        for (T doc:docs){
            Field[] declaredFields = doc.getClass().getDeclaredFields();
            SolrInputDocument sid = new SolrInputDocument();
            for (Field field : declaredFields){
                field.setAccessible(true);
                String name = field.getName();
                Object value = field.get(doc);
                if (value!=null){
                    sid.addField(name,value);
                }
                list.add(sid);
            }
        }

        server.add(list);
        server.commit();
    }

    /**
     * 添加单条索引
     * @param doc
     * @param <T>
     * @throws Exception
     */
    public <T> void add(T doc) throws Exception {
        List<T> docs = new ArrayList<>();
        docs.add(doc);
        addList(docs);
    }

    /**
     * 更新一条索引
     * @param doc
     * @param <T>
     * @throws Exception
     */
    public <T> void update(T doc)throws Exception{
        Field idField = doc.getClass().getDeclaredField("id");
        if (idField == null){
            throw new RuntimeException("your document doesn't have id");
        }
        idField.setAccessible(true);
        Object id = idField.get(doc);
        deleteById(id+"");
        add(doc);
    }

    /**
     * 批量更新索引
     * @param docs
     * @param <T>
     * @throws Exception
     */
    public <T> void updateList(List<T> docs) throws Exception {
        if (CollectionUtils.isEmpty(docs)){
            return;
        }
        List<String> ids = new ArrayList<>();
        for (T doc : docs){
            Field idField = doc.getClass().getDeclaredField("id");
            if (idField!=null){
                idField.setAccessible(true);
                Object id = idField.get(doc);
                ids.add(id+"");
            }
        }
        deleteByIds(ids);
        addList(docs);
    }

    /**
     * 执行查询
     * @param clazz      对应的class
     * @param page       Page分页
     * @param query      查询条件
     * @param <T>
     * @return
     * @throws Exception
     */
    public <T> Page<T> search(Class<T> clazz,Page<T> page, SolrQuery query) throws Exception {
        query.setStart((page.getCurrentPageNo()-1)*page.getPageSize());
        query.setRows(page.getPageSize());

        QueryResponse response = server.query(query);
        SolrDocumentList results = response.getResults();
        List<T> rtnList = new ArrayList<>();
        for (SolrDocument doc : results){
            T instance = clazz.newInstance();
            Field[] declaredFields = instance.getClass().getDeclaredFields();
            for (Field filed : declaredFields){
                filed.setAccessible(true);
                String name = filed.getName();
                Object fieldValue =doc.getFieldValue(name);
                filed.set(instance,fieldValue);
            }
            rtnList.add(instance);
        }
        page.setResult(rtnList);
        page.setTotalRecord((int)results.getNumFound());
        return page;
    }

    public void close() throws IOException {
        server.close();
    }

}

  我们在创建实体类时,字段的名称按照动态值的规则命名,在构建索引和查询时,就可以使用公共类来实现。

  

 

posted @ 2017-07-03 17:16  牛初九  阅读(446)  评论(0编辑  收藏  举报