微服务篇-深入了解 Elasticsearch DSL 查询和 RestClient 查询、数据聚合(Bucket 聚合、带条件聚合、Metric 聚合)

🔥博客主页: 【小扳_-CSDN博客】
❤感谢大家点赞👍收藏⭐评论✍

文章目录

        1.0 DSL 查询

        1.1 叶子查询

        1.1.1 全文检索查询

        1.1.2 精确查询

        1.2 复合查询

        1.2.1 bool 查询

        1.3 排序

        1.4 分页

        1.4.1 深度分页

        1.5 高亮

        1.5.1 实现高亮

        2.0 RestClient 查询

        2.1 叶子查询

        2.2 复合查询

        2.3 排序和分页

        2.4 高亮

        3.0 数据聚合

        3.1 DSL 实现聚合

        3.1.1 Bucket 聚合

        3.1.2 带条件聚合

        3.1.3 Metric 聚合

        3.2 RestClient 实现聚合


        1.0 DSL 查询

        导入了大量数据到 Elasticsearch 中,实现了数据的存储。不过查询数据时依然采用的是根

据 id 查询,而非模糊搜索。

        所以来研究下 Elasticsearch 的数据搜索功能。Elasticsearch 提供了基于 JSON 的 DSL

(Domain Specific Language)语句来定义查询条件,其 JavaAPI 就是在组织 DSL 条件。

举个例子:

        查询所有数据:

        以最简单的无条件查询为例,无条件查询的类型是:match_all,因此其查询语句如下:

GET /xbs/_search
{"query": {"match_all": {}}
}

查询结果:

        发现虽然是 match_all,但是响应结果中并不会包含索引库中的所有文档,而是仅有 10 条。

这是因为处于安全考虑,elasticsearch 设置了默认的查询页数。

Elasticsearch 的查询可以分为两大类:

        1)叶子查询(Leaf query clauses):一般是在特定的字段里查询特定值,属于简单查询,

很少单独使用。

        2)复合查询(Compound query clauses):以逻辑方式组合多个叶子查询或者更改叶子查

询的行为方式。

        1.1 叶子查询

        叶子查询的类型也可以做进一步细分,详情可以查看官方文档:Query and filter context | Elasticsearch Guide [7.12] | Elastic

这里列举一些常见的,例如:

        1)全文检索查询(Full Text Queries):利用分词器对用户输入搜索条件先分词,得到词

条,然后再利用倒排索引搜索词条。例如:

        match、multi_match

        2)精确查询(Term-level queries):不对用户输入搜索条件分词,根据字段内容精确值匹

配。但只能查找 keyword、数值、日期、boolean 类型的字段。例如:

        ids、term、range

        1.1.1 全文检索查询

        以全文检索中的 match 为例,语法如下:

GET /xbs/_search
{"query": {"match": {"msg": "天才"}}
}

执行结果:

        与 match 类似的还有 multi_match,区别在于可以同时对多个字段搜索,而且多个字段都要

满足,语法示例:

GET /{索引库名}/_search
{"query": {"multi_match": {"query": "搜索条件","fields": ["字段1", "字段2"]}}
}

        1.1.2 精确查询

        精确查询,英文是 Term-level query,顾名思义,词条级别的查询。也就是说不会对用户输

入的搜索条件再分词,而是作为一个词条,与搜索的字段内容精确值匹配。因此推荐查找

keyword、数值、日期、boolean 类型的字段。

        以 term 查询为例,其语法如下:

GET /xbs/_search
{"query": {"term": {"name": {"value": "唐唐"}}}
}

执行结果:

再来看下 range 查询,语法如下:

GET /xbs/_search
{"query": {"range": {"age": {"gte": 20,"lte": 30}}}
}

执行结果:

range 是范围查询,对于范围筛选的关键字有:

        1)gte:大于等于

        2)gt:大于

        3)lte:小于等于

        4)lt:小于

        1.2 复合查询

        复合查询大致可以分为两类:

        第一类:基于逻辑运算组合叶子查询,实现组合条件,例如:bool

        第二类:基于某种算法修改查询时的文档相关性算分,从而改变文档排名。例如:

        function_score、dis_max

其它复合查询及相关语法可以参考官方文档:Compound queries | Elasticsearch Guide [7.12] | Elastic

        1.2.1 bool 查询

        bool 查询,即布尔查询,就是利用逻辑运算来组合一个或多个查询子句的组合。bool 查询支

持的逻辑运算有:

        1)must:必须匹配每个子查询,类似“与”。

        2)should:选择性匹配子查询,类似“或”。

        3)must_not:必须不匹配,不参与算分,类似“非”。

        4)filter:必须匹配,不参与算分。

bool 查询的语法如下:

GET /xbs/_search
{"query": {"bool": {"must": [{"match": {"msg": "天才"}}],"filter": [{"range": {"age": {"gte": 20,"lte": 30}}}]}}
}

执行结果:

        出于性能考虑,与搜索关键字无关的查询尽量采用 must_not 或 filter 逻辑运算,避免参与相

关性算分。

        1.3 排序

        elasticsearch 默认是根据相关度算分(_score)来排序,但是也支持自定义方式对搜索结果

排序。不过分词字段无法排序,能参与排序字段类型有:keyword 类型、数值类型、地理坐标类

型、日期类型等。

语法说明:

GET /indexName/_search
{"query": {"match_all": {}},"sort": [{"排序字段": {"order": "排序方式asc和desc"}}]
}

举个例子:

GET /xbs/_search
{"query": {"match_all": {}},"sort": [{"age": {"order": "desc"}}]
}

执行结果:

        1.4 分页

        elasticsearch 默认情况下只返回 top10 的数据。而如果要查询更多数据就需要修改分页参数

了。

        elasticsearch 中通过修改 from、size 参数来控制要返回的分页结果:

        1)from:从第几个文档开始

        2)size:总共查询几个文档

        简单来说,类似于 mysql 中的 limit ?, ?

语法如下:

GET /items/_search
{"query": {"match_all": {}},"from": 0, // 分页开始的位置,默认为0"size": 10,  // 每页文档数量,默认10"sort": [{"price": {"order": "desc"}}]
}

举个例子:

GET /xbs/_search
{"query": {"match_all": {}},"from": 0,"size": 2
}

执行结果:

        1.4.1 深度分页

        elasticsearch 的数据一般会采用分片存储,也就是把一个索引中的数据分成 N 份,存储到不

同节点上。这种存储方式比较有利于数据扩展,但给分页带来了一些麻烦。

        比如一个索引库中有 100000 条数据,分别存储到 4 个分片,每个分片 25000 条数据。现在

每页查询 10 条,查询第 99 页。那么分页查询的条件如下:

GET /items/_search
{"from": 990, // 从第990条开始查询"size": 10, // 每页查询10条"sort": [{"price": "asc"}]
}

从语句来分析,要查询第 990~1000 名的数据。

        从实现思路来分析,肯定是将所有数据排序,找出前 1000 名,截取其中的 990~1000 的部

分。但问题来了,我们如何才能找到所有数据中的前 1000 名呢?

        要知道每一片的数据都不一样,第 1 片上的第 900~1000,在另 1 个节点上并不一定依然是

900~1000 名。所以我们只能在每一个分片上都找出排名前 1000 的数据,然后汇总到一起,重新

排序,才能找出整个索引库中真正的前 1000 名,此时截取 990~1000 的数据即可。

如图:

        试想一下,假如我们现在要查询的是第 999 页数据呢,是不是要找第 9990~10000 的数据,

那岂不是需要把每个分片中的前 10000 名数据都查询出来,汇总在一起,在内存中排序?如果查

询的分页深度更深呢,需要一次检索的数据岂不是更多?

        由此可知,当查询分页深度较大时,汇总数据过多,对内存和 CPU 会产生非常大的压力。

        因此 elasticsearch 会禁止 from+ size 超过 10000 的请求。

针对深度分页,elasticsearch 提供了两种解决方案:

        1)search after:分页时需要排序,原理是从上一次的排序值开始,查询下一页数据。官方

推荐使用的方式。

        2)scroll:原理将排序后的文档 id 形成快照,保存下来,基于快照做分页。官方已经不推荐

使用。

小结:

        大多数情况下,采用普通分页就可以了。查看百度、京东等网站,会发现其分页都有限

制。例如百度最多支持 77 页,每页不足 20 条。京东最多 100 页,每页最多 60 条。

        因此,一般采用限制分页深度的方式即可,无需实现深度分页。

        1.5 高亮

        什么是高亮显示呢?

在百度,京东搜索时,关键字会变成红色,比较醒目,这叫高亮显示:

观察页面源码,会发现两件事情:

        1)高亮词条都被加了 <em> 标签

        2)<em> 标签都添加了红色样式

        css 样式肯定是前端实现页面的时候写好的,但是前端编写页面的时候是不知道页面要展示

什么数据的,不可能给数据加标签。而服务端实现搜索功能,要是有 elasticsearch 做分词搜索,

是知道哪些词条需要高亮的。

        因此词条的高亮标签肯定是由服务端提供数据的时候已经加上的。

因此实现高亮的思路就是:

        1)用户输入搜索关键字搜索数据。

        2)服务端根据搜索关键字到 elasticsearch 搜索,并给搜索结果中的关键字词条添加 html

标签。

        3)前端提前给约定好的 html 标签添加 CSS 样式。

        1.5.1 实现高亮

事实上 elasticsearch 已经提供了给搜索关键字加标签的语法,无需自己编码。

基本语法如下:

GET /{索引库名}/_search
{"query": {"match": {"搜索字段": "搜索关键字"}},"highlight": {"fields": {"高亮字段名称": {"pre_tags": "<em>","post_tags": "</em>"}}}
}

举个例子:

GET /xbs/_search
{"query": {"match": {"msg": "天才"}},"highlight": {"fields": {"msg": {"pre_tags": "<em>","post_tags": "</em>"}}}
}

执行结果:

        1)搜索必须有查询条件,而且是全文检索类型的查询条件,例如 match 。

        2)参与高亮的字段必须是 text 类型的字段

        3)默认情况下参与高亮的字段要与搜索字段一致,除非添加:required_field_match=false

        2.0 RestClient 查询

        文档的查询依然使用 RestHighLevelClient 对象,查询的基本步骤如下:

        1)创建 request 对象,这次是搜索,所以是 SearchRequest

        2)准备请求参数,也就是查询 DSL 对应的 JSON 参数

        3)发起请求

        4)解析响应,响应结果相对复杂,需要逐层解析

        match_all 查询为例,其 DSL 和 JavaAPI 的对比如图:

代码解读:

        第一步,创建 SearchRequest 对象,指定索引库名 

        第二步,利用 request.source() 构建 DSL,DSL 中可以包含查询、分页、排序、高亮等 

        query():代表查询条件,利用 QueryBuilders.matchAllQuery() 构建一个 match_all 查询

的 DSL

        第三步,利用 client.search() 发送请求,得到响应 这里关键的 API 有两个,一个是

request.source(),它构建的就是 DSL 中的完整 JSON 参数。其中包含了 query、sort、from、

size、highlight 等所有功能:

        另一个是 QueryBuilders,其中包含了各种叶子查询、复合查询等:

        2.1 叶子查询

        所有的查询条件都是由 QueryBuilders 来构建的,叶子查询也不例外。因此整套代码中变化

的部分仅仅是 query 条件构造的方式,其它不动。

        先建立连接:

代码如下:

public class ItemApplicationText {private RestHighLevelClient client;@BeforeEach//先初始化,连接到es服务端public void init() {this.client = new RestHighLevelClient(RestClient.builder(HttpHost.create("http://113.45.166.112:9200")));}@Testpublic void test() throws Exception {if (client != null){System.out.println("连接成功:" + client);}else {System.out.println("连接失败");}}@AfterEach//程序结束后,关闭连接public void close() throws Exception {this.client.close();}}

        1)match 查询:

具体代码如下:

    @Test//叶子查询public void leafQuery() throws IOException {//1.创建Request对象SearchRequest request = new SearchRequest("xbs");//2.准备DSLrequest.source().query(QueryBuilders.matchQuery("msg", "天才"));//3.发送请求SearchResponse response = client.search(request, RequestOptions.DEFAULT);//4,解析结果SearchHits hits = response.getHits();//获取到总记录数long count = hits.getTotalHits().value;SearchHit[] hitsHits = hits.getHits();for (SearchHit hitsHit : hitsHits) {String sourceAsString = hitsHit.getSourceAsString();System.out.println(sourceAsString);}System.out.println("总记录数:"+count);}

执行结果:

        2)range 查询:

代码如下:

    @Test//范围查询public void rangeQuery() throws IOException {//1.创建request对象SearchRequest request = new SearchRequest("xbs");//2.准备DSLrequest.source().query(QueryBuilders.rangeQuery("age").gte(20).lte(30));//3.发送请求SearchResponse search = client.search(request, RequestOptions.DEFAULT);//4.解析结果SearchHits hits = search.getHits();//获取到总记录数long count = hits.getTotalHits().value;SearchHit[] hitsHits = hits.getHits();for (SearchHit hitsHit : hitsHits) {String sourceAsString = hitsHit.getSourceAsString();System.out.println(sourceAsString);}System.out.println("总记录数:"+count);}

执行结果:

        3)term 查询

代码如下:

    @Test//精确查询public void termQuery() throws IOException {//1.创建Request对象SearchRequest request = new SearchRequest("xbs");//2.准备DSLrequest.source().query(QueryBuilders.termQuery("name","唐唐"));//3,发送请求SearchResponse search = client.search(request, RequestOptions.DEFAULT);//4,解析结果SearchHits hits = search.getHits();//获取总数记录long value = hits.getTotalHits().value;//获取具体内容SearchHit[] hitsHits = hits.getHits();for (SearchHit hit : hitsHits) {String sourceAsString = hit.getSourceAsString();System.out.println(sourceAsString);}System.out.println("总记录数:"+value);}

执行结果:

        2.2 复合查询

        复合查询也是由 QueryBuilders 来构建,以 bool 查询为例,DSL 和 JavaAPI 的对比如图:

举个例子:

    @Test//复合查询public void compoundQuery() throws IOException {//1.创建request对象SearchRequest request = new SearchRequest("xbs");//2.准备DSLBoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery();boolQueryBuilder.must(QueryBuilders.matchQuery("msg","天才"));boolQueryBuilder.filter(QueryBuilders.rangeQuery("age").gte(20).lte(30));request.source().query(boolQueryBuilder);//3.发送请求SearchResponse search = client.search(request, RequestOptions.DEFAULT);//4.解析结果SearchHits hits = search.getHits();//获取记录总数long value = hits.getTotalHits().value;//获取具体内容SearchHit[] hitsHits = hits.getHits();for (SearchHit hit : hitsHits) {String sourceAsString = hit.getSourceAsString();User bean = JSONUtil.toBean(sourceAsString, User.class);System.out.println(bean);}System.out.println("总记录数:"+value);}

执行结果:

        2.3 排序和分页

        之前说过,requeset.source() 就是整个请求 JSON 参数,所以排序、分页都是基于这个来

设置,其 DSL 和 JavaAPI 的对比如下:

举个例子:

    @Test//分页查询public void pageQuery() throws IOException {//1.创建request对象SearchRequest request = new SearchRequest("xbs");//2.准备DSL//2.1查询所有文档request.source().query(QueryBuilders.matchAllQuery());//2.2分页查询,从索引0开始,查询一共2条request.source().from(0).size(2);//3.0发送请求SearchResponse search = client.search(request, RequestOptions.DEFAULT);//4.解析数据SearchHits hits = search.getHits();//获取数据总数long value = hits.getTotalHits().value;//获取具体内容SearchHit[] hitsHits = hits.getHits();for (SearchHit hitsHit : hitsHits) {String sourceAsString = hitsHit.getSourceAsString();User bean = JSONUtil.toBean(sourceAsString, User.class);System.out.println(bean);}System.out.println("总记录数:"+value);}

执行结果:

        总数据一共有三条,通过分页查询从第一页开始,一页中指的数量为两条,所以查询出来的

数据只有两条。

        2.4 高亮

        高亮查询与前面的查询有两点不同:

        1)条件同样是在 request.source() 中指定,只不过高亮条件要基于 HighlightBuilder 来构

造。

        2)高亮响应结果与搜索的文档结果不在一起,需要单独解析。

首先来看高亮条件构造,其 DSL 和 JavaAPI 的对比如图:

举个例子:

    @Test//高亮展示public void highlightQuery() throws IOException {//1.创建request对象SearchRequest request = new SearchRequest("xbs");//2.准备DSL//2.1 通过全文查询request.source().query(QueryBuilders.matchQuery("msg","天才"));//2.2 添加高亮request.source().highlighter(SearchSourceBuilder.highlight().field("msg").preTags("<em>").postTags("</em>"));//3.发送请求SearchResponse search = client.search(request, RequestOptions.DEFAULT);//4.解析结果SearchHits hits = search.getHits();//获取查询的总条数long value = hits.getTotalHits().value;//获取高亮数据SearchHit[] hitsHits = hits.getHits();for (SearchHit hit : hitsHits) {//原始数据,不带高亮展示String sourceAsString = hit.getSourceAsString();User u = JSONUtil.toBean(sourceAsString, User.class);System.out.println("用户对象:"+u);//接下来获取展示的高亮数据Map<String, HighlightField> highlightFields = hit.getHighlightFields();if (CollUtils.isNotEmpty(highlightFields)){HighlightField msg = highlightFields.get("msg");if (msg != null){Text[] fragments = msg.getFragments();//拼接String s = "";for (Text fragment : fragments) {s += fragment.string();}u.setMsg(s);}}//最后再来展示高亮数据System.out.println("添加了高亮的用户"+u);}}

执行结果:

        3.0 数据聚合

        聚合(aggregations)可以让我们极其方便的实现对数据的统计、分析、运算。例如:

        什么品牌的手机最受欢迎?

        这些手机的平均价格、最高价格、最低价格?

        这些手机每月的销售情况如何?

        实现这些统计功能的比数据库的sql要方便的多,而且查询速度非常快,可以实现近实时搜索

效果。

官方文档:

Aggregations | Elasticsearch Guide [7.12] | Elastic

聚合常见的有三类:

1)桶(Bucket)聚合:用来对文档做分组 

        TermAggregation:按照文档字段值分组,例如按照品牌值分组、按照国家分组

        Date Histogram:按照日期阶梯分组,例如一周为一组,或者一月为一组

2)度量(Metric)聚合:用以计算一些值,比如:最大值、最小值、平均值等 

        Avg:求平均值

        Max:求最大值

        Min:求最小值

        Stats:同时求max、min、avg、sum等

3)管道(pipeline)聚合:其它聚合的结果为基础做进一步运算 

        3.1 DSL 实现聚合

        介绍三种聚合方式:Bucket 聚合、带条件聚合、Metric 聚合

        3.1.1 Bucket 聚合

基本语法如下:

GET /items/_search
{"size": 0, "aggs": {"category_agg": {"terms": {"field": "category","size": 20}}}
}

语法说明:

        size:设置 size 为 0,就是每页查 0 条,则结果中就不包含文档,只包含聚合。

        aggs:定义聚合。

        category_agg:聚合名称,自定义,但不能重复。

        terms:聚合的类型,按分类聚合,所以用 term 。

        field:参与聚合的字段名称。

        size:希望返回的聚合结果的最大数量。
 

举个例子:

GET /xbs/_search
{"size": 0,"aggs": {"category_gender": {"terms": {"field": "gender","size": 10}}}
}

        根据性别进行分类。

执行结果:

        3.1.2 带条件聚合

        默认情况下,Bucket 聚合是对索引库的所有文档做聚合。

        但真实场景下,用户会输入搜索条件,因此聚合必须是对搜索结果聚合。那么聚合必须添加

限定条件。

举个例子:

GET /xbs/_search
{"query": {"range": {"age": {"gte": 10,"lte": 30}}},"aggs": {"category_gender": {"terms": {"field": "gender","size": 10}}}
}

        首先对年龄做了限制,只查询 10 到 30 岁的用户,紧接着再进行根据性别进行分组。

执行结果:

        3.1.3 Metric 聚合

        在之前使用桶聚合将用户根据性别进行分组,那么分组之后的结果进行 Metric 聚合,简单来

说就是对用户获取最小值、最大值、平均值运算。

举个例子:

GET /xbs/_search
{"size": 0,"aggs": {"category_gender": {"terms": {"field": "gender","size": 10},"aggs": {"avg_age": {"avg": {"field": "age"}}}}}
}

        将用户根据性别分组之后,获取分组之后的平均年龄。

执行结果:

        也可以同时获取到最大值、最小最、平均值、总和

代码如下:

GET /xbs/_search
{"size": 0,"aggs": {"category_gender": {"terms": {"field": "gender","size": 10},"aggs": {"avg_age": {"stats": {"field": "age"}}}}}
}

        只需要将类型改为 stats 。

执行结果:

小结:

        1)aggs 代表聚合,与 query 同级,此时 query 的作用是?

        限定聚合的的文档范围

        2)聚合必须的三要素:

                聚合名称

                聚合类型

                聚合字段

        3)聚合可配置属性有:

                size:指定聚合结果数量

                order:指定聚合结果排序方式

                field:指定聚合字段

        3.2 RestClient 实现聚合

        可以看到在 DSL 中,aggs 聚合条件与 query 条件是同一级别,都属于查询 JSON 参数。因

此依然是利用 request.source() 方法来设置。

不过聚合条件的要利用 AggregationBuilders 这个工具类来构造。DSL 与 JavaAPI 的语法对比

如下:

聚合结果与搜索文档同一级别,因此需要单独获取和解析。具体解析语法如下:

举个例子:

    @Test//数据聚合public void aggregation() throws IOException {//1.创建request对象SearchRequest request = new SearchRequest("xbs");//2.准备DSL//2.1 查询全部数据request.source().query(QueryBuilders.matchAllQuery());//2.2 聚合查询request.source().aggregation(AggregationBuilders.terms("category_gender").field("gender").size(10));//3.发送请求SearchResponse search = client.search(request, RequestOptions.DEFAULT);//4.解析结果Aggregations aggregations = search.getAggregations();Terms categoryGender = aggregations.get("category_gender");List<? extends Terms.Bucket> buckets = categoryGender.getBuckets();for (Terms.Bucket bucket : buckets) {String keyAsString = bucket.getKeyAsString();System.out.println("keyAsString = " + keyAsString+":"+bucket.getDocCount());}}

执行结果:

        以上就是桶聚合最简单的写法。

        接下来 Metric 聚合:

代码如下:

    @Test//Metric 聚合public void metricAggregation() throws IOException {//1.创建request对象SearchRequest request = new SearchRequest("xbs");//2.准备DSLrequest.source().query(QueryBuilders.matchAllQuery());request.source().aggregation(AggregationBuilders.terms("category_gender").field("gender").size(10).subAggregation(AggregationBuilders.stats("age_stats").field("age")));//3.发送请求SearchResponse search = client.search(request, RequestOptions.DEFAULT);//4.解析数据Aggregations aggregations = search.getAggregations();Terms categoryGender = aggregations.get("category_gender");if (categoryGender != null){List<? extends Terms.Bucket> buckets = categoryGender.getBuckets();if (CollUtils.isEmpty(buckets)){throw new RuntimeException("没有数据");}for (Terms.Bucket bucket : buckets) {//获取分组字段String keyAsString = bucket.getKeyAsString();//获取统计数long docCount = bucket.getDocCount();//获取平均数Stats ageStats = bucket.getAggregations().get("age_stats");double avg = ageStats.getAvg();System.out.println("分组字段:"+keyAsString+"分组数量:"+docCount+"平均年龄:"+avg);}}}

执行结果:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/66339.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用Apache Mahout制作 推荐引擎

目录 创建工程 基本概念 关键概念 基于用户与基于项目的分析 计算相似度的方法 协同过滤 基于内容的过滤 混合方法 创建一个推荐引擎 图书评分数据集 加载数据 从文件加载数据 从数据库加载数据 内存数据库 协同过滤 基于用户的过滤 基于项目的过滤 添加自定…

javaEE-网络编程4.TCP回显服务器

目录 TCP流套接字编程 一.API介绍 ServerSocket类 构造方法&#xff1a; ​编辑方法&#xff1a; Socket类 构造方法&#xff1a; 方法&#xff1a; 二、TCP连接 三、通过TCP实现回显服务器 TCP服务端&#xff1a; 1.创建Socket对象 2.构造方法 3.start方法 TCP客…

数据库1-4讲

各种名词区分 内模式也叫物理模式、存储模式。 概念模式也叫全局模式、逻辑模式。 外模式也叫用户模式。 笛卡尔积&#xff1a;D1、D2、D3集合中任取一个的所有可能情况。 因此上述笛卡尔积的基数22312 关系模型的三个完整性&#xff1a; 实体完整性&#x…

UnityWebGl:打包成webgl后UGUI不显示文字(中文)问题

是由于unity默认使用的是Arial,导致打包成webgl时中文不显示 解决方案&#xff1a; 可在电脑C盘下&#xff0c;路径为C:\Windows\Fonts 找个中文简体的字体文件放到unity里面&#xff0c;格式必须为. ttf

朴素贝叶斯方法

一般来说训练时的一个实例有很多属性用一个<a1,a2,....,an>来表示一个数据&#xff0c;那么此时根据最大后验概率的计算公式可以得出&#xff1a; 其中&#xff0c; H 是目标值集合。 估计每个 P&#xff08;hi&#xff09;很容易&#xff0c; 只要计算每个目标值 hi出现…

Launcher3主页面加载显示流程分析

布局结构 抓取布局后&#xff0c;可以看到每个图标是一个DoubleShadowBubbleTextView&#xff0c;父布局是CellLayout、workspace。 我们可以在CellLayout添加子view打印出调用堆栈信息&#xff0c;可以整体上看页面加载显示流程。 主要类 Launcher.java&#xff1a;主界面&…

C++编程进阶:标准库中的算法库解析

文章目录 概述1. 非修改性序列操作2. 修改性序列操作3. 排序相关算法4. 二分查找算法5. 合并与集合操作6. 堆操作7. 最小/最大操作8. 数值算法(`<numeric>`头文件)概述 算法库总览:介绍了C++ 标准库提供的海量算法,这些算法作用于各类容器(如vector、list、set等)和…

Express 加 sqlite3 写一个简单博客

例图&#xff1a; 搭建 命令&#xff1a; 前提已装好node.js 开始创建项目结构 npm init -y package.json:{"name": "ex01","version": "1.0.0","main": "index.js","scripts": {"test": &q…

Linux双端口服务器:端口1的文件系统目录挂载到端口2

目录 一、服务器安装NFS服务并配置二、文件挂载三、持久化挂载总结为什么服务器配置多个端口 目前有一台服务器&#xff0c;不过他设置了两个SSH的端口&#xff0c;通过下面方法可以让这两个端口连接的主机能够共享同一个文件系统&#xff0c;原本这两个端口的文件系统是隔离的…

nginx-灰度发布策略(split_clients)

一. 简述&#xff1a; 基于客户端的灰度发布&#xff08;也称为蓝绿部署或金丝雀发布&#xff09;是一种逐步将新版本的服务或应用暴露给部分用户&#xff0c;以确保在出现问题时可以快速回滚并最小化影响的技术。对于 Nginx&#xff0c;可以通过配置和使用不同的模块来实现基于…

【NLP自然语言处理】Transformer模型的几大核心优势与应用前景

目录 &#x1f354; Transformer的并行计算 &#x1f354; Transformer架构的并行化过程 2.1 Transformer架构中Encoder的并行化 2.2 Transformer架构中Decoder的并行化 &#x1f354; Transformer的特征抽取能力 &#x1f354; 为什么说Transformer可以代替seq2seq? 4…

数据结构与算法之排序

9.1 排序的概念 1. 排序的定义 定义&#xff1a;排序是将表中的记录按关键字递增&#xff08;或递减&#xff09;有序排列的过程。说明&#xff1a;数据中可以存在相同关键字的记录。本章主要考虑递增排序。扩展&#xff1a;排序是数据处理中的基本操作之一&#xff0c;广泛应用…

《C++11》各种初始化方式的详细列举与对比

在 C 中&#xff0c;初始化对象的方式多种多样。随着 C 标准的演进&#xff0c;特别是 C11 的引入&#xff0c;初始化方式得到了显著的扩展和改进。本文将详细列举 C 中的各种初始化方式&#xff0c;并对它们进行对比&#xff0c;帮助开发者更好地理解和应用这些特性。 1. C98…

基于 Python Django 的西西家居全屋定制系统(源码+部署+文档)

博主介绍&#xff1a;✌程序员徐师兄、7年大厂程序员经历。全网粉丝12w、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;…

25考研|重邮软件工程复试攻略!

与计算机一样&#xff0c;重邮复试不合格也很有可能被淘汰&#xff01;快快认真准备起来&#xff01; 一、复试内容 1、笔试&#xff1a;分值100 2、综合面试&#xff1a;满分100 主要考核考生的综合素质和业务能力&#xff0c;由各招生学院具体组织实施&#xff0c;综合面试…

如何制作重识别数据集及如何解决all query identities do not appear in gallery的问题

如何制作重识别数据集 数据集制作链接 注意点&#xff1a; 按照上述方式制作完成数据集之后&#xff0c;分别建立3个文件夹&#xff0c;分别为train&#xff0c;test&#xff0c;query&#xff0c; 值得注意的是&#xff0c;query文件里的相机编号要进行修改&#xff0c;修改…

链地址法(哈希桶)

链地址法&#xff08;哈希桶&#xff09; 解决冲突的思路 开放定址法中所有的元素都放到哈希表⾥&#xff0c;链地址法中所有的数据不再直接存储在哈希表中&#xff0c;哈希表 中存储⼀个指针&#xff0c;没有数据映射这个位置时&#xff0c;这个指针为空&#xff0c;有多个数…

【C语言】可移植性陷阱与缺陷(七): 除法运算时发生的截断

在C语言编程中&#xff0c;除法运算可能会引发一些与可移植性相关的问题&#xff0c;特别是当涉及到整数除法时发生的截断&#xff08;truncation&#xff09;。不同平台对于整数除法的行为和处理方式可能会有所不同&#xff0c;这可能导致代码在不同编译器或硬件平台上的行为不…

了解RabbitMQ的工作原理

RabbitMQ是一个开源的消息代理系统&#xff0c;实现了高级消息队列协议&#xff08;AMQP&#xff09;。在现代分布式系统中&#xff0c;特别是在微服务架构中&#xff0c;RabbitMQ有广泛的应用。本文将详细介绍RabbitMQ的工作原理&#xff0c;并通过实践案例帮助读者理解和应用…

分布式搜索引擎之elasticsearch基本使用3

分布式搜索引擎之elasticsearch基本使用3 1.部署单点es 1.1.创建网络 因为我们还需要部署kibana容器&#xff0c;因此需要让es和kibana容器互联。这里先创建一个网络&#xff1a; docker network create es-net1.2.加载镜像 这里我们采用elasticsearch的7.12.1版本的镜像&…