<?xml version="1.0" encoding="UTF-8"?><rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>standard &#8211; Solr.pl</title>
	<atom:link href="https://solr.pl/tag/standard/feed/" rel="self" type="application/rss+xml" />
	<link>https://solr.pl</link>
	<description>All things to be found - Blog related to Apache Solr &#38; Lucene projects - https://solr.apache.org</description>
	<lastBuildDate>Tue, 10 Nov 2020 08:55:30 +0000</lastBuildDate>
	<language>pl-PL</language>
	<sy:updatePeriod>
	hourly	</sy:updatePeriod>
	<sy:updateFrequency>
	1	</sy:updateFrequency>
	<generator>https://wordpress.org/?v=6.9</generator>
	<item>
		<title>Solr i PhraseQuery, czyli różne sposoby premiowania fraz</title>
		<link>https://solr.pl/2010/07/14/solr-i-phrasequery-czyli-rozne-sposoby-premiowania-fraz/</link>
					<comments>https://solr.pl/2010/07/14/solr-i-phrasequery-czyli-rozne-sposoby-premiowania-fraz/#comments</comments>
		
		<dc:creator><![CDATA[Rafał Kuć]]></dc:creator>
		<pubDate>Wed, 14 Jul 2010 07:39:55 +0000</pubDate>
				<category><![CDATA[Solr]]></category>
		<category><![CDATA[dismax]]></category>
		<category><![CDATA[edismax]]></category>
		<category><![CDATA[fraza]]></category>
		<category><![CDATA[parser]]></category>
		<category><![CDATA[phrase]]></category>
		<category><![CDATA[phrase query]]></category>
		<category><![CDATA[solr]]></category>
		<category><![CDATA[standard]]></category>
		<guid isPermaLink="false">http://sematext.solr.pl/?p=10</guid>

					<description><![CDATA[W większości wdrożeń Lucene/Solr z którymi miałem do czynienia prędzej, czy później pojawiał się problem tuningu jakości wyników wyszukiwania. Jednym z prostszych sposób zwiększenia zadowolenia użytkowników z wyników wyszukiwania, a tym samym zadowolenia nas samych i naszych pracodawców jest premiowanie]]></description>
										<content:encoded><![CDATA[<p>W większości wdrożeń Lucene/Solr z którymi miałem do czynienia prędzej, czy później pojawiał się problem tuningu jakości wyników wyszukiwania. Jednym z prostszych sposób zwiększenia zadowolenia użytkowników z wyników wyszukiwania, a tym samym zadowolenia nas samych i naszych pracodawców jest premiowanie fraz. Mając do wyboru trzy najpopularniejsze parsery zapytań oraz szereg parametrów wpływających na ich zachowanie postanowiłem sprawdzić, jak radzi sobie Solr z&nbsp; premiowaniem fraz znalezionych w dokumentach na etapie wyszukiwania oraz jaki wpływ mają te funkcjonalności na wydajność.</p>
<p><span id="more-10"></span></p>
<p>W obecnym trunk`u Solr, mamy dostępne trzy parsery zapytań:</p>
<ul>
<li>Standard Solr Query Parser, czyli domyślny parser dla Solr oparty na domyślnym parserze zapytań w Lucene</li>
<li>DisMax Query Parser</li>
<li>Extended DisMax Query Parser</li>
</ul>
<p>Każdy z wyżej wymienionych parserów ma inne możliwości jeżeli chodzi premiowanie fraz na etapie zapytań. W tym wpisie nie zajmuję się kwestiami premiowania bliskości słów na etapie indeksowania, o tym napiszę innym razem. A więc wracając do parserów.</p>
<p><strong>Standard Solr Query Parser<br />
</strong></p>
<p>Parser oparty o standardowy parser zapytań Lucene oraz rozszerzający jego możliwości. Możliwości co do premiowania fraz nie są duże. Załóżmy, iż nasz system obsługuje dużą księgarnię internetową, gdzie użytkownicy są mogą oceniać książki, zostawiać komentarze, czy dyskutować na temat książek na forach, a my indeksujemy wszystkie te dane oraz później prezentujemy je w ramach wyników wyszukiwania. Możemy założyć także, iż użytkownik wpisując w pole wyszukiwania &#8222;Java wzorce projektowe&#8221; szuka książek dotyczących wzorców projektowych z przykładami w języku Java i takie książki chcemy mu pokazać. Żaden problem, zadajemy do Solr zapytanie:</p>
<p><code>q=java+wzorce+projektowe</code></p>
<p>I otrzymujemy listę książek. Można spocząć na laurach i stwierdzić, że nasza wyszukiwarka zachowuje się świetnie i chcemy nic więcej poprawiać. Ja jednak dodałbym do tego zapytania kolejną część &#8211; premiowanie fraz, tak aby książki, które w przeszukiwanych polach posiadają odpowiednią frazę (czyli tak naprawdę w których słowa podane przez użytkownika występują obok siebie) były pozycjonowane na początku listy wyników. Zmodyfikowane zapytanie wyglądałoby tak:</p>
<p><code>q=java+wzorce+projektowe+OR+"java+wzorce+projektowe"^30</code></p>
<p>Dodając dodatkowy fragment (<em>+OR+&#8221;java+wzorce+projektowe&#8221;^30</em>) na pierwszych miejscach listy wyników wyszukiwania dostaniemy dokumenty, które nie tylko mają w polach wyszukiwania termy (wyrazy) podane przez użytkownika, ale w tych polach mają te termy umiejscowione obok siebie. A tak wygląda zapytanie wygenerowane do Lucene:
</p>
<pre class="brush:xml">&lt;str name="parsedquery"&gt;name:java name:wzorce name:projektowe PhraseQuery(name:"java wzorce projektowe"^30.0)&lt;/str&gt;
&lt;str name="parsedquery_toString"&gt;name:java name:wzorce name:projektowe name:"java wzorce projektowe"^30.0&lt;/str&gt;</pre>
<p>Wyniki wyszukiwania dla tak skonstruowanego zapytania przedstawiają się następująco:
</p>
<pre class="brush:xml">&lt;?xml version="1.0" encoding="UTF-8"?&gt;
&lt;response&gt;
&lt;lst name="responseHeader"&gt;
   &lt;int name="status"&gt;0&lt;/int&gt;
   &lt;int name="QTime"&gt;0&lt;/int&gt;
   &lt;lst name="params"&gt;
      &lt;str name="q"&gt;java wzorce projektowe OR "java wzorce projektowe"^30&lt;/str&gt;
      &lt;str name="fl"&gt;score,id,name&lt;/str&gt;
   &lt;/lst&gt;
&lt;/lst&gt;
&lt;result name="response" numFound="5" start="0" maxScore="1.2399161"&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;1.2399161&lt;/float&gt;
      &lt;str name="id"&gt;1&lt;/str&gt;
      &lt;str name="name"&gt;Java wzorce projektowe&lt;/str&gt;
   &lt;/doc&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;0.010219089&lt;/float&gt;
      &lt;str name="id"&gt;2&lt;/str&gt;
      &lt;str name="name"&gt;Wzorce projektowe java&lt;/str&gt;
   &lt;/doc&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;0.010219089&lt;/float&gt;
      &lt;str name="id"&gt;3&lt;/str&gt;
      &lt;str name="name"&gt;Wzorce java projektowe&lt;/str&gt;
   &lt;/doc&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;0.010219089&lt;/float&gt;
      &lt;str name="id"&gt;4&lt;/str&gt;
      &lt;str name="name"&gt;Projektowe wzorce java&lt;/str&gt;
   &lt;/doc&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;0.010219089&lt;/float&gt;
      &lt;str name="id"&gt;5&lt;/str&gt;
      &lt;str name="name"&gt;Projektowe java wzorce&lt;/str&gt;
   &lt;/doc&gt;
&lt;/result&gt;
&lt;/response&gt;</pre>
<p><strong>DisMax Query Parser</strong></p>
<p>Oprócz możliwości skonstruowania zapytania w taki sposób, jak opisany powyżej mamy możliwość wykorzystania parametru <strong>pf</strong> oraz modyfikacji jego zachowania za pomocą parametru <strong>ps</strong>. Parametr <strong>pf</strong> przekazuje informacje o tym, w jakich polach mają być identyfikowane frazy. Parametru tego używa się w sposób analogiczny do parametru <strong>qf</strong> określając listę pól w jakich mają być identyfikowane frazy. Dodatkowo, należy dla każdego z podanych pól określić odpowiednie wartości podbicia (<em>boost</em>), bo jak wiemy domyślnym podbiciem jest 1. Zapytanie z wykorzystaniem DisMax`a wyglądałoby następująco:<br />
<code>q=java+wzorce+projektowe&amp;defType=dismax&amp;qf=name&amp;pf=name^30&amp;ps=0</code>
</p>
<p style="text-align: left;">Natomiast zapytanie przekazane do Lucene wygląda w sposób następujący:</p>
<pre class="brush:xml">&lt;str name="parsedquery"&gt;+((DisjunctionMaxQuery((name:java)) DisjunctionMaxQuery((name:wzorce)) DisjunctionMaxQuery((name:projektowe)))~3) DisjunctionMaxQuery((name:"java wzorce projektowe"^30.0))&lt;/str&gt;
&lt;str name="parsedquery_toString"&gt;+(((name:java) (name:wzorce) (name:projektowe))~3) (name:"java wzorce projektowe"^30.0)&lt;/str&gt;</pre>
<p>Wyniki dla tak skonstruowanego zapytania przedstawiają się następująco:
</p>
<pre class="brush:xml">&lt;?xml version="1.0" encoding="UTF-8"?&gt;
&lt;response&gt;
&lt;lst name="responseHeader"&gt;
   &lt;int name="status"&gt;0&lt;/int&gt;
   &lt;int name="QTime"&gt;0&lt;/int&gt;
   &lt;lst name="params"&gt;
      &lt;str name="pf"&gt;name^30&lt;/str&gt;
      &lt;str name="fl"&gt;id,name,score&lt;/str&gt;
      &lt;str name="q"&gt;java wzorce projektowe&lt;/str&gt;
      &lt;str name="qf"&gt;name&lt;/str&gt;
      &lt;str name="defType"&gt;dismax&lt;/str&gt;
      &lt;str name="ps"&gt;0&lt;/str&gt;
   &lt;/lst&gt;
&lt;/lst&gt;
&lt;result name="response" numFound="5" start="0" maxScore="1.2399161"&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;1.2399161&lt;/float&gt;
      &lt;str name="id"&gt;1&lt;/str&gt;
      &lt;str name="name"&gt;Java wzorce projektowe&lt;/str&gt;
   &lt;/doc&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;0.013625451&lt;/float&gt;
      &lt;str name="id"&gt;2&lt;/str&gt;
      &lt;str name="name"&gt;Wzorce projektowe java&lt;/str&gt;
   &lt;/doc&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;0.013625451&lt;/float&gt;
      &lt;str name="id"&gt;3&lt;/str&gt;
      &lt;str name="name"&gt;Wzorce java projektowe&lt;/str&gt;
   &lt;/doc&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;0.013625451&lt;/float&gt;
      &lt;str name="id"&gt;4&lt;/str&gt;
      &lt;str name="name"&gt;Projektowe wzorce java&lt;/str&gt;
   &lt;/doc&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;0.013625451&lt;/float&gt;
      &lt;str name="id"&gt;5&lt;/str&gt;
      &lt;str name="name"&gt;Projektowe java wzorce&lt;/str&gt;
   &lt;/doc&gt;
&lt;/result&gt;
&lt;/response&gt;</pre>
<p>Warto zauważyć, iż kolejność wyników dla obu metod jest taka sama. Wynika to z tego, iż fraza została zidentyfikowana tylko w przypadku dokumentu z identyfikatorem 1. Wartość <em>score</em> dla tego dokumentu w obu przypadkach jest taka sama. Natomiast dokumenty znajdujące się na pozycjach od 2 do 4 mają w ramach danej metody dokładnie taką samą wartość <em>score</em> ze względu na zidentyfikowanie trzech termów i dokładnie taką samą długość pola <em>name</em>. Oczywiście istnieją różnice w wartości <em>score</em> tych dokumentów pomiędzy metodami, ale wynika to z różnic w składaniu zapytań.</p>
<p>Użyłem jednak współczynnika <strong>ps=0</strong> nie wspominając zupełnie o tym co oznacza i do czego służy. W przypadku korzystania z parametru <strong>pf</strong> (oraz pf2, ale o tym później) parametr <strong>ps</strong> oznacza tzw. <em>Phrase Slop</em>, czyli maksymalną odległość słów od siebie, aby tworzyły frazę. Na przykład <strong>ps=2</strong> będzie oznaczało, iż słowa mogą być maksymalnie dwie pozycje od siebie, aby tworzyły frazę. Należy jednak pamiętać, iż pomimo tego, że zarówno &#8222;Java przykładowe wzorce projektowe&#8221;, jak i &#8222;Java wzorce projektowe&#8221; stworzą frazę, jednak dokument o tytule &#8222;Java wzorce projektowe&#8221; będzie posiadał większy <em>score</em> pomimo ustawienia ps=2, ze względu na to, że termy położone są bliżej siebie.</p>
<p>Omówiliśmy już dwa przypadki premiowania fraz. Został nam ostatni opierający się na tzw. <em>Enhanced Term Proximity Boosting</em>, czyli metodzie premiowania nie tylko całych fraz, ale także ich poszczególnych części. Funkcjonalność tą oferuje jednak tylko i wyłączenie:</p>
<p><strong>Extended DisMax Query Parser</strong></p>
<p>Niestety bez skorzystania z trunk`a lub przeniesienia kawałka kodu nie ma możliwości skorzystania z eDisMax`a.</p>
<p>Zapytanie z eDisMax`em wyglądałoby następująco:</p>
<p><code>q=java+wzorce+projektowe&amp;defType=edismax&amp;qf=name&amp;pf2=name^30&amp;ps=0</code></p>
<p>Powyższe zapytanie, tworzy następujące zapytania do Lucene:
</p>
<pre class="brush:xml">&lt;str name="parsedquery"&gt;+(DisjunctionMaxQuery((name:java)) DisjunctionMaxQuery((name:wzorce)) DisjunctionMaxQuery((name:projektowe))) (DisjunctionMaxQuery((name:"java wzorce"^30.0)) DisjunctionMaxQuery((name:"wzorce projektowe"^30.0)))&lt;/str&gt;
&lt;str name="parsedquery_toString"&gt;+((name:java) (name:wzorce) (name:projektowe)) ((name:"java wzorce"^30.0) (name:"wzorce projektowe"^30.0))&lt;/str&gt;</pre>
<p>Jak widać oprócz standardowych DisjunctionMaxQuery charakterystycznych dla DisMax`a (a tym samym dla jego rozszerzonej wersji) które wyszukują poszczególne termy przekazane do parametru <strong>q</strong> stworzone zostały także dwa dodatkowe zapytania, które odpowiadają za funkcjonalność <em>enhanced term proximity boosting</em>. Polega ona na stworzeniu par ze słów występujących obok siebie. W opisywanym testowym przypadku stworzone zostały pary &#8222;java wzorce&#8221; oraz &#8222;wzorce projektowe&#8221;. Jak można się domyślić najbardziej znaczącymi dokumentami na liście wyników, będą dokumenty posiadające obie wygenerowane pary, kolejne dokument będą posiadały jedną z par, a kolejne nie będą posiadały żadnej. Na potwierdzenie przedstawiam wynik zadania powyższego zapytania do Solr:
</p>
<pre class="brush:xml">&lt;?xml version="1.0" encoding="UTF-8"?&gt;
&lt;response&gt;
&lt;lst name="responseHeader"&gt;
   &lt;int name="status"&gt;0&lt;/int&gt;
   &lt;int name="QTime"&gt;0&lt;/int&gt;
   &lt;lst name="params"&gt;
      &lt;str name="fl"&gt;id,name,score&lt;/str&gt;
      &lt;str name="q"&gt;java wzorce projektowe&lt;/str&gt;
      &lt;str name="qf"&gt;name&lt;/str&gt;
      &lt;str name="pf2"&gt;name^30&lt;/str&gt;
      &lt;str name="defType"&gt;edismax&lt;/str&gt;
      &lt;str name="ps"&gt;0&lt;/str&gt;
   &lt;/lst&gt;
&lt;/lst&gt;
&lt;result name="response" numFound="5" start="0" maxScore="1.1705827"&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;1.1705827&lt;/float&gt;
      &lt;str name="id"&gt;1&lt;/str&gt;
      &lt;str name="name"&gt;Java wzorce projektowe&lt;/str&gt;
   &lt;/doc&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;0.3034844&lt;/float&gt;
      &lt;str name="id"&gt;2&lt;/str&gt;
      &lt;str name="name"&gt;Wzorce projektowe java&lt;/str&gt;
   &lt;/doc&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;0.3034844&lt;/float&gt;
      &lt;str name="id"&gt;5&lt;/str&gt;
      &lt;str name="name"&gt;Projektowe java wzorce&lt;/str&gt;
   &lt;/doc&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;0.014451639&lt;/float&gt;
      &lt;str name="id"&gt;3&lt;/str&gt;
      &lt;str name="name"&gt;Wzorce java projektowe&lt;/str&gt;
   &lt;/doc&gt;
   &lt;doc&gt;
      &lt;float name="score"&gt;0.014451639&lt;/float&gt;
      &lt;str name="id"&gt;4&lt;/str&gt;
      &lt;str name="name"&gt;Projektowe wzorce java&lt;/str&gt;
   &lt;/doc&gt;
&lt;/result&gt;
&lt;/response&gt;</pre>
<p>Jak widać pierwszy dokument nie zmienił swojej pozycji i jest to dalej dokument o identyfikatorze 1. Natomiast na drugim i trzecim miejscu znajdują się dokumenty, które mają jedną z wygenerowanych przez parser par słów. Co za tym idzie dokumenty o identyfikatorach 2 i 5 mają tą samą wartość współczynnika <em>score</em>. Listę wyników zamykają dokumenty posiadające jedynie termy wpisane przez użytkownika. Na uwagę zasługuje również fakt, iż tak samo jak w przypadku pierwszych dwóch przykładów, tak i eDisMax wyliczył inną wartość maksymalną współczynnika <em>score</em> dla wynikowego zbioru dokumentów &#8211; spowodowane jest to dokładnie tym samym co w poprzednim przykładzie, a mianowicie zapytaniem wygenerowanym przez parser i złożonym do Lucene.</p>
<p><strong>Wydajność</strong></p>
<p>W każdym wypadku należy wziąć pod uwagę, jaki poszczególne funkcjonalności będą miały wpływ na wydajność aplikacji opartej o Solr. W tym celu postanowiłem zrobić prosty test wydajnościowy. Założenia testu są dość proste &#8211; indeksuję dane z wikipedii, i dla każdej z metod tworze zapytanie dla fraz składających się z dwóch do sześciu tokenów. Zapytania zadaje przy wyłączonych cache`ach, za każdym razem restartując Solr. Wynik to średnia arytmetyczna z 10 powtórzeń każdego z testów. Przed wynikami testu, kilka słów o samym indeksie:</p>
<ul>
<li>Ilość dokumentów w indeksie: 1.177.239</li>
<li>Ilość segmentów: 1</li>
<li>Ilość termów:&nbsp; 18.506.646</li>
<li>Ilość par term/dokument: 230.297.212</li>
<li>Ilość tokenów: 418.135.268</li>
<li>Wielkość indeksu: 4.6GB (zoptymalizowany)</li>
<li>Wersja Lucene wykorzystana do zbudowania: 4.0-dev 964000</li>
</ul>
<p>Frazy jakie wybierane były dla każdej iteracji testu:</p>
<ul>
<li>Iteracja I: &#8222;wielki piotr&#8221;</li>
<li>Iteracja II: &#8222;druga wojna światowa&#8221;</li>
<li>Iteracja III: &#8222;druga wojna światowa niemcy&#8221;</li>
<li>Iteracja IV: &#8222;czas zmian reformacja w polsce&#8221;</li>
<li>Iteracja V: &#8222;zmiana czasu letniego na czas zimowy&#8221;</li>
</ul>
<p>Same wyniki przedstawiają się następująco:</p>
[table “1” not found /]<br />

<p>Należy pamiętać, iż przedstawione wyniki dotyczą tylko i wyłącznie kwestii wydajnościowej i nie są sugestią której metody premiowania fraz należy używać. Wybór metody to kwestia wymagań i danego wdrożenia. Co do samych wyników, widać, iż zgodnie z przypuszczeniami najszybsza jest metoda druga, czyli wykorzystanie <em>term proximity boost</em> i parsera DisMax, który okazuje się szybszy zarówno od Standard Query Parsera, jak i od eDisMax`a.</p>
]]></content:encoded>
					
					<wfw:commentRss>https://solr.pl/2010/07/14/solr-i-phrasequery-czyli-rozne-sposoby-premiowania-fraz/feed/</wfw:commentRss>
			<slash:comments>2</slash:comments>
		
		
			</item>
	</channel>
</rss>
