Lucene 中自定义排序的实现

使用Lucene来搜索内容,搜索结果的显示顺序当然是比较重要的.Lucene中Build-in的几个排序定义在大多数情况下是不适合我们使用的.要适合自己的应用程序的场景,就只能自定义排序功能,本节我们就来看看在Lucene中如何实现自定义排序功能.

Lucene中的自定义排序功能和Java集合中的自定义排序的实现方法差不多,都要实现一下比较接口. 在Java中只要实现Comparable接口就可以了.但是在Lucene中要实现SortComparatorSource接口和ScoreDocComparator接口.在了解具体实现方法之前先来看看这两个接口的定义吧.

SortComparatorSource接口的功能是返回一个用来排序ScoreDocs的comparator(Expert: returns a comparator for sorting ScoreDocs).该接口只定义了一个方法.如下:

Java代码 复制代码
  1. /**
  2. *Createsacomparatorforthefieldinthegivenindex.
  3. *@paramreader-Indextocreatecomparatorfor.
  4. *@paramfieldname-Fieldtocreatecomparatorfor.
  5. *@returnComparatorofScoreDocobjects.
  6. *@throwsIOException-Ifanerroroccursreadingtheindex.
  7. */
  8. publicScoreDocComparatornewComparator(IndexReaderreader,Stringfieldname)throwsIOException
/**
 * Creates a comparator for the field in the given index. 
 * @param reader - Index to create comparator for. 
 * @param fieldname - Field to create comparator for.
 * @return Comparator of ScoreDoc objects. 
 * @throws IOException - If an error occurs reading the index. 
 */
public ScoreDocComparator newComparator(IndexReader reader,String fieldname) throws IOException

该方法只是创造一个ScoreDocComparator 实例用来实现排序.所以我们还要实现ScoreDocComparator 接口.来看看ScoreDocComparator 接口.功能是比较来两个ScoreDoc 对象来排序(Compares two ScoreDoc objects for sorting) 里面定义了两个Lucene实现的静态实例.如下:

Java代码 复制代码
  1. //Specialcomparatorforsortinghitsaccordingtocomputedrelevance(documentscore).
  2. publicstaticfinalScoreDocComparatorRELEVANCE;
  3. //Specialcomparatorforsortinghitsaccordingtoindexorder(documentnumber).
  4. publicstaticfinalScoreDocComparatorINDEXORDER;
//Special comparator for sorting hits according to computed relevance (document score). 
public static final ScoreDocComparator RELEVANCE;
	
//Special comparator for sorting hits according to index order (document number). 
public static final ScoreDocComparator INDEXORDER;

有3个方法与排序相关,需要我们实现 分别如下:

Java代码 复制代码
  1. /**
  2. *ComparestwoScoreDocobjectsandreturnsaresultindicatingtheirsortorder.
  3. *@paramiFirstScoreDoc
  4. *@paramjSecondScoreDoc
  5. *@return-1ifishouldcomebeforej;
  6. *1ifishouldcomeafterj;
  7. *0iftheyareequal
  8. */
  9. publicintcompare(ScoreDoci,ScoreDocj);
  10. /**
  11. *Returnsthevalueusedtosortthegivendocument.Theobjectreturnedmustimplementthejava.io.Serializableinterface.Thisisusedbymultisearcherstodeterminehowtocollateresultsfromtheirsearchers.
  12. *@paramiDocument
  13. *@returnSerializableobject
  14. */
  15. publicComparablesortValue(ScoreDoci);
  16. /**
  17. *Returnsthetypeofsort.ShouldreturnSortField.SCORE,SortField.DOC,SortField.STRING,SortField.INTEGER,SortField.FLOATorSortField.CUSTOM.ItisnotvalidtoreturnSortField.AUTO.Thisisusedbymultisearcherstodeterminehowtocollateresultsfromtheirsearchers.
  18. *@returnOneoftheconstantsinSortField.
  19. */
  20. publicintsortType();
               /**
	 * Compares two ScoreDoc objects and returns a result indicating their sort order. 
	 * @param i First ScoreDoc 
	 * @param j Second ScoreDoc
	 * @return -1 if i should come before j; 
	 *         1 if i should come after j;
	 *         0 if they are equal
	 */
	public int compare(ScoreDoc i,ScoreDoc j);

	/**
	 * Returns the value used to sort the given document. The object returned must implement the java.io.Serializable interface. This is used by multisearchers to determine how to collate results from their searchers. 
	 * @param i Document
	 * @return Serializable object
	 */
	public Comparable sortValue(ScoreDoc i);

	/**
	 * Returns the type of sort. Should return SortField.SCORE, SortField.DOC, SortField.STRING, SortField.INTEGER, SortField.FLOAT or SortField.CUSTOM. It is not valid to return SortField.AUTO. This is used by multisearchers to determine how to collate results from their searchers. 
	 * @return One of the constants in SortField. 
	 */
	public int sortType();

看个例子吧!

该例子为Lucene in Action中的一个实现,用来搜索距你最近的餐馆的名字. 餐馆坐标用字符串"x,y"来存储.

Java代码 复制代码
  1. packagecom.nikee.lucene;
  2. importjava.io.IOException;
  3. importorg.apache.lucene.index.IndexReader;
  4. importorg.apache.lucene.index.Term;
  5. importorg.apache.lucene.index.TermDocs;
  6. importorg.apache.lucene.index.TermEnum;
  7. importorg.apache.lucene.search.ScoreDoc;
  8. importorg.apache.lucene.search.ScoreDocComparator;
  9. importorg.apache.lucene.search.SortComparatorSource;
  10. importorg.apache.lucene.search.SortField;
  11. //实现了搜索距你最近的餐馆的名字.餐馆坐标用字符串"x,y"来存储
  12. //DistanceComparatorSource实现了SortComparatorSource接口
  13. publicclassDistanceComparatorSourceimplementsSortComparatorSource{
  14. privatestaticfinallongserialVersionUID=1L;
  15. //xy用来保存坐标位置
  16. privateintx;
  17. privateinty;
  18. publicDistanceComparatorSource(intx,inty){
  19. this.x=x;
  20. this.y=y;
  21. }
  22. //返回ScoreDocComparator用来实现排序功能
  23. publicScoreDocComparatornewComparator(IndexReaderreader,Stringfieldname)throwsIOException{
  24. returnnewDistanceScoreDocLookupComparator(reader,fieldname,x,y);
  25. }
  26. //DistanceScoreDocLookupComparator实现了ScoreDocComparator用来排序
  27. privatestaticclassDistanceScoreDocLookupComparatorimplementsScoreDocComparator{
  28. privatefloat[]distances;//保存每个餐馆到指定点的距离
  29. //构造函数,构造函数在这里几乎完成所有的准备工作.
  30. publicDistanceScoreDocLookupComparator(IndexReaderreader,Stringfieldname,intx,inty)throwsIOException{
  31. System.out.println("fieldName2="+fieldname);
  32. finalTermEnumenumerator=reader.terms(newTerm(fieldname,""));
  33. System.out.println("maxDoc="+reader.maxDoc());
  34. distances=newfloat[reader.maxDoc()];//初始化distances
  35. if(distances.length>0){
  36. TermDocstermDocs=reader.termDocs();
  37. try{
  38. if(enumerator.term()==null){
  39. thrownewRuntimeException("notermsinfield"+fieldname);
  40. }
  41. inti=0,j=0;
  42. do{
  43. System.out.println("indo-while:"+i++);
  44. Termterm=enumerator.term();//取出每一个Term
  45. if(term.field()!=fieldname)//与给定的域不符合则比较下一个
  46. break;
  47. //SetsthistothedataforthecurrentterminaTermEnum.
  48. //Thismaybeoptimizedinsomeimplementations.
  49. termDocs.seek(enumerator);//参考TermDocsDoc
  50. while(termDocs.next()){
  51. System.out.println("inwhile:"+j++);
  52. System.out.println("inwhile,Term:"+term.toString());
  53. String[]xy=term.text().split(",");//去处xy
  54. intdeltax=Integer.parseInt(xy[0])-x;
  55. intdeltay=Integer.parseInt(xy[1])-y;
  56. //计算距离
  57. distances[termDocs.doc()]=(float)Math.sqrt(deltax*deltax+deltay*deltay);
  58. }
  59. }
  60. while(enumerator.next());
  61. }finally{
  62. termDocs.close();
  63. }
  64. }
  65. }
  66. //有上面的构造函数的准备这里就比较简单了
  67. publicintcompare(ScoreDoci,ScoreDocj){
  68. if(distances[i.doc]<distances[j.doc])
  69. return-1;
  70. if(distances[i.doc]>distances[j.doc])
  71. return1;
  72. return0;
  73. }
  74. //返回距离
  75. publicComparablesortValue(ScoreDoci){
  76. returnnewFloat(distances[i.doc]);
  77. }
  78. //指定SortType
  79. publicintsortType(){
  80. returnSortField.FLOAT;
  81. }
  82. }
  83. publicStringtoString(){
  84. return"Distancefrom("+x+","+y+")";
  85. }
  86. }
package com.nikee.lucene;

import java.io.IOException;

import org.apache.lucene.index.IndexReader;
import org.apache.lucene.index.Term;
import org.apache.lucene.index.TermDocs;
import org.apache.lucene.index.TermEnum;
import org.apache.lucene.search.ScoreDoc;
import org.apache.lucene.search.ScoreDocComparator;
import org.apache.lucene.search.SortComparatorSource;
import org.apache.lucene.search.SortField;

//实现了搜索距你最近的餐馆的名字. 餐馆坐标用字符串"x,y"来存储
//DistanceComparatorSource 实现了SortComparatorSource接口
public class DistanceComparatorSource implements SortComparatorSource {
	private static final long serialVersionUID = 1L;
	
	// x y 用来保存 坐标位置
	private int x;
	private int y;
	
	public DistanceComparatorSource(int x, int y) {
		this.x = x;
		this.y = y;
	}
	
	// 返回ScoreDocComparator 用来实现排序功能
	public ScoreDocComparator newComparator(IndexReader reader, String fieldname) throws IOException {
		return new DistanceScoreDocLookupComparator(reader, fieldname, x, y);
	}
	
	//DistanceScoreDocLookupComparator 实现了ScoreDocComparator 用来排序
	private static class DistanceScoreDocLookupComparator implements ScoreDocComparator {
		private float[] distances;  // 保存每个餐馆到指定点的距离
		
		// 构造函数 , 构造函数在这里几乎完成所有的准备工作.
		public DistanceScoreDocLookupComparator(IndexReader reader, String fieldname, int x, int y) throws IOException {
			System.out.println("fieldName2="+fieldname);
			final TermEnum enumerator = reader.terms(new Term(fieldname, ""));
			
			System.out.println("maxDoc="+reader.maxDoc());
			distances = new float[reader.maxDoc()];  // 初始化distances
			if (distances.length > 0) {
				TermDocs termDocs = reader.termDocs();
				try {
					if (enumerator.term() == null) {
						throw new RuntimeException("no terms in field " + fieldname);
					}
					int i = 0,j = 0;
					do {
						System.out.println("in do-while :" + i ++);
						Term term = enumerator.term();  // 取出每一个Term 
						if (term.field() != fieldname)  // 与给定的域不符合则比较下一个
							break;
						
						//Sets this to the data for the current term in a TermEnum. 
						//This may be optimized in some implementations.
						termDocs.seek(enumerator); //参考TermDocs Doc
						while (termDocs.next()) {
							System.out.println("    in while :" + j ++);
							System.out.println("    in while ,Term :" + term.toString());
							
							String[] xy = term.text().split(","); // 去处x y
							int deltax = Integer.parseInt(xy[0]) - x;
							int deltay = Integer.parseInt(xy[1]) - y;
							// 计算距离
							distances[termDocs.doc()] = (float) Math.sqrt(deltax * deltax + deltay * deltay);
						}
					} 
					while (enumerator.next());
				} finally {
					termDocs.close();
				}
			}
		}

		//有上面的构造函数的准备 这里就比较简单了
		public int compare(ScoreDoc i, ScoreDoc j) {
			if (distances[i.doc] < distances[j.doc])
				return -1;
			if (distances[i.doc] > distances[j.doc])
				return 1;
			return 0;
		}
		
		// 返回距离
		public Comparable sortValue(ScoreDoc i) {
			return new Float(distances[i.doc]);
		}
		
		//指定SortType
		public int sortType() {
			return SortField.FLOAT;
		}
	}
		
	public String toString() {
		return "Distance from (" + x + "," + y + ")";
	}
} 



这是一个实现了上面两个接口的两个类, 里面带有详细注释, 可以看出 自定义排序并不是很难的. 该实现能否正确实现,我们来看看测试代码能否通过吧.

Java代码 复制代码
  1. packagecom.nikee.lucene.test;
  2. importjava.io.IOException;
  3. importjunit.framework.TestCase;
  4. importorg.apache.lucene.analysis.WhitespaceAnalyzer;
  5. importorg.apache.lucene.document.Document;
  6. importorg.apache.lucene.document.Field;
  7. importorg.apache.lucene.index.IndexWriter;
  8. importorg.apache.lucene.index.Term;
  9. importorg.apache.lucene.search.FieldDoc;
  10. importorg.apache.lucene.search.Hits;
  11. importorg.apache.lucene.search.IndexSearcher;
  12. importorg.apache.lucene.search.Query;
  13. importorg.apache.lucene.search.ScoreDoc;
  14. importorg.apache.lucene.search.Sort;
  15. importorg.apache.lucene.search.SortField;
  16. importorg.apache.lucene.search.TermQuery;
  17. importorg.apache.lucene.search.TopFieldDocs;
  18. importorg.apache.lucene.store.RAMDirectory;
  19. importcom.nikee.lucene.DistanceComparatorSource;
  20. publicclassDistanceComparatorSourceTestextendsTestCase{
  21. privateRAMDirectorydirectory;
  22. privateIndexSearchersearcher;
  23. privateQueryquery;
  24. //建立测试环境
  25. protectedvoidsetUp()throwsException{
  26. directory=newRAMDirectory();
  27. IndexWriterwriter=newIndexWriter(directory,newWhitespaceAnalyzer(),true);
  28. addPoint(writer,"ElCharro","restaurant",1,2);
  29. addPoint(writer,"CafePocaCosa","restaurant",5,9);
  30. addPoint(writer,"LosBetos","restaurant",9,6);
  31. addPoint(writer,"Nico'sTacoShop","restaurant",3,8);
  32. writer.close();
  33. searcher=newIndexSearcher(directory);
  34. query=newTermQuery(newTerm("type","restaurant"));
  35. }
  36. privatevoidaddPoint(IndexWriterwriter,Stringname,Stringtype,intx,inty)throwsIOException{
  37. Documentdoc=newDocument();
  38. doc.add(newField("name",name,Field.Store.YES,Field.Index.TOKENIZED));
  39. doc.add(newField("type",type,Field.Store.YES,Field.Index.TOKENIZED));
  40. doc.add(newField("location",x+","+y,Field.Store.YES,Field.Index.UN_TOKENIZED));
  41. writer.addDocument(doc);
  42. }
  43. publicvoidtestNearestRestaurantToHome()throwsException{
  44. //使用DistanceComparatorSource来构造一个SortField
  45. Sortsort=newSort(newSortField("location",newDistanceComparatorSource(0,0)));
  46. Hitshits=searcher.search(query,sort);//搜索
  47. //测试
  48. assertEquals("closest","ElCharro",hits.doc(0).get("name"));
  49. assertEquals("furthest","LosBetos",hits.doc(3).get("name"));
  50. }
  51. publicvoidtestNeareastRestaurantToWork()throwsException{
  52. Sortsort=newSort(newSortField("location",newDistanceComparatorSource(10,10)));//工作的坐标10,10
  53. //上面的测试实现了自定义排序,但是并不能访问自定义排序的更详细信息,利用
  54. //TopFieldDocs可以进一步访问相关信息
  55. TopFieldDocsdocs=searcher.search(query,null,3,sort);
  56. assertEquals(4,docs.totalHits);
  57. assertEquals(3,docs.scoreDocs.length);
  58. //取得FieldDoc利用FieldDoc可以取得关于排序的更详细信息请查看FieldDocDoc
  59. FieldDocfieldDoc=(FieldDoc)docs.scoreDocs[0];
  60. assertEquals("(10,10)->(9,6)=sqrt(17)",newFloat(Math.sqrt(17)),fieldDoc.fields[0]);
  61. Documentdocument=searcher.doc(fieldDoc.doc);
  62. assertEquals("LosBetos",document.get("name"));
  63. dumpDocs(sort,docs);//显示相关信息
  64. }
  65. //显示有关排序的信息
  66. privatevoiddumpDocs(Sortsort,TopFieldDocsdocs)throwsIOException{
  67. System.out.println("Sortedby:"+sort);
  68. ScoreDoc[]scoreDocs=docs.scoreDocs;
  69. for(inti=0;i<scoreDocs.length;i++){
  70. FieldDocfieldDoc=(FieldDoc)scoreDocs[i];
  71. Floatdistance=(Float)fieldDoc.fields[0];
  72. Documentdoc=searcher.doc(fieldDoc.doc);
  73. System.out.println(""+doc.get("name")+"@("+doc.get("location")+")->"+distance);
  74. }
  75. }
  76. }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值