国产 无码 综合区,色欲AV无码国产永久播放,无码天堂亚洲国产AV,国产日韩欧美女同一区二区

ElasticSearch中實(shí)現(xiàn)模糊查詢效果(類似數(shù)據(jù)庫(kù)中l(wèi)ike功能)

這篇具有很好參考價(jià)值的文章主要介紹了ElasticSearch中實(shí)現(xiàn)模糊查詢效果(類似數(shù)據(jù)庫(kù)中l(wèi)ike功能)。希望對(duì)大家有所幫助。如果存在錯(cuò)誤或未考慮完全的地方,請(qǐng)大家不吝賜教,您也可以點(diǎn)擊"舉報(bào)違法"按鈕提交疑問(wèn)。

ElasticSearch中實(shí)現(xiàn)模糊查詢效果(類似數(shù)據(jù)庫(kù)中l(wèi)ike功能)

場(chǎng)景:

業(yè)務(wù)要求提供一個(gè)es查詢功能,實(shí)現(xiàn)類似模糊查詢效果,并且命中字段顯示紅色。舉例說(shuō)明:

es中字段內(nèi)容 輸入(即關(guān)鍵字) 是否輸出
你好,中國(guó),強(qiáng)大的祖國(guó) 中國(guó) 是(則‘中國(guó)’兩個(gè)字飄紅)
你好,中國(guó),強(qiáng)大的祖國(guó) 俄國(guó)
你好,中國(guó),強(qiáng)大的祖國(guó) 最大

實(shí)現(xiàn)方式:

這種實(shí)現(xiàn)方式主要是用es的query_string查詢方式,不過(guò)需要對(duì)輸入條件做區(qū)分處理才能實(shí)現(xiàn)模糊查詢效果。

首先,先復(fù)習(xí)一下query_string查詢方式的特點(diǎn):

{
	"query": {
		"query_string": {
			"query": "中國(guó)"
		}
	},
	"size": 10,
	"from": 0,
	"sort": []
}
pom依賴:

我用的elasticsearch版本是6.8.1,springboot版本是2.3.5.RELEASE,因此pom依賴的版本不對(duì),需要先排除再引入正確的包。

    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-data-elasticsearch</artifactId>
        <exclusions>
            <exclusion>
                <artifactId>spring-data-elasticsearch</artifactId>
                <groupId>org.springframework.data</groupId>
            </exclusion>
        </exclusions>
    </dependency>
    <dependency>
        <groupId>org.springframework.data</groupId>
        <artifactId>spring-data-elasticsearch</artifactId>
        <version>3.2.6.RELEASE</version>
    </dependency>
代碼:
import org.apache.commons.lang3.StringUtils;
import org.elasticsearch.action.search.SearchRequest;
import org.elasticsearch.action.search.SearchResponse;
import org.elasticsearch.client.RequestOptions;
import org.elasticsearch.client.RestHighLevelClient;
import org.elasticsearch.client.core.CountRequest;
import org.elasticsearch.client.core.CountResponse;
import org.elasticsearch.index.query.BoolQueryBuilder;
import org.elasticsearch.index.query.QueryBuilder;
import org.elasticsearch.index.query.QueryBuilders;
import org.elasticsearch.search.SearchHit;
import org.elasticsearch.search.builder.SearchSourceBuilder;
import org.elasticsearch.search.fetch.subphase.highlight.HighlightBuilder;
import org.elasticsearch.search.fetch.subphase.highlight.HighlightField;
import org.springframework.stereotype.Service;

import javax.annotation.Resource;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;

/**
 * 彈性搜索核心服務(wù)
 *
 * @author lukou
 * @date 2023/05/15
 */
@Service
public class ElasticSearchCoreService {

    /**
     * 常量和
     */
    private static final String CONSTANT_AND = " AND ";
    /**
     * 常量不
     */
    private static final String CONSTANT_NOT = " NOT ";
    /**
     * 常量或
     */
    private static final String CONSTANT_OR = " OR ";

    @Resource
    private RestHighLevelClient restHighLevelClient;

    /**
     * 通過(guò)query_string方法查詢統(tǒng)計(jì)
     *
     * @param index   指數(shù)
     * @param keyword 關(guān)鍵字
     * @return {@link CountResponse}
     * @throws IOException ioexception
     */
    public CountResponse queryCount(String index, String keyword) throws IOException {
        SearchSourceBuilder searchSourceBuilder = new SearchSourceBuilder();
        searchSourceBuilder.query(this.queryString(keyword));
        CountRequest request = new CountRequest(index);
        request.source(searchSourceBuilder);
        return restHighLevelClient.count(request, RequestOptions.DEFAULT);
    }

    /**
     * 通過(guò)query_string方法查詢搜索
     *
     * @param index   指數(shù)
     * @param keyword 關(guān)鍵字
     * @param from    起始位置
     * @param size    大小
     * @return {@link SearchResponse}
     * @throws IOException ioexception
     */
    public SearchResponse querySearch(String index, String keyword, int from, int size) throws IOException {
        SearchSourceBuilder searchSourceBuilder = new SearchSourceBuilder();
        searchSourceBuilder.query(this.queryString(keyword)).highlighter(redHighlightBuilder())
                .from(from)
                .size(size);
        SearchRequest request = new SearchRequest(index);
        // preference解決分頁(yè)數(shù)據(jù)不準(zhǔn)確的問(wèn)題(和分片有關(guān)系)
        request.source(searchSourceBuilder).preference(String.valueOf(keyword.hashCode()));
        return restHighLevelClient.search(request, RequestOptions.DEFAULT);
    }

    /**
     * 通過(guò)query_string方法查詢字符串
     * 按照邏輯表達(dá)式切割(AND OR NOT)
     *
     * <pre>
     * "123 AND abc"
     * "123 OR abc"
     * "123 NOT abc"
     * "NOT 123 NOT abc"
     * </pre>
     *
     * @param key 關(guān)鍵
     * @return {@link QueryBuilder}
     */
    public QueryBuilder queryString(String key) {
        //Bool查找
        BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery();
        if (StringUtils.contains(key, CONSTANT_AND)) {
            // 包含AND
            String[] keys = key.split(CONSTANT_AND);
            for (String s : keys) {
                boolQueryBuilder.must(QueryBuilders.queryStringQuery(convert(s)));
            }
            return boolQueryBuilder;
        } else if (StringUtils.contains(key, CONSTANT_OR)) {
            String[] keys = key.split(CONSTANT_OR);
            for (String s : keys) {
                boolQueryBuilder.should(QueryBuilders.queryStringQuery(convert(s)));
            }
            return boolQueryBuilder;
        } else if (StringUtils.contains(key, CONSTANT_NOT)) {
            List<String> keys = new ArrayList<>(Arrays.asList(key.split(CONSTANT_NOT)));
            if (keys.get(0).startsWith("NOT ")) {
                keys.set(0, keys.get(0).replaceAll("Not ", ""));
                for (String s : keys) {
                    boolQueryBuilder.mustNot(QueryBuilders.queryStringQuery(convert(s)));
                }
                return boolQueryBuilder;
            }
            String keyword = keys.remove(0);
            boolQueryBuilder.must(QueryBuilders.queryStringQuery(convert(keyword)));
            for (String s : keys) {
                boolQueryBuilder.mustNot(QueryBuilders.queryStringQuery(convert(s)));
            }
            return boolQueryBuilder;
        }
        boolQueryBuilder.must(QueryBuilders.queryStringQuery(convert(key)));
        return boolQueryBuilder;
    }

    /**
     * 轉(zhuǎn)換
     * 判斷是不是字母、數(shù)字、漢字
     *
     * @param key 關(guān)鍵
     * @return {@link String}
     */
    public String convert(String key) {
        //在執(zhí)行查詢時(shí),搜索的詞不會(huì)被分詞器分詞,而是直接以一個(gè)短語(yǔ)的形式查詢
        String res = "\"" + key + "\"";
        if (key.matches("^[A-Za-z0-9]*$")) {
            res = "*" + key + "*";
        }
        if (key.matches("^[\u4e00-\u9fa5][A-Za-z0-9]*$")) {
            res = key + "*";
        }
        return res;
    }

    /**
     * 紅色突出顯示生成器
     *
     * @return {@link HighlightBuilder}
     */
    public HighlightBuilder redHighlightBuilder() {
        HighlightBuilder highlightBuilder = new HighlightBuilder();
        //高亮的字段
        highlightBuilder.field("*");
        //是否多個(gè)字段都高亮
        highlightBuilder.requireFieldMatch(true);
        //前綴后綴
        highlightBuilder.preTags("<span style='color:red'>");
        highlightBuilder.postTags("</span>");
        return highlightBuilder;
    }

    /**
     * 構(gòu)建突出標(biāo)記
     *
     * @param hits 支安打
     * @return {@link List}<{@link Map}<{@link String}, {@link Object}>>
     */
    public List<Map<String, Object>> buildHighlightTags(SearchHit[] hits) {
        //解析結(jié)果
        List<Map<String, Object>> result = new LinkedList<>();
        for (SearchHit hit : hits) {
            //解析高亮的字段
            //獲取高亮字段
            Map<String, HighlightField> highlightFields = hit.getHighlightFields();
            Map<String, Object> sourceAsMap = hit.getSourceAsMap();
            for (String s : highlightFields.keySet()) {
                if (s != null) {
                    sourceAsMap.put(s, highlightFields.get(s).getFragments()[0].toString());//替換掉原來(lái)的內(nèi)容
                }
            }
            result.add(sourceAsMap);
        }
        return result;
    }
}
測(cè)試:
import org.elasticsearch.action.search.SearchResponse;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

import javax.annotation.Resource;
import java.io.IOException;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

@RestController
public class TestController {

    @Resource
    private ElasticSearchCoreService elasticSearchCoreService;

    @GetMapping("/test")
    public Map<String, Object> test(String index, String keyword) throws IOException {
        SearchResponse searchResponse = elasticSearchCoreService.querySearch(index, keyword, 0, 10);
        long total = searchResponse.getHits().getTotalHits();
        List<Map<String, Object>> mapList = elasticSearchCoreService.buildHighlightTags(searchResponse.getHits().getHits());
        Map<String, Object> result = new HashMap<>();
        result.put("total", total);
        result.put("data", mapList);
        return result;
    }
}
造數(shù)據(jù):

新建索引tmp_1以及插入5條數(shù)據(jù)

PUT http://localhost:9200/tmp_1

{
	"settings": {
		"number_of_shards": 1,
		"number_of_replicas": 0
	},
	"mappings": {
		"_doc": {
			"properties": {
				"@timestamp": {
					"type": "date"
				},
				"@version": {
					"type": "text",
					"fields": {
						"keyword": {
							"type": "keyword",
							"ignore_above": 256
						}
					}
				},
				"name": {
					"type": "keyword"
				},
				"sfz": {
					"type": "text"
				},
				"content": {
					"type": "text"
				},
				"address": {
					"type": "text"
				}
			}
		}
	}
}
POST http://localhost:9200/tmp_1/_doc

{
	"@version": "1",
	"@timestamp": "2020-06-19T09:06:26.446Z",
	"name": "唐伯虎",
	"sfz": "3212111990018989",
	"content": "當(dāng)你無(wú)話可說(shuō)的時(shí)候就別說(shuō)話,在你不知如何回答別人的話的時(shí)候就保持沉默,這就是生活中一個(gè)很好的策略",
	"address": "江蘇省南京市"
}
POST http://localhost:9200/tmp_1/_doc

{
	"@version": "1",
	"@timestamp": "2020-06-19T09:06:26.446Z",
	"name": "唐伯龍",
	"sfz": "3212111990018989",
	"content": "凡事都有偶然的湊巧,結(jié)果卻又如宿命的必然",
	"address": "江蘇省無(wú)錫市"
}
POST http://localhost:9200/tmp_1/_doc

{
	"@version": "1",
	"@timestamp": "2020-06-19T09:06:26.446Z",
	"name": "唐小虎",
	"sfz": "321211199709227654",
	"content": "一個(gè)人如果刻意逃避他所懼怕的東西,到頭來(lái)會(huì)發(fā)現(xiàn)自己只是抄了近路去見(jiàn)它",
	"address": "江蘇省蘇州市"
}
POST http://localhost:9200/tmp_1/_doc

{
	"@version": "1",
	"@timestamp": "2020-06-19T09:06:26.446Z",
	"name": "李小龍",
	"sfz": "1234211186709222348",
	"content": "蟲(chóng)子被踩后縮起來(lái),這是明智的,它借此減少重新被踩的概率。用道德的語(yǔ)言就叫:謙恭",
	"address": "江蘇省常州市"
}
POST http://localhost:9200/tmp_1/_doc

{
	"@version": "1",
	"@timestamp": "2020-06-19T09:06:26.446Z",
	"name": "李四",
	"sfz": "436754187709087623",
	"content": "你好,1234,你好5678",
	"address": "上海市"
}
調(diào)用接口:
http://localhost:8081/test?index=tmp_1&keyword=你好
# 響應(yīng)
{
    "total": 1,
    "data": [
        {
            "@timestamp": "2020-06-19T09:06:26.446Z",
            "address": "上海市",
            "sfz": "436754187709087623",
            "@version": "1",
            "name": "李四",
            "content": "<span style='color:red'>你</span><span style='color:red'>好</span>,1234,<span style='color:red'>你</span><span style='color:red'>好</span>5678"
        }
    ]
}
http://localhost:8081/test?index=tmp_1&keyword=唐小虎
# 響應(yīng)
{
    "total": 1,
    "data": [
        {
            "@timestamp": "2020-06-19T09:06:26.446Z",
            "address": "江蘇省蘇州市",
            "sfz": "321211199709227654",
            "@version": "1",
            "name": "<span style='color:red'>唐小虎</span>",
            "content": "一個(gè)人如果刻意逃避他所懼怕的東西,到頭來(lái)會(huì)發(fā)現(xiàn)自己只是抄了近路去見(jiàn)它"
        }
    ]
}

測(cè)試場(chǎng)景沒(méi)有全面覆蓋,如有錯(cuò)誤,歡迎指正。文章來(lái)源地址http://www.zghlxwxcb.cn/news/detail-532717.html

到了這里,關(guān)于ElasticSearch中實(shí)現(xiàn)模糊查詢效果(類似數(shù)據(jù)庫(kù)中l(wèi)ike功能)的文章就介紹完了。如果您還想了解更多內(nèi)容,請(qǐng)?jiān)谟疑辖撬阉鱐OY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!

本文來(lái)自互聯(lián)網(wǎng)用戶投稿,該文觀點(diǎn)僅代表作者本人,不代表本站立場(chǎng)。本站僅提供信息存儲(chǔ)空間服務(wù),不擁有所有權(quán),不承擔(dān)相關(guān)法律責(zé)任。如若轉(zhuǎn)載,請(qǐng)注明出處: 如若內(nèi)容造成侵權(quán)/違法違規(guī)/事實(shí)不符,請(qǐng)點(diǎn)擊違法舉報(bào)進(jìn)行投訴反饋,一經(jīng)查實(shí),立即刪除!

領(lǐng)支付寶紅包贊助服務(wù)器費(fèi)用

相關(guān)文章

  • JavaWeb04(登錄&綁值&模糊查詢&功能實(shí)現(xiàn)&連接數(shù)據(jù)庫(kù))

    JavaWeb04(登錄&綁值&模糊查詢&功能實(shí)現(xiàn)&連接數(shù)據(jù)庫(kù))

    目錄 一.實(shí)現(xiàn)登錄功能 ?2.2 制作簡(jiǎn)易驗(yàn)證碼 2.3 完成登錄驗(yàn)證 2.4 登錄實(shí)現(xiàn) ①連接字符串 private static final String URL=\\\"jdbc:oracle:thin:@localhost:1521:orcl\\\"; ②加載驅(qū)動(dòng)? OracleDriver private static final String URL=\\\"jdbc:oracle:thin:@localhost:1521:orcl\\\"; ? ?? 二.實(shí)現(xiàn)表格版的綁定數(shù)據(jù) 2.1 效果預(yù)覽 2.2?代

    2024年02月02日
    瀏覽(24)
  • 微信小程序 連接云數(shù)據(jù)庫(kù)(不使用云函數(shù))進(jìn)行 登錄、注冊(cè)、查詢(包括模糊查詢)快速實(shí)現(xiàn) 親測(cè)可用

    微信小程序 連接云數(shù)據(jù)庫(kù)(不使用云函數(shù))進(jìn)行 登錄、注冊(cè)、查詢(包括模糊查詢)快速實(shí)現(xiàn) 親測(cè)可用

    當(dāng)連接MySQL的時(shí)候總是出現(xiàn)各種各樣的小問(wèn)題,可以選用微信小程序自帶的云數(shù)據(jù)庫(kù) 注:測(cè)試號(hào)不能建立云數(shù)據(jù)庫(kù) 按圖中步驟來(lái) 第4步中,有兩種添加方式 第一種 第二種 注意: 如果你是粘貼復(fù)制的,他會(huì)報(bào)錯(cuò) 非數(shù)字字符\\\"無(wú)法使用外部字符串表示法位于第1行 。這時(shí),在第

    2023年04月20日
    瀏覽(26)
  • ES 實(shí)現(xiàn)數(shù)據(jù)庫(kù)or查詢效果

    ES :有兩種常用查詢?? ????????must? ?必須滿足查詢條件 ????????should 非必須滿足查詢條件 ?如果想實(shí)現(xiàn)類似與mysql中的or 查詢效果,必須使用should查詢。但是shuold 會(huì)查詢出不滿足條件的數(shù)據(jù) ,這必須加一個(gè)屬性?\\\"minimum_should_match\\\": \\\"1\\\" 必須滿足should條件中的一個(gè)查詢

    2024年02月11日
    瀏覽(20)
  • 如何玩mysql5.7實(shí)現(xiàn)分詞查詢,來(lái)實(shí)現(xiàn)類似ES的效果

    背景:在做海外主數(shù)據(jù)項(xiàng)目的時(shí)候,PM提出了一個(gè)需求,說(shuō)是類似于搜索的功能。但是需要實(shí)現(xiàn)根據(jù)輸入的字符串進(jìn)行相似度的查詢,并且計(jì)算出輸入的字符串與查出的字符串的相似度是多少。 思考:第一次聽(tīng)到需求的時(shí)候感覺(jué)還挺簡(jiǎn)單的,以為不就是個(gè)迷糊查詢嗎?但仔細(xì)

    2024年02月01日
    瀏覽(21)
  • Elasticsearch模糊查詢

    Elasticsearch模糊查詢 不計(jì)算相關(guān)度評(píng)分 前綴搜索匹配的是term,而不是field。即搜索倒排索引 前綴搜索的性能很差 前綴搜索沒(méi)有緩存 前綴搜索盡可能把前綴長(zhǎng)度設(shè)置的更長(zhǎng)命中索引越少,性能會(huì)好 要注意分詞器,如果分詞器分詞了會(huì)導(dǎo)致前綴匹配失敗.尤其是中文要設(shè)置中文分詞

    2024年02月12日
    瀏覽(23)
  • ElasticSearch中文分詞和模糊查詢

    ElasticSearch中文分詞和模糊查詢

    ? ? ? ? ElasticSearch是一個(gè)一個(gè)分布式的實(shí)時(shí)文檔存儲(chǔ),每一個(gè)字段都可以被索引與搜索,并且能支持PB級(jí)別的結(jié)構(gòu)化或者非結(jié)構(gòu)化數(shù)據(jù)。早期我們應(yīng)用的全局搜索是簡(jiǎn)單的SQL模糊查詢,為了分擔(dān)數(shù)據(jù)庫(kù)壓力所以用了ES,選擇他的原因除了以上幾點(diǎn)外,還有其提供的API方式簡(jiǎn)單

    2024年02月03日
    瀏覽(25)
  • Elasticsearch模糊查詢之Wildcard

    在?Elasticsearch?中,Wildcard 查詢通常用于在文本中查找匹配通配符模式的詞語(yǔ)。Wildcard 查詢是一種基于通配符的查詢,它使用單個(gè)字符(?)代表一個(gè)字符,使用星號(hào)(*)代表零個(gè)或多個(gè)字符。 Wildcard 查詢可用于對(duì)單個(gè)詞執(zhí)行模糊匹配,也可以用于對(duì)短語(yǔ)進(jìn)行模糊匹配。它可

    2024年02月14日
    瀏覽(20)
  • Springboot 整合ElasticSearch 常用的插入查詢,模糊查詢,范圍查詢

    Springboot 整合ElasticSearch 常用的插入查詢,模糊查詢,范圍查詢

    repository.deleteById(id); } catch (Exception ex) { ex.printStackTrace(); return false; } return true; } public Dog saveDog(Dog dog) { try { Dog save = repository.save(dog); System.out.println(“結(jié)果:”+save.toString()); return save; } catch (Exception ex) { ex.printStackTrace(); return null; } } public Boolean saveDogAll(List dogsList) { try { reposit

    2024年04月22日
    瀏覽(22)
  • elasticsearch7.17.3 實(shí)現(xiàn)類似mysql的like查詢

    前言:現(xiàn)在想要實(shí)現(xiàn)在elasticsearch中類似于mysql的like查詢方式,有下面幾種方法可以參考 建議 :wildcard方法是純純的like查詢方式平替,但是性能差,上百GB的數(shù)據(jù)量后就會(huì)很慢。根據(jù)自己業(yè)務(wù)量需求,前面兩種方式能解決的情況下盡量用前面兩種方式。前兩種方式可以修改索

    2024年02月08日
    瀏覽(26)
  • springboot整合elasticsearch實(shí)現(xiàn)類似于mysql的like查詢

    目錄 一、ES分頁(yè)查詢常用方式 二、引入es的依賴 三、es配置文件 四、es工具類 五、分頁(yè)查詢示例 1.from + size from表示從第幾行開(kāi)始,size表示查詢多少條文檔。from默認(rèn)為0,size默認(rèn)為10,最靈活的分頁(yè)方式。 2.scroll 不適合用來(lái)做實(shí)時(shí)搜索,而更適用于后臺(tái)批處理任務(wù),如日志導(dǎo)

    2023年04月09日
    瀏覽(19)

覺(jué)得文章有用就打賞一下文章作者

支付寶掃一掃打賞

博客贊助

微信掃一掃打賞

請(qǐng)作者喝杯咖啡吧~博客贊助

支付寶掃一掃領(lǐng)取紅包,優(yōu)惠每天領(lǐng)

二維碼1

領(lǐng)取紅包

二維碼2

領(lǐng)紅包