微服务实用篇6-分布式搜索elasticsearch篇2

news2024/11/16 18:30:56

今天我们继续学习分布式搜索引擎elasticsearch,今天主要学习四个模块,分别为DSL查询文档,搜索结果处理,RestClient查询文档,还有最好演示一个旅游案例。下面开始今天的学习吧。

目录

一、DSL查询文档

1.1、DSL查询分类

1.2、DSL查询之复合查询function_score

1.3、DSL查询之复合查询boolean

二、ES搜索结果处理

2.1、搜索结果处理-排序

2.2、搜索结果处理-分页

2.3、搜索结果处理-高亮

三、RestClient查询文档

3.1、快速入门

3.2、RestClient查询文档-match、term、range、bool

3.3、RestClient搜索结果处理-排序和分页

3.4、RestClient搜索结果处理-高亮显示

四、旅游案例

4.1、搜索与分页

4.2、条件过滤

4.3、附近的酒店距离升序排序


一、DSL查询文档

1.1、DSL查询分类

常见的DSL查询分为如下几类:查询所有、全文查询、精确查询、地理查询、复合查询。


#查询所有
GET /hotel/_search
{
  "query":{ 
    "match_all": {}
  }
}

#全文检索-match单字段查询
GET /hotel/_search
{
  "query": {
    "match": {
      "name": "上海"
    }
  }
}

#全文检索-multi_match多字段查询
GET /hotel/_search
{
  "query": {
    "multi_match": {
      "query": "上海如家外滩"
      , "fields": ["brand","name","business"]
    }
  }
}

#精确查询-term查询
GET /hotel/_search
{
  "query": {
    "term": {
      "city.keyword": {
       "value": "上海"
       }
    }
  }
}

#精确查询-range查询
GET /hotel/_search
{
  "query": {
    "range": {
      "price": {
       "gt": 100
       , "lt": 10000
       }
    }
  }
}

#distance查询
GET /hotel/_search
{
  "query": {
    "geo_distance": {
      "distance": "30km",
      "location": "31, 121"
    }
  }
}

1.2、DSL查询之复合查询function_score

function_score的查询方式可以修改文档的相关性分,根据新的得分进行排序,主要包含四个部分,分别为:原始查询条件、过滤条件、算分函数、加权模式。

我们看下面的一个案例,给名为如家的酒店排名设计靠前一点,具体如下:

我们查询所有名称包含“外滩”的酒店名,然后使用算分函数对brand为如家的进行过滤,将其排名靠前一些,如下所示。

#function-score查询
GET /hotel/_search
{
  "query": {
    "function_score": {
      "query": {
        "match": {
          "name": "外滩"
        }
      }
      , "functions": [
        {
          "filter": {
            "term": {
              "brand.keyword": "如家"
            }
          },
          "weight": 12
        }
      ],
      "boost_mode": "sum"
    }
  }
}

1.3、DSL查询之复合查询boolean

boolean查询和function_score都是复合查询,但是boolean查询是不参与算分的,只返回是否满足,must是必须满足、should是满足其中之一、must_not是必须不满足、filter是过滤。

我们看下面的案例,首先是must必须包含如家,价格大于400取反,用filter过滤在坐标范围的10km内的酒店,具体如下:

 

二、ES搜索结果处理

2.1、搜索结果处理-排序

我们先看第一个案例,使用sort对酒店先按照评价进行降序,再按照价格进行升序排序。

 

#sort排序
GET /hotel/_search
{
  "query": {
    "match_all": {} 
    },
    "sort":[{
      "score": "desc"
    },
    {
      "price": "asc"
    }]
  }

下面演示按地理位置进行升序排序,具体如下:

#sort排序
GET /hotel/_search
{
  "query": {
    "match_all": {} 
    },
    "sort":[
      {
     "_geo_distance": {
       "location": {
         "lat": 22.507276,
         "lon": 113.931251
       } ,
       "order": "asc"
       , "unit": "km"
     }
    }
    ]
}

2.2、搜索结果处理-分页

ES通过修改from和size的参数来控制返回分页的结果,具体如下:按价格进行升序排序,然后返回10条数据,如下所示:

因为ES是分布式搜索,因此会面临深度分页的问题,他需要把所有的如下1000条数据都查询并聚合后重新排序,再从中截取10条文档数据,这样的话,可能会导致搜索页数过深,对内存和CPU的消耗也很大。

 ES分页一般有如下三种分页方式,目前用的比较多的还是from+size的搜索分页方式。

2.3、搜索结果处理-高亮

高亮:即对搜索的结果进行关键字突出显示,给高亮的字段加标签即可。

 将名称为如家的字段进行高亮显示,具体如下,标签可以省略,自动默认em标签:

GET /hotel/_search
{
  "query": {
    "match": {
      "name": "如家"
    }
  },
  "highlight": {
    "fields": {
      "name": {}
    }
  }
}

三、RestClient查询文档

3.1、快速入门

Match_all查询:主要包括两部分,DSL请求的发送,以及Json结果的解析,具体如下:


    @Test
    void testMatchAll() throws IOException {
        //1.创建request对象
        SearchRequest searchRequest = new SearchRequest("hotel") ;
        //2.准备DSL
        searchRequest.source().query(QueryBuilders.matchAllQuery()) ;
        //3.发送请求
        SearchResponse response = client.search(searchRequest, RequestOptions.DEFAULT) ;
        //4.解析响应
        SearchHits searchHits = response.getHits() ;
        //获取总条数
        long total = searchHits.getTotalHits().value ;
        System.out.println("一共搜索到了" + total + "条数据!");
        //获取文档数组
        SearchHit [] searchHits1 = searchHits.getHits() ;
        for(SearchHit searchHits2 : searchHits1){
            String json = searchHits2.getSourceAsString() ;
            //反序列化
            HotelDoc hotelDoc = JSON.parseObject(json, HotelDoc.class) ;
            System.out.println(hotelDoc);
        }
    }

3.2、RestClient查询文档-match、term、range、bool

下面演示mtach查询,查询name中包含“如家”的酒店名称,具体如下,其中我把解析响应单独抽取出来作为一个方法,直接调用方法进行解析即可,不用重复撰写。


    @Test
    void testMatch() throws IOException {
        //1.创建request对象
        SearchRequest searchRequest = new SearchRequest("hotel") ;
        //2.准备DSL
        searchRequest.source().query(QueryBuilders.matchQuery("name","如家")) ;
        //3.发送请求
        SearchResponse response = client.search(searchRequest, RequestOptions.DEFAULT) ;
        handleResponse(response);
    }

    private void handleResponse(SearchResponse response) {
        //4.解析响应
        SearchHits searchHits = response.getHits() ;
        //获取总条数
        long total = searchHits.getTotalHits().value ;
        System.out.println("一共搜索到了" + total + "条数据!");
        //获取文档数组
        SearchHit[] searchHits1 = searchHits.getHits() ;
        for(SearchHit searchHits2 : searchHits1){
            String json = searchHits2.getSourceAsString() ;
            //反序列化
            HotelDoc hotelDoc = JSON.parseObject(json, HotelDoc.class) ;
            System.out.println(hotelDoc);
        }
    }

下面演示bool查询,使用了must和filter,同时其中包含term精准查询和range查询,具体如下:

    @Test
    void testAll() throws IOException {
        //1.创建request对象
        SearchRequest searchRequest = new SearchRequest("hotel") ;
        //2.准备DSL
        BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery() ;
        //添加term
        boolQueryBuilder.must(QueryBuilders.termQuery("city.keyword","上海")) ;
        //添加range
        boolQueryBuilder.filter(QueryBuilders.rangeQuery("price").lte(500)) ;
        searchRequest.source().query(boolQueryBuilder) ;
        //3.发送请求
        SearchResponse response = client.search(searchRequest, RequestOptions.DEFAULT) ;
        handleResponse(response);
    }

3.3、RestClient搜索结果处理-排序和分页

定义页码和页面大小,每次将查询到的对象按照price升序排序,然后进行分页,每5个一页,具体如下所示:

    @Test
    void testPageAndSort() throws IOException {
        int page = 1 , pageSize = 5 ;

        //1.创建request对象
        SearchRequest searchRequest = new SearchRequest("hotel") ;
        //2.准备DSL
        searchRequest.source().query(QueryBuilders.matchAllQuery()) ;
        //排序
        searchRequest.source().sort("price", SortOrder.ASC) ;
        //分页
        searchRequest.source().from((page-1)*pageSize).size(pageSize) ;
        //3.发送请求
        SearchResponse response = client.search(searchRequest, RequestOptions.DEFAULT) ;
        //4.解析响应
        handleResponse(response);
    }

3.4、RestClient搜索结果处理-高亮显示

将查询的结果进行高亮显示,查询到name为如家的,进行高亮显示,对处理的结果进行覆盖,然后就可以形成高亮显示了。

 @Test
    void testHighlight() throws IOException {

        //1.创建request对象
        SearchRequest searchRequest = new SearchRequest("hotel") ;
        //2.准备DSL
        searchRequest.source().query(QueryBuilders.matchQuery("name","如家")) ;
        //高亮显示
        searchRequest.source().highlighter(new HighlightBuilder().field("name").requireFieldMatch(false)) ;
        //3.发送请求
        SearchResponse response = client.search(searchRequest, RequestOptions.DEFAULT) ;
        //4.解析响应
        handleResponse1(response);
    }

    private void handleResponse1(SearchResponse response) {
        SearchHits searchHits = response.getHits() ;
        //获取总条数
        long total = searchHits.getTotalHits().value ;
        System.out.println("一共搜索到了" + total + "条数据!");
        //获取文档数组
        SearchHit[] searchHits1 = searchHits.getHits() ;
        for(SearchHit searchHits2 : searchHits1){
            String json = searchHits2.getSourceAsString() ;
            //反序列化
            HotelDoc hotelDoc = JSON.parseObject(json, HotelDoc.class) ;
            //获取高亮的结果
            Map<String, HighlightField> highlightFields = searchHits2.getHighlightFields();
            if(!CollectionUtils.isEmpty(highlightFields)){
                HighlightField highlightField = highlightFields.get("name") ;
                if(highlightField != null){
                    String name = highlightField.getFragments()[0].string() ;
                    hotelDoc.setName(name);
                }
            }
            System.out.println(hotelDoc);
        }
    }

四、旅游案例

4.1、搜索与分页

实现旅游网站的酒店搜索功能,完成关键字搜索和分页,具体包括下面的三个步骤:

 定义实体类如下:


import lombok.Data;

@Data
public class RequestParams {
    private String key ;
    private Integer page ;
    private Integer size ;
    private String sortBy ;
    
}

import java.util.List;

@Data
public class PageResult {
    private Long total ;
    private List<HotelDoc> hotels ;

    public PageResult() {
    }

    public PageResult(Long total, List<HotelDoc> hotels) {
        this.total = total;
        this.hotels = hotels;
    }
}

定义表现层controller,接收前端页面请求,调用业务层方法实现查找和分页。

import cn.itcast.hotel.pojo.PageResult;
import cn.itcast.hotel.pojo.RequestParams;
import cn.itcast.hotel.service.IHotelService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
@RequestMapping("/hotel")
public class HotelController {

    @Autowired
    private IHotelService iHotelService ;

    @PostMapping("/list")
    public PageResult search(@RequestBody RequestParams params){
        return iHotelService.search(params) ;
    }

}

在业务层定义接口和实现类,如下,在实现类中完成查询和分页业务,将解析得到的结果进行封装后返回。其中注入的client在启动类中注解为bean,变成配置类交给spring管理。


import cn.itcast.hotel.mapper.HotelMapper;
import cn.itcast.hotel.pojo.Hotel;
import cn.itcast.hotel.pojo.HotelDoc;
import cn.itcast.hotel.pojo.PageResult;
import cn.itcast.hotel.pojo.RequestParams;
import cn.itcast.hotel.service.IHotelService;
import com.alibaba.fastjson.JSON;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
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.index.query.QueryBuilders;
import org.elasticsearch.search.SearchHit;
import org.elasticsearch.search.SearchHits;
import org.elasticsearch.search.sort.SortOrder;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import java.io.IOException;
import java.util.ArrayList;
import java.util.List;

@Service
public class HotelService extends ServiceImpl<HotelMapper, Hotel> implements IHotelService {

    @Autowired
    private RestHighLevelClient client ;

    @Override
    public PageResult search(RequestParams params) {

        try {
            //1.创建request对象
            SearchRequest searchRequest = new SearchRequest("hotel");
            //2.准备DSL
            String key = params.getKey();
            if (key == null || "".equals(key)) {
                searchRequest.source().query(QueryBuilders.matchAllQuery());
            } else {
                searchRequest.source().query(QueryBuilders.matchQuery("name", key));
            }
            //分页
            int page = params.getPage();
            int pageSize = params.getSize();
            searchRequest.source().from((page - 1) * pageSize).size(pageSize);
            //3.发送请求
            SearchResponse response = client.search(searchRequest, RequestOptions.DEFAULT);
            //4.解析响应
            return handleResponse(response);
        }catch (IOException e){
            throw new RuntimeException(e) ;
        }

    }

    private PageResult handleResponse(SearchResponse response) {
        //4.解析响应
        SearchHits searchHits = response.getHits() ;
        //获取总条数
        long total = searchHits.getTotalHits().value ;
        //获取文档数组
        SearchHit[] searchHits1 = searchHits.getHits() ;
        //遍历
        List<HotelDoc> list = new ArrayList<>() ;
        for(SearchHit searchHits2 : searchHits1){
            String json = searchHits2.getSourceAsString() ;
            //反序列化
            HotelDoc hotelDoc = JSON.parseObject(json, HotelDoc.class) ;
            list.add(hotelDoc) ;
        }
        return new PageResult(total, list) ;
    }
}

4.2、条件过滤

添加品牌、价格、星级、城市等条件过滤功能,具体实现步骤如下:

修改实体类,添加一些参数,如下:


import lombok.Data;

@Data
public class RequestParams {
    private String key ;
    private Integer page ;
    private Integer size ;
    private String sortBy ;
    private String city ;
    private String brand ;
    private String starName ;
    private Integer minPrice ;
    private Integer maxPrice ;

}

然后在业务层的实现方法中进行条件过滤即可,具体如下:


import cn.itcast.hotel.mapper.HotelMapper;
import cn.itcast.hotel.pojo.Hotel;
import cn.itcast.hotel.pojo.HotelDoc;
import cn.itcast.hotel.pojo.PageResult;
import cn.itcast.hotel.pojo.RequestParams;
import cn.itcast.hotel.service.IHotelService;
import com.alibaba.fastjson.JSON;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
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.index.query.BoolQueryBuilder;
import org.elasticsearch.index.query.QueryBuilders;
import org.elasticsearch.search.SearchHit;
import org.elasticsearch.search.SearchHits;
import org.elasticsearch.search.sort.SortOrder;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import java.io.IOException;
import java.util.ArrayList;
import java.util.List;

@Service
public class HotelService extends ServiceImpl<HotelMapper, Hotel> implements IHotelService {

    @Autowired
    private RestHighLevelClient client ;

    @Override
    public PageResult search(RequestParams params) {

        try {
            //1.创建request对象
            SearchRequest searchRequest = new SearchRequest("hotel");
            buildBasicQuery(params, searchRequest);
            //分页
            int page = params.getPage();
            int pageSize = params.getSize();
            searchRequest.source().from((page - 1) * pageSize).size(pageSize);
            //3.发送请求
            SearchResponse response = client.search(searchRequest, RequestOptions.DEFAULT);
            //4.解析响应
            return handleResponse(response);
        }catch (IOException e){
            throw new RuntimeException(e) ;
        }

    }

    private void buildBasicQuery(RequestParams params, SearchRequest searchRequest) {
        //2.关键字搜索
        BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery() ;
        String key = params.getKey();
        if (key == null || "".equals(key)) {
            boolQueryBuilder.must(QueryBuilders.matchAllQuery());
        } else {
            boolQueryBuilder.must(QueryBuilders.matchQuery("name", key));
        }
        //城市条件搜索
        if(params.getCity() != null && !"".equals(params.getCity())){
            boolQueryBuilder.filter(QueryBuilders.termQuery("city.keyword", params.getCity())) ;
        }
        //品牌条件搜索
        if(params.getBrand() != null && !"".equals(params.getBrand())){
            boolQueryBuilder.filter(QueryBuilders.termQuery("brand.keyword", params.getBrand())) ;
        }
        //星级条件搜索
        if(params.getStarName() != null && !"".equals(params.getStarName())){
            boolQueryBuilder.filter(QueryBuilders.termQuery("starName.keyword", params.getStarName())) ;
        }
        //价格条件搜索
        if(params.getMinPrice() != null && params.getMaxPrice() != null){
            boolQueryBuilder.filter(QueryBuilders.rangeQuery("price").
                    gte(params.getMinPrice()).lte(params.getMaxPrice())) ;
        }

        searchRequest.source().query(boolQueryBuilder) ;

    }

    private PageResult handleResponse(SearchResponse response) {
        //4.解析响应
        SearchHits searchHits = response.getHits() ;
        //获取总条数
        long total = searchHits.getTotalHits().value ;
        //获取文档数组
        SearchHit[] searchHits1 = searchHits.getHits() ;
        //遍历
        List<HotelDoc> list = new ArrayList<>() ;
        for(SearchHit searchHits2 : searchHits1){
            String json = searchHits2.getSourceAsString() ;
            //反序列化
            HotelDoc hotelDoc = JSON.parseObject(json, HotelDoc.class) ;
            list.add(hotelDoc) ;
        }
        return new PageResult(total, list) ;
    }
}

4.3、附近的酒店距离升序排序

获取前端的位置数据,根据计算得到的位置升序排序,找出最近的酒店位置,如下:

import cn.itcast.hotel.mapper.HotelMapper;
import cn.itcast.hotel.pojo.Hotel;
import cn.itcast.hotel.pojo.HotelDoc;
import cn.itcast.hotel.pojo.PageResult;
import cn.itcast.hotel.pojo.RequestParams;
import cn.itcast.hotel.service.IHotelService;
import com.alibaba.fastjson.JSON;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
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.common.geo.GeoPoint;
import org.elasticsearch.common.unit.DistanceUnit;
import org.elasticsearch.index.query.BoolQueryBuilder;
import org.elasticsearch.index.query.QueryBuilders;
import org.elasticsearch.search.SearchHit;
import org.elasticsearch.search.SearchHits;
import org.elasticsearch.search.sort.SortBuilder;
import org.elasticsearch.search.sort.SortBuilders;
import org.elasticsearch.search.sort.SortOrder;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import java.io.IOException;
import java.util.ArrayList;
import java.util.List;

@Service
public class HotelService extends ServiceImpl<HotelMapper, Hotel> implements IHotelService {

    @Autowired
    private RestHighLevelClient client ;

    @Override
    public PageResult search(RequestParams params) {

        try {
            //1.创建request对象
            SearchRequest searchRequest = new SearchRequest("hotel");
            buildBasicQuery(params, searchRequest);
            //分页
            int page = params.getPage();
            int pageSize = params.getSize();
            searchRequest.source().from((page - 1) * pageSize).size(pageSize);
            //排序
            String location = params.getLocation() ;
            if(location != null && !"".equals(location)){
                searchRequest.source().sort(SortBuilders.
                        geoDistanceSort("location", new GeoPoint(location)).
                        order(SortOrder.ASC).unit(DistanceUnit.KILOMETERS)) ;
            }
            //3.发送请求
            SearchResponse response = client.search(searchRequest, RequestOptions.DEFAULT);
            //4.解析响应
            return handleResponse(response);
        }catch (IOException e){
            throw new RuntimeException(e) ;
        }

    }

    private void buildBasicQuery(RequestParams params, SearchRequest searchRequest) {
        //2.关键字搜索
        BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery() ;
        String key = params.getKey();
        if (key == null || "".equals(key)) {
            boolQueryBuilder.must(QueryBuilders.matchAllQuery());
        } else {
            boolQueryBuilder.must(QueryBuilders.matchQuery("name", key));
        }
        //城市条件搜索
        if(params.getCity() != null && !"".equals(params.getCity())){
            boolQueryBuilder.filter(QueryBuilders.termQuery("city.keyword", params.getCity())) ;
        }
        //品牌条件搜索
        if(params.getBrand() != null && !"".equals(params.getBrand())){
            boolQueryBuilder.filter(QueryBuilders.termQuery("brand.keyword", params.getBrand())) ;
        }
        //星级条件搜索
        if(params.getStarName() != null && !"".equals(params.getStarName())){
            boolQueryBuilder.filter(QueryBuilders.termQuery("starName.keyword", params.getStarName())) ;
        }
        //价格条件搜索
        if(params.getMinPrice() != null && params.getMaxPrice() != null){
            boolQueryBuilder.filter(QueryBuilders.rangeQuery("price").
                    gte(params.getMinPrice()).lte(params.getMaxPrice())) ;
        }

        searchRequest.source().query(boolQueryBuilder) ;

    }

    private PageResult handleResponse(SearchResponse response) {
        //4.解析响应
        SearchHits searchHits = response.getHits() ;
        //获取总条数
        long total = searchHits.getTotalHits().value ;
        //获取文档数组
        SearchHit[] searchHits1 = searchHits.getHits() ;
        //遍历
        List<HotelDoc> list = new ArrayList<>() ;
        for(SearchHit searchHits2 : searchHits1){
            String json = searchHits2.getSourceAsString() ;
            //反序列化
            HotelDoc hotelDoc = JSON.parseObject(json, HotelDoc.class) ;
            //获取排序值
            Object [] objects = searchHits2.getSortValues() ;
            if(objects.length > 0){
                Object value = objects[0] ;
                hotelDoc.setDistance(value);
            }
            list.add(hotelDoc) ;
        }
        return new PageResult(total, list) ;
    }
}

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

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

相关文章

Hadoop学习----软件安装

Hadoop源码下载重新编译 软件下载&#xff1a;https://hadoop.apache.org/releases.html 建议是下载源码包。 源码包和官方编译安装包有什么不一样呢&#xff1f; 正常情况下&#xff0c;非生产环境直接使用官方编译安装包即可&#xff0c;但是官方提供的安装包不支持本地库。…

ANTLR4入门(二):图示说明eclipse安装Antlr4IDE插件的过程

如果你能正常通过Eclipse Market找到antlr4的插件并正常安装&#xff0c;可以忽略本文。 如果不能&#xff0c;那多半是因为网络问题导致安装Antlr4IDE插件时无法下载文件造成的。我就遇到了这个问题&#xff0c;无法下载的原因很复杂&#xff0c;我不想去深究了&#xff0c;我…

WPF/XAML关于x:key和x:name的区别,全面解读超详细

x:key和x:name的区别 x:Keyx:Name用于xaml Resources&#xff0c;ResourceDictionary用在ResourceDictionary以外任何地方使用key访问xaml指定对象使用name访问xaml对象标识资源创建和引用&#xff0c;存在于 ResourceDictionary 中的元素唯一标识对象元素&#xff0c;以便于从…

【Spring】SpringCloud

目录 一、SpringCloud 二、微服务介绍 1.系统架构演变 1.1 单体应用架构 1.2 垂直应用架构 1.3 分布式架构 1.4 SOA架构&#xff08;面向服务的架构&#xff09; 1.5 微服务架构&#xff08;服务的原子化拆分&#xff09; 2.微服务架构介绍 2.1 问题&#xff1a; 2.2…

与新手一起快速了解「什么是次世代」?

次世代&#xff08;英文&#xff1a;Next Generation&#xff09;&#xff0c;源自日本语&#xff0c;即下一个时代&#xff0c;未来的时代。与传统游戏相比&#xff0c;次世代游戏是把次世代游戏开发技术融入到现代游戏之中&#xff0c;通过增加模型的面数和贴图的数据量并使用…

JDK8系列之使用Function函数式接口

一、函数式接口是jdk8的新特性之一&#xff0c;函数式接口是只包含一个抽象方法声明的接口。按分类主要分为四大接口类型: Function、Consumer、Predicate、Supplier。 接口参数返回值说明Supplier<T>无T供给型&#xff0c;无参&#xff0c;返回一个指定泛型的对象Consu…

设计解谜游戏的30堂课

设计解谜游戏的30堂课 文章目录1.什么是Eureka Moment&#xff1f;2.谜题与幽默是同构的3.最大限度提高Sparkle4.避免无意义的谜题5.惊喜是Sparkle的重要源泉6.有趣的事实是惊喜的源泉7.尤里卡时刻并不是自豪8.不同解密者所寻求的解密体验是不尽相同的9.尤里卡是可以分享的10.创…

OVS-DPDK

要使用 ovs-dpdk&#xff0c;需要在node上构建 DPDK 并使用相应的 DPDK flag重新构建 ovs。 OVS-DPDK需要从源码编译&#xff0c;因为高度依赖内核等所在机器的环境&#xff0c;并需要配置很多参数以达到高性能。这意味着很难提供一个ovs-dpdk docker镜像来满足所有情况。OVS-D…

推荐系统学习笔记-基于图的模型

由来 基于图的模型&#xff08; graph-based model &#xff09;是推荐系统中的重要内容。其实&#xff0c;很多研究人员把基于邻域的模型也称为基于图的模型&#xff0c;因为可以把基于邻域的模型看做基于图的模型的简单形式。 在研究基于图的模型之前&#xff0c;首先需要将…

毫米波传感器原理介绍:角度估计

前边两篇博文从距离和速度两个维度进行了介绍&#xff0c;本篇将沿着第三个维度&#xff0c;角度进行分析&#xff0c;这也是毫米波传感器原理介绍的最后一篇理论基础。还是老规矩&#xff0c;我们先把公式列出来。 假设雷达两个接收天线之间的距离为d&#xff0c;θ 是物体相…

IS-95前向链路系统误码率matlab仿真,包括扩频调制,匹配滤波,RAKE接收

目录 1.算法描述 2.仿真效果预览 3.MATLAB核心程序 4.完整MATLAB 1.算法描述 前向链路指由基站发往移动台的无线通信链路&#xff0c;也称作下行链路。IS-95系统前向链路最多可以有64个同时传输的信道&#xff0c;它们是在PN序列上再采用正交的Walsh码进行区分的信道&#…

网上书店系统/书店管理系统的设计与实现

摘 要 随着信息技术和网络技术的飞速发展&#xff0c;人类已进入全新信息化时代&#xff0c;传统管理技术已无法高效&#xff0c;便捷地管理信息。为了迎合时代需求&#xff0c;优化管理效率&#xff0c;各种各样的管理系统应运而生&#xff0c;各行各业相继进入信息管理时代&…

C51 ——433M 射频信号控制喇叭

遥控按下后信号怎么被接受接 接收后的表现 &#xff1a; 厂商会告诉我们 这款告诉我们 接收模块对应针脚输出高电平 。 遥控控制喇叭 #include "reg52.h" sbit switcher P1^1; // 把继电器IN口 接到 P1.1 sbit D0_ON P1^2; // 把433M 射频信号接收器D0 口 接到P1.…

postman上传文件(multipart/form-data请求)

postman上传文件&#xff08;multipart/form-data请求&#xff09; 背景 网页的form表单中&#xff0c;如果存在上传文件的表单&#xff0c;则需要将form标签设置enctype"multipart/form-data"属性&#xff0c;意思是将Content-Type设置成multipart/form-data。 那…

今日小惊喜

今日限定小惊喜&#xff0c;一抬头突然发现有花开。

ESP32基础应用之lvgl显示中文

文章目录1 工程简介2 工程实现2.1 制作字库2.2 为字库自作分区表2.3 将字库移植到lvgl工程中2.4 将字库myFont.bin烧录到分区表中2.5 编写程序测试3 存在问题1 工程简介 该工程在《ESP32基础应用之LVGL基础》之上实现中文的显示。 参考文章 《ESP32 IDF LVGL8.0 flash 外部字…

CSC7720

CSC7720是一款用于5V2.1A开关电源的高效率同步整流控制IC。其具备较高的集成度&#xff0c;在有效的提升开关电源的转换效率的同时&#xff0c;减少了外围元器件的应用。CSC7720可用于DCM/QR开关电源系统。CSC7720内置45V的功率管&#xff0c;在系统中替代次级肖特基管,并提高整…

启动单文件组件项目及项目文件解释

启动项目文件&#xff1a;“package.json”&#xff1a; “package.json”&#xff1a;这个文件能记录当前项目中安装的所有模块&#xff0c;里面也有脚本&#xff0c;这个脚本可以快速启动我们的项目。 打开文件&#xff0c;可以看到“serve”&#xff0c;serve就是启动文件…

李沐精读论文:transformer 《Attention Is All You Need》 by Google

论文&#xff1a;Attention Is All You Need 视频&#xff1a;Transformer论文逐段精读【论文精读】_哔哩哔哩_bilibili 课程&#xff08;推荐先看这个&#xff09;&#xff1a;李宏毅机器学习&#xff1a;self-attention&#xff08;自注意力机制&#xff09;和transformer及其…

SpringBoot整合Shiro环境搭建

SpringBoot整合Shiro环境搭建导入 SpringBoot 和 Shiro 整合包的依赖&#xff1a; <!-- https://mvnrepository.com/artifact/org.apache.shiro/shiro-spring --> <dependency><groupId>org.apache.shiro</groupId><artifactId>shiro-spring<…