compass学习笔记

2023-12-01 10:32
文章标签 学习 笔记 compass

本文主要是介绍compass学习笔记,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

学习compass最好和hibernate对照着学,因为compass 和 hibernate有很多相似之处。

lucene 相当于 JDBC。compass就相当于 hibernate。

Lucene 中的 文档(Document) 相当于 JDBC中数据库的一个表(table)。

Lucene 中的 Filed 相当于 表中的字段。

一个Document中有很多个Filed,一个表中也有多个字段。

 

使用hibernate的步骤和使用compass的步骤基本一致:

 

使用hibernate时步骤一般为:

1.建立实体类,也就是Entity,如Student,Teacher,这些实体类。

   

2.配置实体类到数据库的映射。配置方法一般有两种:

  (1)使用xml。如(Student.hbm.xml)

  (2)使用annotation 注解 @Entity ,@Id等

    Hiberante中要求每个实体都要有个标识(Id),compass中也这么要求。

 

3.使用hibernate的API操作实体

 

Configuration cfg = new Configuration().configure(); //加载hibernate.cfg.xml文件

SessionFactory sessionFactory =cfg.buildSessionFactory(); //全局唯一

Session session = sessionFactory.openSession();

session.beginTransaction(); //开启事务

session.save(xxx);//保存

session.delete(xxx);//删除

session.get(xxx.class,id)//获得对象

........//其他操作

session.getTransaction().commit(); //提交事务

session.close(); //关闭session

   

而使用compass 和 hibernate的步骤几乎是一样的:

 

1.建立实体类,如Student,Teacher等实体类。

 

2.配置实体类到索引库的映射。配置方法一般有两种,xml,annotation,这里主要介绍annotation的配置方法。

掌握五个注解:

@Searchable

@SearchableId

@SearchableProperty

@SearchableComponent

@SearchableReference

 

这里先讲讲@SearchableComponent和@SearchableReference,前三个注解请看后面的例子中的解释。

Compass中一个类对应一个document,而类中的符合属性如

@Searchable

Public class A{

       //@SearchableComponent

       //@SearchableReference

       privateB b;

}

类A和类B在数据库上有关联关系,在java类中,B是A的属性。而在compass中,也可以表示这种关联属性。

使用@SearchableReference标识类B的话,在将实体A存储到compass索引库中后,在查询到A的实体时,如果其对应的B实体也存在索引库中,则可以通过A的实体加载出索引库中的B。

 

@SearchableComponent标识类B的话,则compass会将B作为A的一部分,也就是创建document的时,B的数据元属性值也会被保存到A的document中,而且在进行索引查询时,如果要查询A实体的数据元属性中是否有某个关键字,也会对其对应的B中的数据元属性值进行索引,就算A中没有该关键字,而如果B中有该关键字,则也会把A作为命中对象。

但@SearchableReference只会通过A对B进行加载,不会在索引A的时候也去索引B。这一点要注意。

 

 

以Student类为例:

 

    @Entity //在hibernate中,使用@Entity(放在类的声明上)将Student类映射到数据库的student表

    @Searchable(alias="product") //在compass中,使用@Searchable(放在类的声明上)将Student映射到索引库中的product 文档(Document),alias用来指定该实体类对应的document的名称,不设置alias则默认使用小写的类名。如果searchable的root属性设为false,则该类不会被创建为一个单独的document。一般该类作为另外一个类的组件时,才设置该属性为false。这事要想通过该类的属性搜索该类就会报错。

     public class Student{

 

    //hibernate 和compass中都规定,实体必须有一个标识字段

   

    private intid;

       

   

       

        privateString name;

        privateString fond;

        privateTeacher teacher;

@Id //hiberante中,使用@Id(放在id属性的get方法上)来将一个字段映射为实体的标识ID

    @SearchableId //compass中,则使用@SearchableId(放在id属性的get方法上)将一个字段映射为实体的标识ID

        publicint getId(){

            returnid;

        }

 

//在hibernate中,对应属性如果不设置的话,则默认的映射为对应名称的字段

     //在compass中实体类的属性则需要使用@SearchableProperty(放在属性的get方法上)来设置,设置用来检索的属性,以及属性值是否被索引、是否被存储,以及相关度boost

     //@SearchableProperty中的name属性用来指定filed的名称

//index指定是否建立索引,Index.ANALYZE表示建立索引并进行分词(为默认值),即建立索引时会对该filed进行分词,并在检索的时候可以通过该filed进行搜索

//Index.NOT_ANALYZED表示会对该字段建立索引,但不会进行分词。

//Index.NO 表示不会对该字段建立索引,当然也不会进行分词了。

     //store表示该字段是否要被存储到索引库中,一般指定index=Index.ANALYZED都指定store=Store.YES(默认值),

//指定Index.NOT_ANALYZED并且指定Store.YES,则是为了搜索到实体时,将该filed值加载出来。

     //boost表示相关度,相关度是一个float型的值,值越大,通过字段匹配的结果排序越靠前

@SearchableProperty(name="name",index=Index.ANALYZED,store=Store.YES,boost=3f)

        publicString getName(){

            returnname;

        }

 

        publicString getFond(){

            returnfond;

        }

       

//@SearchableComponent复合类型,该类型的对象Teacher被看做是当前类Student的一部分,一个组件,创建document时,复合类型的属性会和student的属性放在同一个document中。

当进行查询时,会先查询主类的数据元字段,也会查询组件类如Teacher类中的字段,因为在创建document时,teacher的数据元字段作为student的一部分放到了document中。

@SearchableComponent

        public TeachergetTeacher(){

            returnteacher;

}

        public voidsetTeacher(Teacher teacher){

    this.teacher = teacher;

}

    }

 

//把root属性设为false,表示该类不是一个独立的实体,它只是其他类(Student)的一部分

@Searchable(root=false)

public class Teacher {

//此时该类的id就不能做为索引库中的标识了,因为该类被当做是Student的

//的一部分,而Student中已经有标识了。

@SearchableProperty(name="tid",index=Index.NO,store=Store.YES)

    private int id;

    //应为在创建索引时,该name对应的索引字段为name,而在Student中name对应的索引字段也为name,会造成冲突,所以需要将复合类型中与主类属性名字相同的

    //的,该为其他的的名称,以免冲突。

    @SearchableProperty(name="tname",index=Index.ANALYZED,store=Store.YES)

    private String name;

    @SearchableProperty(name="tage",index=Index.NO,store=Store.YES)

    private int age;

    @SearchableProperty(name="tfond",index=Index.ANALYZED,store=Store.YES)

    private String fond;

    //也就是说,查询到Teacher类的某个实体时,可以根据该实体加载出来其对应的student对象。

    @SearchableReference

    private Student student;

    //…………………………省略get/set方法 

}

3.使用Compass中的API对实体进行操作:

    Compasscompass = new CompassConfiguration().configure().bulidCompass();

    //compass 相当于 hibernate的sessionfactory

     //上面的configure()方法会自动去classpath找compass.cfg.xml配置文件,这个方法和hibernate中的configure()方法很相似,hibernate中的该方法用来加载hibernate.cfg.xml.

     //如果要制定其他名称的配置文件,可以使用configure(xxx).

// CompassSession 相当于hibernate中的Session

CompassSessioncompassSession = compass.openSession();

compassSession.beginTransaction(); // CompassTransaction 相当于 Hiberante中的Transaction,即compss中也是有事务支持的,和数据库很相似(也有事务回滚tx.rollback())

compassSession.create(entity)/.create(entity); //保存到索引文件

compassSession.delete(entity);//将实体从索引库中删除

compassSession.commit();

compassSession.close();

 

 

 

下面是测试时使用compass进行保存,查询等操作:

public class TestCompass {

    static Compass compass = null;

    @BeforeClass

    public static void beforeClass(){

       compass = new CompassAnnotationsConfiguration()

                     .setSetting(CompassEnvironment.CONNECTION, "file://index")

//                   .setSetting(CompassEnvironment.CONNECTION,"ram://index") //索引创建在内存中

                     .setSetting("compass.engine.highlighter.default.formatter.simple.pre", "<fontcolor='red'>")

                      .setSetting("compass.engine.highlighter.default.formatter.simple.post","</font>")

                     .addScan("com.wang.domain")

                     .buildCompass();    

    }

    @Test

    public void save(){

       CompassTransaction tx = null;

       CompassSession session = null;

       try {

           session = compass.openSession();

           tx = session.beginTransaction();

           Student st = new Student();

           st.setId(99);

           st.setAge(33);

           st.setName("凌飞");

           st.setFond("篮球飞");

           st.setSex("飞机");

           session.create(st);

//         session.save(st); //save和create方法是一样的。

           tx.commit();

       } catch (CompassException e) {

           tx.rollback();

           e.printStackTrace();

       } finally {

           if(null!=session&&!session.isClosed()){

              session.close();

            }

       }  

    }

    @Test

    public void getAll(){

       CompassTransaction tx = null;

       CompassSession session = null;

       try {

           session = compass.openSession();

           tx = session.beginTransaction();

//         如果直接find("楠"),则默认查找所有的索引filed(出去id标识之外的字段)。

//         CompassHits hits = session.find("楠");

          

//         如果要单在某个索引字段中查找,如只在name字段中查找,可以使用"name:楠"的形式。

//         CompassHits hits = session.find("name:楠");

          

//         如果要在多个filed中查找,则使用"name:楠 and fond:女" 或 "name:楠 + fond:女"。

//         compass在通过session.find查找到结果后,并不会直接把结果集保存到内存中(也就是hits),hits保存的只是结果记录在索引库中的索引号

//         在hits调用data(i)方法时,才会真正的返回指定索引号(i)的结果

           CompassHits hits = session.find("name:楠 and fond:女");

           List<Student> stuList = new ArrayList<Student>();

           for(int i = 0;i<hits.length();i++){

//            在hits调用data(i)方法时,才会真正的返回指定索引号(i)的结果

              Student st = (Student)hits.data(i);

              st.setName(hits.highlighter(i).fragment("name"));

              st.setFond(hits.highlighter(i).fragment("fond"));

              stuList.add(st);

           }

          

           for(Student st:stuList){

              System.out.println(st.getName()+"|"+st.getFond()+"|"+st.getId());

           }

           tx.commit();

       } catch (CompassException e) {

           tx.rollback();

           e.printStackTrace();

       } finally {

           if(null!=session&&!session.isClosed()){

              session.close();

           }

       }

    }

   

    //compass有保存,删除,但没有更新方法,要更新的话,可以先查询出来对象,然后删除对象,修改后再保存一次。

    @Test

    public void deleteAll(){

       CompassTransaction tx = null;

       CompassSession session = null;

       try {

           session = compass.openSession();

           tx = session.beginTransaction();

           CompassHits hits = session.find("name:楠 and fond:女");

           List<Student> stuList = new ArrayList<Student>();

           for(int i = 0;i<hits.length();i++){

              Student st = (Student)hits.data(i);

              session.delete(st);//删除

           }

          

           for(Student st:stuList){

              System.out.println(st.getName()+"|"+st.getFond()+"|"+st.getId());

           }

           tx.commit();

       } catch (CompassException e) {

           tx.rollback();

           e.printStackTrace();

       } finally {

           if(null!=session&&!session.isClosed()){

              session.close();

           }

       }

    }

}

 

Compass 与Spring的集成

在compass与spring的集成 和 hibernate与spring的集成也有很多相似之处:

1、 compass的事务也是交由spring管理的。

先思考一下compass不与spring集成时的情况。

我们将一个entity保存到数据库的同时,也要将该实体相关的数据信息保存到索引库中。所以代码常为一下情况:

Public  void saveStudent(Student stu){

        hibernateTeplate.save(stu);//保存到数据库中

      compassSession.save(stu);//保存到索引库中

}

我们有一点需要保证的是,必须在数据库保存成功后才能将实体对象保存到索引库中,然而在没有任何外界条件限制的情况先,数据库的事务和compass的事务是两个不同体系的事务,他们不可能自动的将事务并为一起。这样就无法保证整个操作的流程控制在同一个事务中,所以我们就要想办法来解决这个问题。

使用compass与spring集成,能很好的解决这个问题,spring会为hibernate与compass提供同一个事务管理,将整个操作流程(如上的saveStudent()方法)纳入同一个事务支持当中。

下面看一下配置中hibernate、compass、spring的集成:

其中hibernate与spring集成与前面完全一致,就是在后面加了一个compass的配置:

-------------下面就是 spring+JPA(Hiberante实现)+compass的集成----------

<aop:aspectj-autoproxy/>

<context:component-scan base-package="cn.itcast"/>  

<bean id="entityManagerFactory"class="org.springframework.orm.jpa.LocalEntityManagerFactoryBean">

       <property name="persistenceUnitName" value="itcast"/>

</bean>  

<bean id="transactionManager"class="org.springframework.orm.jpa.JpaTransactionManager">

        <property name="entityManagerFactory"ref="entityManagerFactory"/>

</bean>

  

<tx:annotation-driven transaction-manager="transactionManager"/>

<!—compass对象相当于 hibernate中的sessionFactory,在整个应用中时单例,这里使用org.compass.spring.LocalCompassBean,表示由spring来生成并管理compass对象,生成该对象需要一些属性设置。

 -->

<bean id="compass" class="org.compass.spring.LocalCompassBean">

    <!—配置compass的实体类映射-->

    <property name="classMappings">

       <list>

           <value>cn.itcast.bean.product.ProductInfo</value>

           <value>cn.itcast.bean.product.Brand</value>

           <value>cn.itcast.bean.product.ProductStyle</value>

           <value>cn.itcast.bean.product.ProductType</value>

       </list>

    </property>

    <!---compass相关的一些属性设置-->

    <property name="compassSettings">

       <props>

<!—- 指定compass的默认分词器,为:paoding分词,paoding分词器是基于字典的分词器,所以最好字典,使用paoding分词器时,将paoding分词器压缩目录下的字典目录(dic)放到项目的src目录下,并将paoding分词的paoding-dic-home.properties文件拷贝到src下,并在该配置文件中指定字典dic的路径为:classpath:dic。如果需要自定义一些字词,只需在dic目录下的字典文件中加上该词即可。-->

<prop key="compass.engine.analyzer.default.type">

               net.paoding.analysis.analyzer.PaodingAnalyzer

</prop>

<!—- compass索引库的位置 -->

<!-- file://e:/index 中以file://开头表示在硬盘上指定位置创建索引库-->

<!—  ram:// index 中以ram://开头表示在内存中创建索引库-->

           <prop key="compass.engine.connection">

file://e:/index

</prop>

<!—- 对关键字进行高亮(highlighter)设置 -->

           <prop key="compass.engine.highlighter.default。。。。pre">

<![CDATA[<font color='red'>]]>

</prop>

           <prop key="compass.engine.highlighter.。。。.simple.post">

<![CDATA[</font>]]>

</prop>

<!-- transaction.factory用于指定compass的事务工厂,这里指定为spring提供的事务工厂类,该工厂类会从spring中获得当前的事务,compass就可以使用当前事务了 -->

           <prop key="compass.transaction.factory">

org.compass.spring.transaction.SpringSyncTransactionFactory

</prop>

       </props>

    </property>

<!—-spring中的事务是由事务管理器来管理的,上面的指定的compass的transaction.factory要从spring获得当前事务,就必须去访问spring的事务管理器,这里就指定了compass的事务工厂要访问的spring的事务管理器,该事务管理器在前面已经配置好了,和hibernate那个一致,这里只需要引用就可以了-->

    <property name="transactionManager" ref="transactionManager" />

</bean>

    <!-- SingleCompassGps 类是compass的GPS机制的实现类,spring在实例化该类的对象后,会立即执行init-method属性指定的SingleCompassGps中的start()方法,start方法用于启动对 hibernate或JPA等持久化工具对实体类的增删改查的操作的监听捕获,因为hibernate或JPA等的持久化操作都会触发一些生命周期事件,而GPS可以对这些生命周期事件进行监听捕获,从而可以使用compass同步的将实体类信息更新到索引库中去-->

<bean id="compassGps" class="org.compass.gps.impl.SingleCompassGps"init-method="start" destroy-method="stop">

<!—GPS在监听到hibernate、JPA等对实体类的操作时,要使用compass对象将实体信息更新到索引库中,这里就是指出GPS要使用的compass对象,该compass对象在前面已经配置,这里只需要引用即可 -->

    <property name="compass" ref="compass"/>

    <property name="gpsDevices">

        <list>

<bean class="org.compass.spring.device.SpringSyncTransactionGpsDeviceWrapper">

               <property name="gpsDevice" >

<!—GPS要监听与实体类对象相关的操作的生命周期,就必须知道是什么产品的工具在对实体类进行操作,这些工具可能是hibernate,JPA等,本项目使用的是JPA,所以要配置JPA相关的驱动,用来监听JPA操作实体时生命周期所发生的变化-->

              <bean class="org.compass.gps.device.jpa.JpaGpsDevice">

                    <!—- 这是设置的名字,随便取 -->

                  <property name="name" value="jpaDevice" />

<!—GPS要监听实体被操作的生命周期的变化,就需要去访问管理实体的工厂对象,这里指定entityManagerFactory 为前面配置的JPA提供的实体管理工厂-->

           <property name="entityManagerFactory"ref="entityManagerFactory" />

<!—        指定是否监听与实体相关的声明周期事件     -->

                  <property name="injectEntityLifecycleListener" value="true"/>

              </bean>

               </property>

           </bean>

        </list>

</property>

</bean>

 

-------------------------------compass+hibernate+spring---------------------

Compass+hibernate+spring的集成和前面的配置过程大致差不多,最后配置gps时是这样写的:

这篇关于compass学习笔记的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/440820

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss

【学习笔记】 陈强-机器学习-Python-Ch15 人工神经网络(1)sklearn

系列文章目录 监督学习:参数方法 【学习笔记】 陈强-机器学习-Python-Ch4 线性回归 【学习笔记】 陈强-机器学习-Python-Ch5 逻辑回归 【课后题练习】 陈强-机器学习-Python-Ch5 逻辑回归(SAheart.csv) 【学习笔记】 陈强-机器学习-Python-Ch6 多项逻辑回归 【学习笔记 及 课后题练习】 陈强-机器学习-Python-Ch7 判别分析 【学

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识

线性代数|机器学习-P36在图中找聚类

文章目录 1. 常见图结构2. 谱聚类 感觉后面几节课的内容跨越太大,需要补充太多的知识点,教授讲得内容跨越较大,一般一节课的内容是书本上的一章节内容,所以看视频比较吃力,需要先预习课本内容后才能够很好的理解教授讲解的知识点。 1. 常见图结构 假设我们有如下图结构: Adjacency Matrix:行和列表示的是节点的位置,A[i,j]表示的第 i 个节点和第 j 个

Node.js学习记录(二)

目录 一、express 1、初识express 2、安装express 3、创建并启动web服务器 4、监听 GET&POST 请求、响应内容给客户端 5、获取URL中携带的查询参数 6、获取URL中动态参数 7、静态资源托管 二、工具nodemon 三、express路由 1、express中路由 2、路由的匹配 3、路由模块化 4、路由模块添加前缀 四、中间件