【苍穹外卖】学习日志-day1

目录

nginx 反向代理介绍

nginx 的优势

提高访问速度

负载均衡

保证后端服务安全

高并发静态资源

Swagger 生成 API 文档

Swagger 的使用方式

导入knife4j的maven坐标

在配置类中加入knife4j相关配置

 设置静态资源映射

通过注解控制生成的接口文档

项目技术点

Token 模式

 MD5 加密

新增员工开发

DTO 设计模式

隔离线程 ThreadLocal

全局异常处理器

为什么需要全局异常处理器

代码测试

员工分页查询

代码测试


本章代码地址:苍穹外卖

 

nginx 反向代理介绍

        在没学习 nginx 之前,我们的项目都是前端直接发请求 tomcat 服务器的。如下图

tomcat 的作用:Tomcat 可以处理 HTTP 请求并将其传递给 Java 应用程序进行处理。

        学习了 nginx 后,我们更希望将用户的所有请求交给 nginx 反向代理,再转发给 tomcat 处理:

nginx 的优势

提高访问速度

        nginx 可以做缓存,如果我们请求的是同一个接口地址,则无需请求后端服务,直接在 nginx 把缓存数据响应给前端。

负载均衡

        nginx 可以把大量的请求按照指定的方式均衡的分配给集群中的每台服务器


举个例子:

        以百度为例,百度的后台肯定是不止一台服务器的,但我们在访问百度的时候,只需要输入百度的地址,就会被分配到一个服务器上去,以获得服务。而我们访问的是哪个服务器我们并不知道,我们只管访问 www.baidu.com,后面的事都会有相应的机制帮我们实现。

        要实现此类效果,即无论应用有多少实例,我们只需要访问一个地址就可以得到服务。就需要在客户端与服务端之间加上一层服务器 nginx 。

        这样客户端只管访问 nginx ,再由 nginx 服务器将请求代理到真正部署有实例的服务器上去即可。

保证后端服务安全

        我们真实的服务器不应该直接暴露到公网上去,否则更加容易泄露服务器的信息,也更加容易受到攻击。而使用 nginx 可以接收来自客户端的请求并将其转发到后端服务器。这样做的好处是可以隐藏服务器的真实 IP 地址,提供额外的安全层

高并发静态资源

        nginx 专注于处理静态资源,具有出色的性能和高并发处理能力。将 nginx 作为静态资源服务器可以提高系统的响应速度,并减轻 tomcat 的负担。

         而苍穹外卖也是通过 nginx 反向代理来访问我们后台的,nginx 文件:nginx反向代理

Swagger 生成 API 文档

        使用 Swagger 你只需要按照它的规范去定义接口及接口相关的信息,就可以做到生成接口文档,以及在线接口调试页面。这样我们就不用去 Postman 配置路径再来测试了。

Knife4j 是为 Java MVC 框架集成 Swagger 生成 API 文档的增强解决方案~

Swagger 的使用方式

导入knife4j的maven坐标

            <dependency><groupId>com.github.xiaoymin</groupId><artifactId>knife4j-spring-boot-starter</artifactId><version>3.0.2</version></dependency>

在配置类中加入knife4j相关配置

WebMvcConfiguration

  • Swagger 实例Bean是Docket,所以通过配置Docket实例来配置Swaggger
  • Docket 实例关联上 apiInfo
    /*** 通过knife4j生成接口文档*/@Beanpublic Docket docket() {ApiInfo apiInfo = new ApiInfoBuilder().title("苍穹外卖项目接口文档")  // 生成标题.version("2.0")              // 版本.description("苍穹外卖项目接口文档").build();Docket docket = new Docket(DocumentationType.SWAGGER_2).apiInfo(apiInfo)// 通过select()方法,去配置扫描接.select()// RequestHandlerSelectors 配置如何扫描接口.apis(RequestHandlerSelectors.basePackage("com.sky.controller")).paths(PathSelectors.any()).build();return docket;}

 设置静态资源映射

    /*** 设置静态资源映射*/protected void addResourceHandlers(ResourceHandlerRegistry registry) {log.info("开始设置静态资源映射...");registry.addResourceHandler("/doc.html").addResourceLocations("classpath:/META-INF/resources/");registry.addResourceHandler("/webjars/**").addResourceLocations("classpath:/META-INF/resources/webjars/");}

通过注解控制生成的接口文档

比如

    @PostMapping@ApiOperation("新增员工")public Result save(@RequestBody EmployeeDTO employeeDTO) {// ...}
注解说明
@Api用在类上,例如Controller,表示对类的说明
@ApiModel用在类上,例如 entity、DTO、VO
@ApiModelProperty用在属性上,描述属性信息
@ApiOperation用在方法上,例如Controller的方法,说明方法的用途、作用 

当程序运行时,我们访问 http://localhost:8080/doc.html 便可访问我们生成的 Swagger 接口文档。

 

项目技术点


Token 模式

        JWT 是 JSON Web Token 的缩写,即 JSON Web 令牌,JWT 是通过对JSON进行加密签名来实现授权验证的方案,就是登陆成功后将相关信息组成 json 对象,然后对这个对象进行某中方式的加密,返回给客户端,客户端在下次请求时带上这个 token,服务端再收到请求时校验 token 合法性,其实也是在校验请求的合法性,只有通过校验成功才能访问后台。 

        所以 JWT 常用于完成客户端的登入系统,以及拦截器;只有当用户输入的账号与密码与数据库中的匹配,系统就会生成 JWT 令牌。而只有这个令牌,后台的接口的拦截器才会放行。

 JWT 依赖:

    <dependency><groupId>com.auth0</groupId><artifactId>java-jwt</artifactId><version>0.9.1</version></dependency>

  JWT 令牌生成工具类:

public class JwtUtil {/*** 生成jwt* 使用Hs256算法, 私匙使用固定秘钥** @param secretKey jwt秘钥* @param ttlMillis jwt过期时间(毫秒)* @param claims    设置的信息* @return*/public static String createJWT(String secretKey, long ttlMillis, Map<String, Object> claims) {// 指定签名的时候使用的签名算法,也就是header那部分SignatureAlgorithm signatureAlgorithm = SignatureAlgorithm.HS256;// 生成JWT的时间long expMillis = System.currentTimeMillis() + ttlMillis;Date exp = new Date(expMillis);// 设置jwt的bodyJwtBuilder builder = Jwts.builder()// 如果有私有声明,一定要先设置这个自己创建的私有的声明,这个是给builder的claim赋值,一旦写在标准的声明赋值之后,就是覆盖了那些标准的声明的.setClaims(claims)// 设置签名使用的签名算法和签名使用的秘钥.signWith(signatureAlgorithm, secretKey.getBytes(StandardCharsets.UTF_8))// 设置过期时间.setExpiration(exp);return builder.compact();}/*** Token解密** @param secretKey jwt秘钥 此秘钥一定要保留好在服务端, 不能暴露出去, 否则sign就可以被伪造, 如果对接多个客户端建议改造成多个* @param token     加密后的token* @return*/public static Claims parseJWT(String secretKey, String token) {// 得到DefaultJwtParserClaims claims = Jwts.parser()// 设置签名的秘钥.setSigningKey(secretKey.getBytes(StandardCharsets.UTF_8))// 设置需要解析的jwt.parseClaimsJws(token).getBody();return claims;}}

拦截器

/*** jwt令牌校验的拦截器*/
@Component
@Slf4j
public class JwtTokenAdminInterceptor implements HandlerInterceptor {@Autowiredprivate JwtProperties jwtProperties;/*** 校验jwt*/public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {//判断当前拦截到的是Controller的方法还是其他资源if (!(handler instanceof HandlerMethod)) {//当前拦截到的不是动态方法,直接放行return true;}//1、从请求头中获取令牌String token = request.getHeader(jwtProperties.getAdminTokenName());//2、校验令牌try {log.info("jwt校验:{}", token);Claims claims = JwtUtil.parseJWT(jwtProperties.getAdminSecretKey(), token);Long empId = Long.valueOf(claims.get(JwtClaimsConstant.EMP_ID).toString());BaseContext.setCurrentId(empId);log.info("当前员工id:", empId);//3、通过,放行return true;} catch (Exception ex) {//4、不通过,响应401状态码response.setStatus(401);return false;}}

可以通过 yml 配置文件来定义 JWT 令牌的一些属性

sky:jwt:# 设置jwt签名加密时使用的秘钥admin-secret-key: itcast# 设置jwt过期时间admin-ttl: 7200000# 设置前端传递过来的令牌名称admin-token-name: token

 

 MD5 加密

        当我们需要保存某些密码信息以用于身份确认时,如果直接将密码信息以明码方式保存在数据库中,不使用任何保密措施,系统管理员就很容易能得到原来的密码信息,这些信息一旦泄露, 密码也很容易被破译。

        为了增加安全性,有必要对数据库中需要保密的信息进行加密,MD5 算法可以很好地解决这个问题,因为它可以将任意长度的输入串经过计算得到固定长度的输出,而且只有在明文相同的情况下,才能等到相同的密文,并且这个算法是不可逆,即便得到了加密以后的密文,也不可能通过解密算法反算出明文。

 spring 提供了一个工具类 DigestUtils,我们可以利用该工具类来对数据进行加密

依赖坐标

<dependency><groupId>commons-codec</groupId><artifactId>commons-codec</artifactId><version>1.15</version></dependency>

 员工登入代码

    public Employee login(EmployeeLoginDTO employeeLoginDTO) {String username = employeeLoginDTO.getUsername();String password = employeeLoginDTO.getPassword();//1、根据用户名查询数据库中的数据Employee employee = employeeMapper.getByUsername(username);//2、处理各种异常情况(用户名不存在、密码不对、账号被锁定)if (employee == null) {//账号不存在throw new AccountNotFoundException(MessageConstant.ACCOUNT_NOT_FOUND);}//密码比对// TODO 后期需要进行md5加密,然后再进行比对// 对输入密码进行加密password = DigestUtils.md5DigestAsHex(password.getBytes());log.info(password);// 输入密码与数据库密码做比较if (!password.equals(employee.getPassword())) {//密码错误throw new PasswordErrorException(MessageConstant.PASSWORD_ERROR);}if (employee.getStatus() == StatusConstant.DISABLE) {//账号被锁定throw new AccountLockedException(MessageConstant.ACCOUNT_LOCKED);}//3、返回实体对象return employee;}

        以上是将用户输入的密码通过 DigestUtils 工具类转化成密文再与数据中的密码进行匹配,注意数据库中的密码存储的是密文。这样即使是数据库的管理者,也不能知道用户的真实密码。

password = DigestUtils.md5DigestAsHex(password.getBytes());

新增员工开发


DTO 设计模式

        数据传输对象 DTO 是一种设计模式,用于封装和传输应用程序不同层之间的数据。

        DTO 是轻量级对象,通常只包含必要的字段,不包含任何业务逻辑。DTO作用于应用程序中不同的业务之间的数据传输,例如在前端和后端之间或在分布式系统中不同的微服务之间。
        在 Spring Boot 应用程序中,DTO 特别有用,因为需要在控制器层、服务层和持久层之间传输数据。通过使用 DTO 就可以将内部数据模型与外部表示
解耦,从而更好地控制数据传输。

         如上图,前端传入这么一组数据,如果有那么一两个数据不是这个实体类 Employee 的,这个时候我们就不能直接用 EmployeeDTO 来接收;而是创建一个 DTO 类 EmployeeDTO 并在数据的 Service 层对 EmployeeDTO 与 Employee 中的共同属性进行赋值操作。这样前端传入的不同属性就不会对 Employee 产生影响,起到一定的解耦效果。

        DTO 允许将暴露给外部的数据与内部的模型隔离。这可以防止暴露敏感和不必要的数据,并为数据交换提供清晰的字段,也在一定程度上保证了内部数据的安全性。


Controller

    /** @description:新增员工**/@PostMapping@ApiOperation("新增员工")public Result save(@RequestBody EmployeeDTO employeeDTO) {log.info("新增员工:{}",employeeDTO);employeeService.save(employeeDTO);return Result.success();}

Service

        由于 EmployeeDTO 的属于与 Employee 的属性都是共同属性,所以可以直接使用 copyProperties 将数据拷贝给 Employee。然后再设置 EmployeeDTO 没有的属性即可。

(1) copyProperties 将一个类的属性值拷贝到另一个类上,但是一定要满足这个属性是两个类共有的。

(2)使用 MD5 给输入的密码加密,保证账号的安全性。

@Overridepublic void save(EmployeeDTO employeeDTO) {// 参数类型向员工类型转化Employee employee = new Employee();BeanUtils.copyProperties(employeeDTO, employee);// 设置账号状态employee.setStatus(StatusConstant.ENABLE);// 更新创建与修改信息时间employee.setCreateTime(LocalDateTime.now());employee.setUpdateTime(LocalDateTime.now());// 设置密文密码employee.setPassword(DigestUtils.md5DigestAsHex(PasswordConstant.DEFAULT_PASSWORD.getBytes()));// 隔离线程中获取数据Long  userId = BaseContext.getCurrentId();// 设置创建人和修改人employee.setCreateUser(userId);employee.setUpdateUser(userId);employeeMapper.insert(employee);}

隔离线程 ThreadLocal

        在新增员工列表中需要设置创建人与修改人,毫无疑问就是获取当前用户的信息,那么该怎么获取呢?我们可以使用隔离线程 -- ThreadLocal。

        ThreadLocal,也称为线程局部变量,是一种特殊的变量。它的特点是,每个线程都有该变量的一个副本,线程之间互不影响,实现了线程间的数据隔离。

        简单来讲就是客户端为每位用户都提供了单独的线程,而每个线程在 ThreadLocal 设置存取的值都是相互独立的。

这里可以简单的验证一下: 

        我们在令牌验证、controller、service 的地方加上一下代码,判断当前线程是否相同:

我们发现是相同的,那么就可以利用这一条特性,获取当前用户的信息:

        我们可以在获取令牌这里将解析的用户id存入 ThreadLocal 线程中,并在设置创建人或者修改人的时候取出,这样就可以获取当前的用户信息了。

Long  userId = BaseContext.getCurrentId();

Mapper

    @Insert("insert into employee(name, username, password, phone, sex, id_number, status, create_time, update_time, create_user, update_user) " +"VALUES (#{name},#{username},#{password},#{phone},#{sex},#{idNumber},#{status},#{createTime},#{updateTime},#{createUser},#{updateUser})")void insert(Employee Employee);

全局异常处理器

         因为我们数据库表的设计中,用户名是唯一属性,那么再次添加这个用户名的时候就会抛出500异常:

 这个异常是一个 SQL 异常是因为 username 冲突造成的 "Duplicate entry..."

        那么我们肯定不能直接将这个异常返回给用户,必须对它进行一定的处理,这个时候就需要用到全局异常处理器:

为什么需要全局异常处理器

不用强制写 try-catch,由全局异常处理器统一捕获处理。
自定义异常,只能用全局异常来捕获。不能直接返回给客户端,客户端是看不懂的,需要接入全局异常处理器。

处理 SQL 异常 

    @ExceptionHandlerpublic Result exceptionHandler(SQLIntegrityConstraintViolationException ex){// Duplicate entry '123' for key 'idx_username'String msg = ex.getMessage();if(msg.contains("Duplicate entry")){String[] split = msg.split(" ");String key = split[2];return Result.error(key+ MessageConstant.ALREADY_EXISTS);}return Result.error(MessageConstant.UNKNOWN_ERROR);}

        利用 ex.getMessage() 来获取报错信息,如果是 Duplicate entry 开头的异常信息那么铁定是 username 冲突问题,那么我们就需要提取 username 信息。并返回我们自定义的报错信息。如果开头不是 Duplicate entry 那么久返回未知异常。

        Result.error() 返回的错误信息不要写死,要同一进行管理,否则以后项目写大了需要更改需求就十分难找。

        我们可以采用静态常量的属性类来解决这个问题。

        这样我们的控制台就不会再抛出异常,前端也会接收到 username 已存在的信息。

代码测试

 

员工分页查询


 分页查询接口设计

 分析需求:

<1>员工信息分页查询后端返回的对象类型为:Result<PageResult>

<2>其实任何分页查询的底层都是加入了 limit 关键字分页

<3>为了内部数据模型与外部参数解耦,我们依旧采用 DTO 的设计模式

mybatis 提供的分页查询框架 pagehelper 依赖:

            <dependency><groupId>com.github.pagehelper</groupId><artifactId>pagehelper-spring-boot-starter</artifactId><version>1.3.0</version></dependency>

 Controller

    @GetMapping("/page")@ApiOperation("员工分页查询")public Result<PageResult> pageQuery(EmployeePageQueryDTO employeePageQueryDTO){log.info("员工分页查询:{}",employeePageQueryDTO);PageResult pageResult = employeeService.pageQuery(employeePageQueryDTO);return Result.success(pageResult);}

Service

        分页的核心就一行代码, PageHelper.startPage(page,pageSize) 这个就表示开始分页。加了这个之后 pagehelper 插件就会通过其内部的拦截器,将执行的 sql 语句,转化为分页的 sql 语句。

    @Overridepublic PageResult pageQuery(EmployeePageQueryDTO employeePageQueryDTO) {//开启分页查询PageHelper.startPage(employeePageQueryDTO.getPage(),employeePageQueryDTO.getPageSize());Page<Employee> page =  employeeMapper.pageQuery(employeePageQueryDTO);long total = page.getTotal();List<Employee> employees = page.getResult();return new PageResult(total,employees);}

 Mapper

使用动态SQL对name进行模糊查询,并以创建时间排降序。

    <select id="pageQuery" resultType="com.sky.entity.Employee">SELECT * FROM employee<where><if test="name!=null and name!=''">and name like concat('%',#{name},'%')</if></where>ORDER BY create_time DESC</select>

代码测试

 

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

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

相关文章

Java poi 模板导出Word 带图片

Java poi 模板导出Word 带图片 重点&#xff01;&#xff01;&#xff01; 官方文档&#xff1a;https://deepoove.com/poi-tl/#_maven 最终效果 模板 其实内容都在官方文档里写的非常明白了 我这里只是抛砖引玉。 Maven依赖 <poi.version>4.1.2</poi.version>…

HMSC联合物种分布模型在群落生态学中的贝叶斯统计分析应用

联合物种分布模型&#xff08;Joint Species Distribution Modelling&#xff0c;JSDM&#xff09;在生态学领域&#xff0c;特别是群落生态学中发展最为迅速&#xff0c;它在分析和解读群落生态数据的革命性和独特视角使其受到广大国内外学者的关注。在学界不同研究团队研发出…

postman变量和脚本功能介绍

1、基本概念——global、collection、environment 在postman中&#xff0c;为了更好的管理各类变量、测试环境以及脚本等&#xff0c;创建了一些概念&#xff0c;包括&#xff1a;globals、collection、environment。其实在postman中&#xff0c;最上层还有一个Workspaces的概…

Java NIO 核心知识总结

NIO 简介 在传统的 Java I/O 模型&#xff08;BIO&#xff09;中&#xff0c;I/O 操作是以阻塞的方式进行的。也就是说&#xff0c;当一个线程执行一个 I/O 操作时&#xff0c;它会被阻塞直到操作完成。这种阻塞模型在处理多个并发连接时可能会导致性能瓶颈&#xff0c;因为需要…

Qwen2.5-Coder-32B-Instruct Docker 部署openai接口

Qwen2.5-Coder-32B-Instruct 模型下载,国内快捷方式: conda create -n modelscope python=3.10 conda activate modelscopepip install modelscopemodelscope download --model Qwen/Qwen2.5-Coder-32B-Instruct --local_dir /ssd/xiedong/Qwen/Qwen2.5-Coder-32B-I

基于STM32的智能语音识别饮水机系统设计

功能描述 1、给饮水机设定称呼&#xff0c;喊出称呼&#xff0c;饮水机回答&#xff1a;我在 2、语音进行加热功能&#xff0c;说&#xff1a;请加热&#xff0c;加热片运行 3、饮水机水位检测&#xff0c;低于阈值播报“水量少&#xff0c;请换水” 4、检测饮水机水温&#xf…

电子应用产品设计方案-10:全自动智能门禁系统设计方案

一、系统概述 本全自动智能门禁系统旨在提供高效、安全、便捷的人员进出管理解决方案。通过融合先进的生物识别技术、传感器技术、网络通信技术和自动化控制技术&#xff0c;实现门禁的自动识别、授权、记录和管理。 二、系统组成 1. 前端识别设备 - 人脸识别模块&#xff1a;采…

聊聊Flink:Flink的分区机制

一、前言 flink任务在执行过程中&#xff0c;一个流&#xff08;stream&#xff09;包含一个或多个分区&#xff08;Stream partition&#xff09;。TaskManager中的一个slot的subtask就是一个stream partition&#xff08;流分区&#xff09;&#xff0c;一个Job的流&#xf…

IO流实用案例:用字节流--输入流(Inpustream)、输出流(OutputStream)写一个拷贝图片的案例--超简单!

案例背景&#xff1a; 我的电脑桌面有一张白敬亭的照片&#xff0c;我们需要把这张照片拷贝到我的电脑D:\学习软件\copyBJT目录下&#xff0c;当前我们这个目录是没有东西的。 代码演示以及注释&#xff1a; ublic class StreamCopy {public static void main(String[] args)…

ArkTS学习笔记:ArkTS起步

ArkTS是HarmonyOS的主力应用开发语言&#xff0c;基于TypeScript扩展&#xff0c;强化了静态检查和分析&#xff0c;旨在提升程序稳定性和性能。它采用静态类型&#xff0c;禁止运行时改变对象布局&#xff0c;并对UI开发框架能力进行扩展&#xff0c;支持声明式UI描述和自定义…

卡尔曼滤波:从理论到应用的简介

卡尔曼滤波&#xff08;Kalman Filter&#xff09;是一种递归算法&#xff0c;用于对一系列噪声观测数据进行动态系统状态估计。它广泛应用于导航、控制系统、信号处理、金融预测等多个领域。本文将介绍卡尔曼滤波的基本原理、核心公式和应用案例。 1. 什么是卡尔曼滤波&#x…

【已解决】git push一直提示输入用户名及密码、fatal: Could not read from remote repository的问题

问题描述&#xff1a; 在实操中&#xff0c;git push代码到github上一直提示输入用户名及密码&#xff0c;并且跳出的输入框输入用户名和密码后&#xff0c;报错找不到远程仓库 实际解决中&#xff0c;发现我环境有两个问题解决&#xff1a; git push一直提示输入用户名及密码…

【Rust 编程语言工具】rustup-init.exe 安装与使用指南

rustup-init.exe 是用于安装和管理 Rust 编程语言工具链的 Windows 可执行文件。Rust 是一种系统级编程语言&#xff0c;旨在提供安全、并发和高性能的功能。rustup-init.exe 是官方提供的安装器&#xff0c;用于将 Rust 安装到 Windows 操作系统中&#xff0c;并配置相关环境。…

Mac 使用mac 原生工具将mp4视频文件提取其中的 mp3 音频文件

简介 Hello! 非常感谢您阅读海轰的文章,倘若文中有错误的地方,欢迎您指出~ ଘ(੭ˊᵕˋ)੭ 昵称:海轰 标签:程序猿|C++选手|学生 简介:因C语言结识编程,随后转入计算机专业,获得过国家奖学金,有幸在竞赛中拿过一些国奖、省奖…已保研 学习经验:扎实基础 + 多做笔…

项目中用户数据获取遇到bug

项目跟练的时候 Uncaught (in promise) TypeError: Cannot read properties of undefined (reading ‘code’) at Proxy.userInfo (user.ts:57:17) 因此我想要用result接受信息的时候会出错&#xff0c;报错显示为result.code没有该值 导致我无法获取到相应的数据 解决如下 给…

【视觉SLAM】1-概述

读书笔记 文章目录 1. 经典视觉SLAM框架2. 数学表述2.1 运动方程2.2 观测方程2.3 问题抽象 1. 经典视觉SLAM框架 传感器信息读取&#xff1a;相机图像、IMU等多源数据&#xff1b;前端视觉里程计&#xff08;Visual Odometry&#xff0c;VO&#xff09;&#xff1a;估计相机的相…

Isaac Sim+SKRL机器人并行强化学习

目录 Isaac Sim介绍 OmniIssacGymEnvs安装 SKRL安装与测试 基于UR5的机械臂Reach强化学习测评 机器人控制 OMNI GYM环境编写 SKRL运行文件 训练结果与速度对比 结果分析 运行体验与建议 Isaac Sim介绍 Isaac Sim是英伟达出的一款机器人仿真平台&#xff0c;适用于做机…

Python学习------第八天

函数 函数的传入参数 掌握函数返回值的作用 掌握函数返回值的定义语法 函数的嵌套调用&#xff1a; 函数的局部变量和全局变量 局部变量的作用&#xff1a;在函数体内部&#xff0c;临时保存数据&#xff0c;即当函数调用完成后&#xff0c;则销毁局部变量。 money 5000000 n…

机器学习基础04

目录 1.朴素贝叶斯-分类 1.1贝叶斯分类理论 1.2条件概率 1.3全概率公式 1.4贝叶斯推断 1.5朴素贝叶斯推断 1.6拉普拉斯平滑系数 1.7API 2.决策树-分类 2.1决策树 2.2基于信息增益的决策树建立 2.2.1信息熵 2.2.2信息增益 2.2.3信息增益决策树建立步骤 2.3基于基…

The Internals of PostgreSQL 翻译版 持续更新...

为了方便自己快速学习&#xff0c;整理了翻译版本&#xff0c;目前翻译的还不完善&#xff0c;后续会边学习边完善。 文档用于自己快速参考&#xff0c;会持续修正&#xff0c;能力有限,无法确保正确!!! 《The Internals of PostgreSQL 》 不是 《 PostgreSQL14 Internals 》…