開發(fā)Web應(yīng)用時,你經(jīng)常要加上搜索功能。甚至還不知能要搜什么,就在草圖上畫了一個放大鏡。
搜索是項非常重要的功能,所以像elasticsearch和SOLR這樣的基于lucene的工具變得很流行。它們都很棒。但使用這些大規(guī)模“殺傷性”的搜索武器前,你可能需要來點輕量級的,但又足夠好的搜索工具。
所謂“足夠好”,我是指一個搜索引擎擁有下列的功能:
幸運的是PostgreSQL對這些功能全支持。
本文的目標(biāo)讀者是:
本文中我們將通過下面的表和數(shù)據(jù)說明PostgreSQL的全文搜索功能。
CREATE TABLE author( id SERIAL PRIMARY KEY, name TEXT NOT NULL);CREATE TABLE post( id SERIAL PRIMARY KEY, title TEXT NOT NULL, content TEXT NOT NULL, author_id INT NOT NULL references author(id) );CREATE TABLE tag( id SERIAL PRIMARY KEY, name TEXT NOT NULL );CREATE TABLE posts_tags( post_id INT NOT NULL references post(id), tag_id INT NOT NULL references tag(id) );INSERT INTO author (id, name) VALUES (1, 'Pete Graham'), (2, 'Rachid Belaid'), (3, 'Robert Berry'); INSERT INTO tag (id, name) VALUES (1, 'scifi'), (2, 'politics'), (3, 'science'); INSERT INTO post (id, title, content, author_id) VALUES (1, 'Endangered species', 'Pandas are an endangered species', 1 ), (2, 'Freedom of Speech', 'Freedom of speech is a necessary right missing in many countries', 2), (3, 'Star Wars vs Star Trek', 'Few words from a big fan', 3); INSERT INTO posts_tags (post_id, tag_id) VALUES (1, 3), (2, 2), (3, 1);
這是一個類博客的應(yīng)用。它有post表,帶有title和content字段。post通過外鍵關(guān)聯(lián)到author。post自身還有多個標(biāo)簽(tag)。
什么是全文搜索
首先,讓我們看一下定義:
在文本檢索中,全文搜索是指從全文數(shù)據(jù)庫中搜索計算機(jī)存儲的單個或多個文檔(document)的技術(shù)。全文搜索不同于基于元數(shù)據(jù)的搜索或根據(jù)數(shù)據(jù)庫中原始文本的搜索。
-- 維基百科
這個定義中引入了文檔的概念,這很重要。當(dāng)你搜索數(shù)據(jù)時,你在尋找你想要找到的有意義的實體,這些就是你的文檔。PostgreSQL的文檔中解釋地很好。
文檔是全文搜索系統(tǒng)中的搜索單元。比如,一篇雜質(zhì)文章或是一封郵件消息。
-- Postgres 文檔
這里的文檔可以跨多個表,代表為我們想要搜索的邏輯實體。
構(gòu)建我們的文檔(document)
上一節(jié),我們介紹了文檔的概念。文檔與表的模式無關(guān),而是與數(shù)據(jù)相關(guān),把字段聯(lián)合為一個有意義的實體。根據(jù)示例中的表的模式,我們的文檔(document)由這些組成:
根據(jù)這些要求產(chǎn)生文檔,SQL查詢應(yīng)該是這樣的:
SELECT post.title || ' ' || post.content || ' ' || author.name || ' ' || coalesce((string_agg(tag.name, ' ')), '') as document FROM post JOIN author ON author.id = post.author_id JOIN posts_tags ON posts_tags.post_id = posts_tags.tag_id JOIN tag ON tag.id = posts_tags.tag_id GROUP BY post.id, author.id; document -------------------------------------------------- Endangered species Pandas are an endangered species Pete Graham politics Freedom of Speech Freedom of speech is a necessary right missing in many countries Rachid Belaid politics Star Wars vs Star Trek Few words from a big fan Robert Berry politics(3 rows)
由于用post和author分組了,因為有多個tag關(guān)聯(lián)到一個post,我們使用string_agg()作聚合函數(shù)。即使author是外鍵并且一個post不能有多個author,也要求對author添加聚合函數(shù)或者把a(bǔ)uthor加到GROUP BY中。
我們還用了coalesce()。當(dāng)值可以是NULL時,使用coalesce()函數(shù)是個很好的辦法,否則字符串連接的結(jié)果將是NULL。
至此,我們的文檔只是一個長string,這沒什么用。我們需要用to_tsvector()把它轉(zhuǎn)換為正確的格式。
SELECT to_tsvector(post.title) || to_tsvector(post.content) || to_tsvector(author.name) || to_tsvector(coalesce((string_agg(tag.name, ' ')), '')) as documentFROM postJOIN author ON author.id = post.author_idJOIN posts_tags ON posts_tags.post_id = posts_tags.tag_idJOIN tag ON tag.id = posts_tags.tag_idGROUP BY post.id, author.id; document -------------------------------------------------- 'endang':1,6 'graham':9 'panda':3 'pete':8 'polit':10 'speci':2,7'belaid':16 'countri':14 'freedom':1,4 'mani':13 'miss':11 'necessari':9 'polit':17 'rachid':15 'right':10 'speech':3,6'berri':13 'big':10 'fan':11 'polit':14 'robert':12 'star':1,4 'trek':5 'vs':3 'war':2 'word':7(3 rows)
這個查詢將返回適于全文搜索的tsvector格式的文檔。讓我們嘗試把一個字符串轉(zhuǎn)換為一個tsvector。
SELECT to_tsvector('Try not to become a man of success, but rather try to become a man of value');這個查詢將返回下面的結(jié)果:
to_tsvector----------------------------------------------------------------------'becom':4,13 'man':6,15 'rather':10 'success':8 'tri':1,11 'valu':17(1 row)
發(fā)生了怪事。首先比原文的詞少了,一些詞也變了(try變成了tri),而且后面還有數(shù)字。怎么回事?
一個tsvector是一個標(biāo)準(zhǔn)詞位的有序列表(sorted list),標(biāo)準(zhǔn)詞位(distinct lexeme)就是說把同一單詞的各種變型體都被標(biāo)準(zhǔn)化相同的。
標(biāo)準(zhǔn)化過程幾乎總是把大寫字母換成小寫的,也經(jīng)常移除后綴(比如英語中的s,es和ing等)。這樣可以搜索同一個字的各種變體,而不是乏味地輸入所有可能的變體。
數(shù)字表示詞位在原始字符串中的位置,比如“man"出現(xiàn)在第6和15的位置上。你可以自己數(shù)數(shù)看。
Postgres中to_tesvetor的默認(rèn)配置的文本搜索是“英語“。它會忽略掉英語中的停用詞(stopword,譯注:也就是am is are a an等單詞)。
這解釋了為什么tsvetor的結(jié)果比原句子中的單詞少。后面我們會看到更多的語言和文本搜索配置。
查詢
我們知道了如何構(gòu)建一個文檔,但我們的目標(biāo)是搜索文檔。我們對tsvector搜索時可以使用@@操作符,使用說明見此處。看幾個查詢文檔的例子。
> select to_tsvector('If you can dream it, you can do it') @@ 'dream'; ?column?---------- t(1 row) > select to_tsvector('It''s kind of fun to do the impossible') @@ 'impossible'; ?column?---------- f(1 row)第二個查詢返回了假,因為我們需要構(gòu)建一個tsquery,使用@@操作符時,把字符串轉(zhuǎn)型(cast)成了tsquery。下面顯示了這種l轉(zhuǎn)型和使用to_tsquery()之間的差別。
SELECT 'impossible'::tsquery, to_tsquery('impossible'); tsquery | to_tsquery--------------+------------ 'impossible' | 'imposs'(1 row)但"dream"的詞位與它本身相同。
SELECT 'dream'::tsquery, to_tsquery('dream'); tsquery | to_tsquery--------------+------------ 'dream' | 'dream'(1 row)從現(xiàn)在開始我們使用to_tsquery查詢文檔。
SELECT to_tsvector('It''s kind of fun to do the impossible') @@ to_tsquery('impossible'); ?column?---------- t(1 row)tsquery存儲了要搜索的詞位,可以使用&(與)、|(或)和!(非)邏輯操作符??梢允褂脠A括號給操作符分組。
> SELECT to_tsvector('If the facts don't fit the theory, change the facts') @@ to_tsquery('! fact'); ?column?---------- f(1 row) > SELECT to_tsvector('If the facts don''t fit the theory, change the facts') @@ to_tsquery('theory & !fact'); ?column?---------- f(1 row) > SELECT to_tsvector('If the facts don''t fit the theory, change the facts.') @@ to_tsquery('fiction | theory'); ?column?---------- t(1 row)我們也可以使用:*來表達(dá)以某詞開始的查詢。
> SELECT to_tsvector('If the facts don''t fit the theory, change the facts.') @@ to_tsquery('theo:*'); ?column?---------- t(1 row)既然我們知道了怎樣使用全文搜索查詢了,我們回到開始的表模式,試著查詢文檔。
SELECT pid, p_titleFROM (SELECT post.id as pid, post.title as p_title, to_tsvector(post.title) || to_tsvector(post.content) || to_tsvector(author.name) || to_tsvector(coalesce(string_agg(tag.name, ' '))) as document FROM post JOIN author ON author.id = post.author_id JOIN posts_tags ON posts_tags.post_id = posts_tags.tag_id JOIN tag ON tag.id = posts_tags.tag_id GROUP BY post.id, author.id) p_search WHERE p_search.document @@ to_tsquery('Endangered & Species'); pid | p_title-----+-------------------- 1 | Endangered species(1 row)這個查詢將找到文檔中包含Endangered和Species或接近的詞。
語言支持
Postgres 內(nèi)置的文本搜索功能支持多種語言: 丹麥語,荷蘭語,英語,芬蘭語,法語,德語,匈牙利語,意大利語,挪威語,葡萄牙語,羅馬尼亞語,俄語,西班牙語,瑞典語,土耳其語。
SELECT to_tsvector('english', 'We are running'); to_tsvector------------- 'run':3(1 row)SELECT to_tsvector('french', 'We are running'); to_tsvector---------------------------- 'are':2 'running':3 'we':1(1 row)基于我們最初的模型,列名可以用來創(chuàng)建tsvector。 假設(shè)post表中包含不同語言的內(nèi)容,且它包含一列l(wèi)anguage。
ALTER TABLE post ADD language text NOT NULL DEFAULT('english');
為了使用language列,現(xiàn)在我們重新編譯文檔。
SELECT to_tsvector(post.language::regconfig, post.title) || to_tsvector(post.language::regconfig, post.content) || to_tsvector('simple', author.name) || to_tsvector('simple', coalesce((string_agg(tag.name, ' ')), '')) as documentFROM postJOIN author ON author.id = post.author_idJOIN posts_tags ON posts_tags.post_id = posts_tags.tag_idJOIN tag ON tag.id = posts_tags.tag_idGROUP BY post.id, author.id;如果缺少顯示的轉(zhuǎn)化符::regconfig,查詢時會產(chǎn)生一個錯誤:
ERROR: function to_tsvector(text, text) does not exist
regconfig是對象標(biāo)識符類型,它表示Postgres文本搜索配置項。:http://www.postgresql.org/docs/9.3/static/datatype-oid.html
現(xiàn)在,文檔的語義會使用post.language中正確的語言進(jìn)行編譯。
我們也使用simple,它也是Postgres提供的一個文本搜索配置項。simple并不忽略禁用詞表,它也不會試著去查找單詞的詞根。使用simple時,空格分割的每一組字符都是一個語義;對于數(shù)據(jù)來說,simple文本搜索配置項很實用,就像一個人的名字,我們也許不想查找名字的詞根。
SELECT to_tsvector('simple', 'We are running'); to_tsvector---------------------------- 'are':2 'running':3 'we':1(1 row)
重音字符
當(dāng)你建立一個搜索引擎支持多種語言時你也需要考慮重音問題。在許多語言中重音非常重要,可以改變這個詞的含義。Postgres附帶一個unaccent擴(kuò)展去調(diào)用 unaccentuate內(nèi)容是有用處的。
CREATE EXTENSION unaccent;SELECT unaccent('èéê?'); unaccent---------- eeee(1 row)
讓我們添加一些重音的你內(nèi)容到我們的post表中。
INSERT INTO post (id, title, content, author_id, language) VALUES (4, 'il était une fois', 'il était une fois un h?tel ...', 2,'french')
如果我們想要忽略重音在我們建立文檔時,之后我們可以簡單做到以下幾點:
SELECT to_tsvector(post.language, unaccent(post.title)) || to_tsvector(post.language, unaccent(post.content)) || to_tsvector('simple', unaccent(author.name)) || to_tsvector('simple', unaccent(coalesce(string_agg(tag.name, ' '))))JOIN author ON author.id = post.author_idJOIN posts_tags ON posts_tags.post_id = posts_tags.tag_idJOIN tag ON author.id = post.author_idGROUP BY p.id這樣工作的話,如果有更多錯誤的空間它就有點麻煩。 我們還可以建立一個新的文本搜索配置支持無重音的字符。
CREATE TEXT SEARCH CONFIGURATION fr ( COPY = french );ALTER TEXT SEARCH CONFIGURATION fr ALTER MAPPINGFOR hword, hword_part, word WITH unaccent, french_stem;
當(dāng)我們使用這個新的文本搜索配置,我們可以看到詞位
SELECT to_tsvector('french', 'il était une fois'); to_tsvector------------- 'fois':4(1 row)SELECT to_tsvector('fr', 'il était une fois'); to_tsvector-------------------- 'etait':2 'fois':4(1 row)這給了我們相同的結(jié)果,第一作為應(yīng)用unaccent并且從結(jié)果建立tsvector。
SELECT to_tsvector('french', unaccent('il était une fois')); to_tsvector-------------------- 'etait':2 'fois':4(1 row)詞位的數(shù)量是不同的,因為il était une在法國是一個無用詞。這是一個問題讓這些詞停止在我們的文件嗎?我不這么認(rèn)為etait不是一個真正的無用詞而是拼寫錯誤。
SELECT to_tsvector('fr', 'H?tel') @@ to_tsquery('hotels') as result; result-------- t(1 row)
如果我們?yōu)槊糠N語言創(chuàng)建一個無重音的搜索配置,這樣我們的post可以寫入并且我們保持這個值在post.language的中,然后我們可以保持以前的文檔查詢。
SELECT to_tsvector(post.language, post.title) || to_tsvector(post.language, post.content) || to_tsvector('simple', author.name) || to_tsvector('simple', coalesce(string_agg(tag.name, ' ')))JOIN author ON author.id = post.author_idJOIN posts_tags ON posts_tags.post_id = posts_tags.tag_idJOIN tag ON author.id = post.author_idGROUP BY p.id如果你需要為每種語言創(chuàng)建無重音的文本搜索配置由Postgres支持,然后你可以使用gist
我們當(dāng)前的文檔大小可能會增加,因為它可以包括無重音的無用詞但是我們并沒有關(guān)注重音字符查詢。這可能是有用的如有人用英語鍵盤搜索法語內(nèi)容。
歸類
當(dāng)你創(chuàng)建了一個你想要的搜索引擎用來搜索相關(guān)的結(jié)果(根據(jù)相關(guān)性歸類)的時候,歸類可以是基于許多因素的,它的文檔大致解釋了這些(歸類依據(jù))內(nèi)容。
歸類試圖處理特定的上下文搜索, 因此有許多個配對的時候,相關(guān)性最高的那個會被排在第一個位置。PostgreSQL提供了兩個預(yù)定義歸類函數(shù),它們考慮到了詞法解釋,接近度和結(jié)構(gòu)信息;他們考慮到了在上下文中的詞頻,如何接近上下文中的相同詞語,以及在文中的什么位置出現(xiàn)和其重要程度。
-- PostgreSQL documentation
通過PostgreSQL提供的一些函數(shù)得到我們想要的相關(guān)性結(jié)果,在我們的例子中我們將會使用他們中的2個:ts_rank() 和 setweight() 。
函數(shù)setweight允許我們通過tsvector函數(shù)給重要程度(權(quán))賦值;值可以是'A', 'B', 'C' 或者 'D'。
SELECT pid, p_titleFROM (SELECT post.id as pid, post.title as p_title, setweight(to_tsvector(post.language::regconfig, post.title), 'A') || setweight(to_tsvector(post.language::regconfig, post.content), 'B') || setweight(to_tsvector('simple', author.name), 'C') || setweight(to_tsvector('simple', coalesce(string_agg(tag.name, ' '))), 'B') as document FROM post JOIN author ON author.id = post.author_id JOIN posts_tags ON posts_tags.post_id = posts_tags.tag_id JOIN tag ON tag.id = posts_tags.tag_id GROUP BY post.id, author.id) p_searchWHERE p_search.document @@ to_tsquery('english', 'Endangered & Species')ORDER BY ts_rank(p_search.document, to_tsquery('english', 'Endangered & Species')) DESC;上面的查詢,我們在文中不同的欄里面賦了不同的權(quán)值。post.title的重要程度超過post.content和tag的總和。最不重要的是author.name。
這意味著如果我們搜索關(guān)鍵詞“Alice”,那么在題目中包含這個關(guān)鍵詞的文檔就會排在搜索結(jié)果的前面,在此之后是在內(nèi)容中包含這些關(guān)鍵詞的文檔,最后才是作者名字中包含這些關(guān)鍵詞的文檔.
基于對文檔各個部分的權(quán)重分配ts_rank()這個函數(shù)返回一個浮點數(shù),這個浮點數(shù)代表了文檔和查詢關(guān)鍵詞的相關(guān)性.
SELECT ts_rank(to_tsvector('This is an example of document'), to_tsquery('example | document')) as relevancy; relevancy----------- 0.0607927(1 row)SELECT ts_rank(to_tsvector('This is an example of document'), to_tsquery('example ')) as relevancy; relevancy----------- 0.0607927(1 row)SELECT ts_rank(to_tsvector('This is an example of document'), to_tsquery('example | unkown')) as relevancy; relevancy----------- 0.0303964(1 row)SELECT ts_rank(to_tsvector('This is an example of document'), to_tsquery('example & document')) as relevancy; relevancy----------- 0.0985009(1 row)SELECT ts_rank(to_tsvector('This is an example of document'), to_tsquery('example & unknown')) as relevancy; relevancy----------- 1e-20(1 row)但是, 相關(guān)性的概念是模糊的,而且是與特定的應(yīng)用相關(guān). 不同的應(yīng)用可能需要額外的信息來得到想要的排序結(jié)果,比如,文檔的修改時間. 內(nèi)建的排序功能如asts_rank只是個例子. 你可以寫出自己的排序函數(shù) 并且/或者 將得到的結(jié)果和其他因素混合來適應(yīng)你自己的特定需求.
這里說明一下, 如果我們想是新的文章比舊的文章更重要,可以講ts_rank函數(shù)的數(shù)值除以文檔的年齡+1(為防止被0除).
優(yōu)化與索引
將一個表中的搜索結(jié)果優(yōu)化為直線前進(jìn)的. PostgreSQL 支持基于索引的功能,因此你可以用tsvector()函數(shù)方便地創(chuàng)建GIN索引.
CREATE INDEX idx_fts_post ON post USING gin(setweight(to_tsvector(language, title),'A') || setweight(to_tsvector(language, content), 'B'));
GIN還是GiST索引? 這兩個索引會成為與他們相關(guān)的博文的主題. GiST會導(dǎo)出一個錯誤的匹配,之后需要一個額外的表行查找來驗證得到的匹配. 另一方面, GIN 可以更快地查找但是在創(chuàng)建時會更大更慢.
一個經(jīng)驗, GIN索引適合靜態(tài)的數(shù)據(jù)因為查找是迅速的. 對于動態(tài)數(shù)據(jù), GiST 可以更快的更新. 具體來說, GiST索引在動態(tài)數(shù)據(jù)上是好用的并且如果單獨的字(詞位)在100,000以下也是快速的,然而GIN 索引在處理100,000詞位以上時是更好的但是更新就要慢點了.
-- Postgres 文檔 : 第12章 全文搜索
在我們的例子中,我們選擇GIN。但是這個選擇不是一定的,你可以根據(jù)你自己的數(shù)據(jù)來作出決定。
我們的架構(gòu)例子中有一個問題; 分當(dāng)時分布在擁有不同權(quán)重的不同表中的. 為了更好的運行,通過觸發(fā)器和物化視圖使得數(shù)據(jù)非規(guī)范化是必要的.
我們并非總是需要非規(guī)范化并且有時也需要加入基于索引的功能,就像上面所做的那樣. 另外你可以通過postgres觸發(fā)器 功能tsvector_update_trigger(...)或者tsvector_update_trigger_column(...)實現(xiàn)相同表的數(shù)據(jù)的非規(guī)范化.參見Postgres文檔以得到更多詳細(xì)的信息.
在我們的應(yīng)用中在結(jié)果返回之前存在著一些可接受的延遲. 這是一個使用物化視圖將額外索引加載其中的好的情況.
CREATE MATERIALIZED VIEW search_index AS SELECT post.id, post.title, setweight(to_tsvector(post.language::regconfig, post.title), 'A') || setweight(to_tsvector(post.language::regconfig, post.content), 'B') || setweight(to_tsvector('simple', author.name), 'C') || setweight(to_tsvector('simple', coalesce(string_agg(tag.name, ' '))), 'A') as documentFROM postJOIN author ON author.id = post.author_idJOIN posts_tags ON posts_tags.post_id = posts_tags.tag_idJOIN tag ON tag.id = posts_tags.tag_idGROUP BY post.id, author.id之后重新索引搜索引擎就是定期運行REFRESH MATERIALIZED VIEW search_index這么簡單.
現(xiàn)在我們可以給物化視圖添加索引.
CREATE INDEX idx_fts_search ON search_index USING gin(document);
查詢也變得同樣簡單.
SELECT id as post_id, titleFROM search_indexWHERE document @@ to_tsquery('english', 'Endangered & Species')ORDER BY ts_rank(p_search.document, to_tsquery('english', 'Endangered & Species')) DESC;如果延遲變得無法忍受,你就應(yīng)該去研究一下使用觸發(fā)器的替代方法.
建立文檔存儲的方式并不唯一;這取決于你文檔的情況: 單表、多表,多國語言,數(shù)據(jù)量 ...
Thoughtbot.com 發(fā)表了文章"Implementing Multi-Table Full Text Search with Postgres in Rails" 我建議閱讀以下.
拼寫錯誤
PostgreSQL 提供了一個非常有用的擴(kuò)展程序pg_trgm。 相關(guān)文檔見pg_trgm doc。
CREATE EXTENSION pg_trgm;
pg_trgm支持N元語法如N==3。N元語法比較有用因為它可以查找相似的字符串,其實,這就是拼寫錯誤的定義 欧美日本高清视频| 国产精品一品二区三区的使用体验| 高清av影院| 伦理在线一区| 亚洲少妇诱惑| 天堂视频中文在线| 日韩一区二区视频在线| 黄色在线视频网站| 97在线免费公开视频| 日本高清不卡一区二区三区视频| 亚洲第一页综合| 亚洲高清在线观看| 亚洲视频图片小说| 国产成人艳妇aa视频在线| 日韩av电影免费在线| 亚洲国产精品美女| 免费无遮挡无码永久在线观看视频| 亚洲网站视频在线观看| www在线观看黄色| 欧美日韩精品在线| 人人妻人人澡人人爽久久av| 亚洲影视资源| 天堂а在线中文在线无限看推荐| www日韩在线| 日韩欧美在线观看一区二区| 中文字幕久久一区| 精品不卡视频| 99精品在线免费在线观看| 国产a视频精品免费观看| 国产国语**毛片高清视频| 欧美人与禽zoz0善交| 国产亚洲视频在线| 天天干在线播放| 美女日韩欧美| 成人综合电影| 国产精品成人国产| 亚洲一区二区在线播放相泽| 亚洲天堂av电影| аⅴ天堂中文在线网| 美国av在线播放| 老牛国内精品亚洲成av人片| 成人午夜在线播放| 55夜色66夜色国产精品视频| 久久网这里都是精品| 激情网站五月天| 日韩欧美一区二区免费| 99久久伊人精品影院| 欧美做受高潮1| 婷婷av一区二区三区| 国产成人精品亚洲日本在线观看| 欧美午夜精品久久久久久超碰| 美女又爽又黄| 国产精品pans私拍| 在线精品一区二区三区| 妺妺窝人体色www在线观看| 久久性色av| 免费视频亚洲| 人人妻人人玩人人澡人人爽| 亚洲综合色噜噜狠狠| 欧美三日本三级三级在线播放| 日韩免费一区二区| 91成人免费在线观看| 国产欧美一区二区三区在线老狼| 亚洲国产精品久久久久秋霞不卡| av成人app永久免费| caoporen人人| 四虎国产精品永久地址998| 视频一区视频二区欧美| 欧美激情国产日韩精品一区18| 一级黄色免费网站| 中国日韩欧美久久久久久久久| 久久久久久久久久久免费视频| 国产精品香蕉国产| 久久久久伊人| 好吊色免费视频| 日韩中文字幕麻豆| 无码av中文一区二区三区桃花岛| 性xx无遮挡| 日韩欧美久久| 7777精品伊人久久久大香线蕉| 一级毛片视频在线观看| 日韩精品 欧美| 成人中文字幕在线观看| 亚洲成a人片77777kkkk| 老司机在线精品视频| 久久麻豆一区二区| 成年人三级视频| 小小水蜜桃在线观看| 国产欧美日本| 粉嫩一区二区三区在线看| 日韩一区二区三区不卡视频| 五月婷婷深爱五月| 天堂蜜桃91精品| 传媒av在线| 超碰97人人在线| 中文字幕不卡一区| 日韩五码在线| 国产麻豆剧传媒精品国产av| 亚洲精品一卡二卡三卡四卡| 国产精品久久久久久婷婷天堂| 国产三级一区二区三区| 久久精品成人| 日本一本a高清免费不卡| 久久久美女视频| 国产精品嫩草影院精东| 成年人影院在线观看| 我不卡手机影院| 成人av黄色| 亚洲欧美日韩网| av资源中文色综合| 亚洲综合小说网| 欧美日韩一区二区免费在线观看| 成人性生交大合| 天天操天天插天天射| 国产成人av在线播放| 少妇久久久久久久| 女人黄色免费在线观看| 国产精品自产拍在线观看| gogo大尺度成人免费视频| 夜夜揉揉日日人人青青一国产精品| 欧美韩日精品| 国产成人啪午夜精品网站男同| 在线播放中文字幕一区| 日韩精品极品在线观看| av在线观看地址| 免费av播放| 国产精品日韩精品欧美在线| 亚洲精品免费电影| 国产精品视频在线观看| brazzers欧美精品| 欧美s码亚洲码精品m码| 青青草中文字幕| 丁香一区二区| 亚洲高清视频的网址| 亚洲成人三级在线| 69**夜色精品国产69乱| 最近最好的中文字幕2019免费| 亚洲视频小说图片| 久久久久久蜜桃一区二区| 欧日韩一区二区三区| 日韩精品久久久毛片一区二区| 国产精品视频精品| 黄色a级片在线观看| 超碰国产精品一区二页| 欧美最猛性xxxxxhd| 欧美va亚洲va香蕉在线| 日本不卡不卡| 91精品国产91久久久久麻豆 主演| 黄色片免费网址| 亚洲一区二区三区在线| 久久香蕉综合色一综合色88| 亚洲天堂福利av| 精品毛片久久久久久| 国产精品av久久久久久麻豆网| 亚洲视频免费在线| 久久精品成人一区二区三区蜜臀| 538在线一区二区精品国产| 国产色综合一区二区三区| 久久精品亚洲欧美日韩精品中文字幕| 亚洲精品亚洲人成人网在线播放| 伊人久久视频| 久草在线免费福利| 136福利视频导航| 三级a三级三级三级a十八发禁止| 99精品在线免费在线观看| 欧美1o一11sex性hdhd| 久久久久久久黄色| 国内揄拍国内精品| 日韩视频在线一区| aa一级黄色片| 久久久久欧美| 污污视频免费看| 成年女人的天堂在线| 国产精品一二三区| 成人午夜免费在线视频| 玖玖爱在线精品视频| 精品国精品国产| 白白色 亚洲乱淫| 99久久99久久精品国产片果冻| а中文在线天堂| 国产精品久久久高清免费| 高清国产午夜精品久久久久久| 亚洲欧美在线第一页| 深夜福利网站在线观看| 免费在线看黄网站| 四虎成人精品永久免费av| 国产中文字幕91| 久久精品国产亚洲AV无码麻豆| jzzjzzjzz亚洲成熟少妇| 影院在线观看全集免费观看| 一区二区冒白浆视频| 被下部羞羞漫画| 宅男噜噜噜66一区二区| 老司机精品免费视频| 密臀av一区二区三区| 免费大片黄在线| 偷窥国产亚洲免费视频| 日韩一区二区三区精品视频| 欧美视频一区在线| 国产传媒视频在线| 国产精品aaa| 成人av片在线观看| 黄色免费在线观看网站| 国产一区二区高清不卡| 国产精品久久久对白| 精品国产视频一区二区三区| 国产精品久久久久婷婷| 国产精品毛片va一区二区三区| 精品亚洲va在线va天堂资源站| 黄网站视频在线观看| 欧美亚一区二区| 婷婷综合一区| 国内外激情在线| 久久免费看少妇高潮v片特黄| 国产黄色大片在线观看| 一二三区在线观看| wwwav91com| 精品久久久无码人妻字幂| 2019中文字幕在线观看| 一区二区中文字| 国产情人综合久久777777| 日韩av电影在线网| 女同性互吃奶乳免费视频| 日本黄色精品| 国产95在线|亚洲| 中文字幕一区二区人妻痴汉电车| 日韩欧美视频在线| 激情久久一区二区| 国产午夜精品一区二区三区四区| 成人免费观看毛片| 激情都市一区二区| 亚洲一区三区| 国产欧美日韩成人| 春意影院在线| 一个人在线视频免费观看www| 欧美网站大全在线观看| 高清精品xnxxcom| 精品国产露脸精彩对白| 中文字幕亚洲欧美日韩2019| 欧洲精品乱码久久久久蜜桃| 99久久婷婷这里只有精品| 一区二区国产欧美| av免费在线网址| 另类激情视频| 中文字幕乱伦视频| 日韩久久免费电影| 天天射天天操天天干| 亚洲图片在线视频| 久久99久国产精品黄毛片色诱| 四虎影视免费看电影| 亚洲AV无码一区二区三区少妇| 曰皮视频在线播放免费的| 国产精品久久午夜| 欧美亚洲日本黄色| 欧美不卡一区| 亚洲最大最好的私人影剧院| 亚洲国产精久久久久久久| 精选一区二区三区四区五区| 亚洲国产精品网站| 欧美成人黄色网址| 久久亚洲成人| 欧美色倩网站大全免费| 星空大象在线观看免费播放| 祥仔av免费一区二区三区四区| 日韩欧美一区二区三区久久| 三上悠亚免费在线观看| 伊人网视频在线| www.日韩av.com| 国产黄色激情视频| 日本电影免费看| 欧美在线性爱视频| 中文字幕一区三区久久女搜查官| 欧美日韩dvd| 欧美日韩亚洲在线| 国产视频久久网| 久久黄色一级视频| 亚洲一区二区三区四区在线免费观看| 日韩av免费网址| 亚洲欧洲日韩国产| 色戒汤唯在线观看| 91视频最新网址| 国产91丝袜在线播放0| 精品美女久久久| 亚洲国产日韩精品| 韩国av一区二区三区在线观看| 精品国产午夜| 北条麻妃av毛片免费观看| 亚洲一区二区在线免费看| 精品国产一区三区| 亚洲天堂免费在线观看视频| 国产午夜精品久久久久久久| 国产视频福利一区| 激情综合色综合啪啪开心| 日韩免费中文专区| 欧美日韩国产限制| 国产刺激高潮av| 全球av集中精品导航福利| 尤物在线视频观看| 久久久久久免费网| 高清在线不卡av| 国产91富婆露脸刺激对白| 国产女人爽到高潮a毛片| 国产精品久久久久久久浪潮网站| 欧美体内she精视频| 国产不卡高清在线观看视频| 五月激情六月综合| 亚洲大香人伊一本线| 中文字幕中文字幕在线中一区高清| 国产精品一区不卡| 国产精品一级片| 亚洲一区久久久| 日本不卡高清视频一区| jizz免费视频| 你真棒插曲来救救我在线观看| 国产精品无码一区二区桃花视频| 国产亚洲成aⅴ人片在线观看| 久久人人97超碰国产公开结果| 日产精品99久久久久久| 女厕盗摄一区二区三区| 亚洲成人av在线电影| 手机在线免费观看毛片| 最近日韩中文字幕| 亚洲欧美一区二区不卡| 四虎成人精品在永久免费| 神马午夜在线观看| 国产视频久久久久久久| 久久福利资源站| 99热这里只有精品2|