第四阶段02-酷鲨商城项目Mybatis相关的配置

news/2024/4/25 3:16:10/文章来源:https://blog.csdn.net/aiheshuicxy/article/details/129248904

14. 添加与Mybatis相关的配置

在每个项目中,当需要使用Mybatis实现数据库编程时,都需要添加2项一次性配置:配置Mapper接口所在的包(package)、配置XML文件在哪里。

关于配置Mapper接口所在的包,可以(以下做法二选一即可):

  • 【推荐】在配置类上使用@MapperScan注解,并指定注解的参数,此参数值就是包名
  • 在每个Mapper接口上添加@Mapper注解

则在项目的根包下创建config.MybatisConfiguration类,在此类上添加@Configuration注解,则可以将此类标记为“配置类”,然后,再添加@MapperScan注解来指定Mapper接口所在的包:

package cn.tedu.csmall.product.config;import org.mybatis.spring.annotation.MapperScan;
import org.springframework.context.annotation.Configuration;@Configuration
@MapperScan("cn.tedu.csmall.product.mapper")
public class MybatisConfiguration {
}

**提示:**目前还没有创建任何Mapper接口,甚至,用于存放Mapper接口的mapper子包都是不存在,但这并不影响以上配置!后续在创建Mapper接口时,保证创建的接口都在以上配置值对应的包中即可!

另外,关于“配置XML文件在哪里”,需要通过配置文件(application.properties / application.yml / 相关Profile文件)中的mybatis.mapper-locations进行配置。由于此项配置的值,不会因为切换开发环境、测试环境等环境而变化,也不会因为使用不同的电脑来开发而变化,所以,应该将其配置在主配置文件中!

先在src/main/resources下创建mapper文件夹,此文件夹将用于存放配置SQL语句的XML文件。

application.yml中添加配置:

mybatis:mapper-locations: classpath:mapper/*.xml

15. 使用Mybatis访问数据

15.1. Mybatis的基本使用步骤

首先,需要创建接口,通常每张数据表都有一个对应的Mapper接口,并在接口中声明抽象方法。

关于抽象方法的声明:

  • 返回值类型:如果要执行的数据访问操作是增、删、改类型的,始终使用int作为返回值类型,表示“受影响的行数”,如果不关心“受影响的行数”,也可以使用void作为返回值类型,但并不推荐;如果要执行数据访问操作是查询类型的,只需要保证返回值类型能够“放得下”查询结果即可。

    • 提示:非统计类型的查询,一般不建议使用实体类作为返回值类型,应该使用其它POJO类型,《阿里巴巴Java开发手册》提供的参考

      1) 数据对象:xxxDO,xxx 即为数据表名。
      2) 数据传输对象:xxxDTO,xxx 为业务领域相关的名称。
      3) 展示对象:xxxVO,xxx 一般为网页名称。
      4) POJO 是 DO/DTO/BO/VO 的统称,禁止命名成 xxxPOJO。
      

      另外,关于POJO的后缀,必须每个字母都是大写的,《阿里巴巴Java开发手册》提供的强制规定:

      【强制】类名使用 UpperCamelCase 风格,必须遵从驼峰形式,但以下情形例外:DO / BO /
      DTO / VO / AO
      正例:MarcoPolo / UserDO / XmlService / TcpUdpDeal / TaPromotion
      反例:macroPolo / UserDo / XMLService / TCPUDPDeal / TAPromotion
      
    • 通常,每张数据表都至少有2个VO类(或使用其它后缀),分别用于表示查询详情、查询列表项的结果类型

  • 方法名称:自定义,建议不要重载,《阿里巴巴Java开发手册》提供的参考

    1) 获取单个对象的方法用 get 做前缀。
    2) 获取多个对象的方法用 list 做前缀。
    3) 获取统计值的方法用 count 做前缀。
    4) 插入的方法用 save/insert 做前缀。
    5) 删除的方法用 remove/delete 做前缀。
    6) 修改的方法用 update 做前缀。
    
  • 参数列表:取决于要执行的SQL语句中有哪些参数,如果参数的数量超过1个,且具有明确的相关性,建议使用封装的类型作为抽象方法的参数

    • 通常,插入单条数据、修改单条数据时,使用实体类作为抽象方法的参数
    • 如果涉及批量数据操作,例如批量删除、批量更新、批量插入等,可以使用数组、可变参数或List集合表示若干个参数值

然后,需要准备好XML文件,在XML文件中配置抽象方法映射的SQL语句。

关于XML文件,应该是从此前的项目中复制过来再修改,因为此文件需要固定的头部声明,即:
请添加图片描述

关于此类文件,可以从 http://doc.canglaoshi.org/config/Mapper.xml.zip 下载,或在网上搜索相关文章,或在Mybatis官网上找到示例代码,都可以得到以上固定的代码片段。

关于此XML文件的内部:

  • 根标签必须是<mapper>,且此标签必须配置namespace属性,取值为对应的接口的全限定名(包名加类名)

  • 在根标签的子级,可以使用<insert> / <delete> / <update> / <select>这些标签来配置增、删、改、查对应的SQL语句,这些标签都必须配置id属性,取值为对应的抽象方法的名称

    • 其实,<update><delete>这2个标签可以随意调换使用,例如,配置DELETE语句可以使用<update>标签,甚至,在不考虑获取自动编号的ID时,<insert>标签和<update><delete>也可以随意调用使用,因为这3种数据访问操作在JDBC的底层实现是一样的,当然,这种乱用的方式肯定是非常不建议的
  • <insert>这4种标签的子级,编写SQL语句,参数部分使用#{名称}格式的占位符来表示,其中,占位符中的“名称”应该是参数名称,或封装类型中的属性名

  • 严格说来,在#{}格式的占位符中的名称,将被用于生成Setter & Getter的名称,然后,Mybatis框架会自动调用相关方法

  • 如果配置的是<insert>标签,并且,插入数据的表的主键是自动编号的,应该在此标签上配置useGeneratedKeys="true"keyProperty="主键属性"这2个属性,以获取自动编号的ID值

  • 如果涉及批量操作,需要通过<foreach>标签来遍历参数对象,关于此标签的使用:

    • collection属性:表示被遍历的对象,当抽象方法的参数只有1个,且没有添加@Param注解时,如果参数是数组或可变参数,则此属性的值为array,如果参数是List集合,则此属性的值为list;当抽象方法的参数超过1个,需要添加@Param注解,则此属性的值为注解的配置值
    • item属性:表示遍历过程中各元素的名称,是自定义名称,并且,在<foreach>子级的#{}占位符中,也使用此名称表示被遍历到的元素
    • separator属性:表示遍历过程中各值之间的分隔符号
    • <foreach>标签是Mybatis的“动态SQL”标签之一
  • 关于动态SQL的<if>标签,可以对参数进行判断,从而确定SQL语句中应该包含或不包含哪些片段

    • 需要注意,没有所谓的else标签,如果一定要实现类似Java中if...else...的效果,可以使用2个条件完全相反的<if>标签(虽然执行效率会差一些),或者,使用<choose>系列标签,语法格式如下:

      <choose><when test="条件">满足条件时的SQL片段</when><otherwise>不满足条件时的SQL片段</otherwise>
      </choose>
      
  • 如果涉及修改数据的操作,当使用了<if>实现动态SQL时,通常,应该使用<set>标签将更新列表框住,并且,SQL语句中不需要写SET关键字

  • 如果配置的是<select>标签,此标签必须配置resultTyperesultMap这2个属性中的某1个

    • resultType的值是基本数据类型时,直接使用类型名称作为此属性值即可,例如resultType="int"
    • resultType的值是引用数据类型时,需要使用类型的全限定名作为此属性值,例如resultType="cn.tedu.xxx.User",但是,如果类型是java.lang包下的,则可以不写包名,例如resultType="Integer"
  • 在编写查询的SQL语句时,可以使用<sql>标签封装查询的字段列表,并使用<include>标签来引用封装的查询字段列表

    • 由于IntelliJ IDEA会检查每个标签的子级内容的语法,所以,在<sql>标签下直接写字段列表,IntelliJ IDEA会提示错误,但是并不影响运行,可以使用<if test="true">将字段列表框住,以“欺骗”IntelliJ IDEA使之不报错
    • 通常,在<include>标签的子级并不编写任何代码,所以,此标签推荐使用单标签格式
  • 建议为所有的“非统计查询”配置<resultMap>

15.2. 关于SQL语句的编写规范

关于INSERT语句:

  • 使用INSERT INTO开头,不要省略INTO关键字
  • 使用VALUES,不要使用VALUE
  • 显式写出字段列表,不要省略字段列表

关于SELECT语句:

  • 统计查询时,count()函数内必须使用星号

  • 不允许使用星号(*)表示被查询的字段列表

  • 查询列表时,必须显式的使用ORDER BY指定排序规则,如果没有使用ORDER BY,则排序结果是不可控的,并且,在使用ORDER BY指定排序规则时,如果第1排序规则无法明确的区分所有数据的顺序,还应该指定第2排序规则,甚至第3排序规则等,直至所有数据都有明确的排序规则
    请添加图片描述

15.3. 关于编写SQL语句时的代码提示

首先,需要在IntelliJ IDEA中,为当前项目配置Database面板,且Database面板中仅有当前使用的这1个数据库。

然后,需要在设置中,指定SQL方言(Dialect),即:
请添加图片描述

**提示:**如果完成以上配置,在XML文件中编写SQL语句仍没有提示,可以尝试删除并重新配置Database面板,或将方言在MariaDB和MySQL之间切换。

15.4. 使用测试检验开发的数据访问功能

在Spring Boot项目中,所有的测试类都应该创建在src/test/java下的根包(创建项目时就存在的包,与src/main/java下的根包是相同的)下。

在Spring Boot项目中,当需要执行的测试是基于加载整个项目环境的(例如需要读取配置文件、需要使用@Autowired自动装配等),测试类必须添加@SpringBootTest注解。

**注意:**由于src/test/java下的根包与src/main/java下的根包是相同的,所以,任何测试类都不要和被测试类的名称相同,一般,测试类的名称建议添加Tests作为最后一个单词,例如AlbumMapperTests

例如,在src/test/java下的根包下创建mapper.AlbumMapperTests类,在类上添加@SpringBootTest注解,然后,在类中对AlbumMapper接口中定义的功能进行测试:

package cn.tedu.csmall.product.mapper;import cn.tedu.csmall.product.pojo.entity.Album;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest
public class AlbumMapperTests {@AutowiredAlbumMapper mapper;@Testvoid insert() {Album album = new Album();album.setName("测试名称001");album.setDescription("测试简介001");album.setSort(100); // 注意:由于MySQL中表设计的限制,此值只能是[0,255]区间内的int rows = mapper.insert(album);System.out.println("插入数据完成,受影响的行数:" + rows);}}

15.5. 使用Mybatis的常见错误

如果Mapper接口不在@MapperScan指定的包下,或根本没有配置@MapperScan,总之,Mybatis找不到Mapper接口时,会出现错误:

Caused by: org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'cn.tedu.csmall.product.mapper.AlbumMapper' available: expected at least 1 bean which qualifies as autowire candidate. Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true)}

如果存在以下问题:

  • 在XML文件中<mapper>标签的namespace值有误
  • 在XML文件中<insert>这类标签的id值有误
  • 在配置文件中mybatis.mapper-locations属性的值有误

总之,当Mybatis找不到抽象方法映射的SQL语句时,会出现错误:

org.apache.ibatis.binding.BindingException: Invalid bound statement (not found): cn.tedu.csmall.product.mapper.AlbumMapper.insert

如果配置SQL语句时,在#{}格式的占位符中,名称写错时,会出现错误:

Caused by: org.apache.ibatis.reflection.ReflectionException: There is no getter for property named 'nama' in 'class cn.tedu.csmall.product.pojo.entity.Album'

如果配置<select>标签时,既没有配置resultType又没有配置resultMap,会出现错误:

Caused by: org.apache.ibatis.executor.ExecutorException: A query was run and no Result Maps were found for the Mapped Statement 'cn.tedu.csmall.product.mapper.AlbumMapper.count'.  It's likely that neither a Result Type nor a Result Map was specified.

如果配置<select>标签时,使用resultType,但值却是某个<resultMap>的id,或是错误的全限定名,会出现错误:

Caused by: java.lang.ClassNotFoundException: Cannot find class: StandardResultMap

如果配置<select>标签时,使用resultMap,但值却是某个类名,或是其它错误值,只要不是某个<resultMap>的id值,会出现错误:

java.lang.IllegalArgumentException: Result Maps collection does not contain value for cn.tedu.csmall.product.pojo.vo.AlbumStandardVO

如果某个抽象方法对应的SQL语句的配置存在多个(通常可能是因为复制粘贴代码,但没有修改对应关系导致的,例如忘记修改新的XML文件中的namespace值等),会出现错误(这个错误,不仅仅只看最后一个Caused by,偏上的Caused by也有参考价值,会告诉你是哪些文件中出现了对应同一个抽象方法的配置):

Caused by: org.apache.ibatis.builder.BuilderException: Error parsing Mapper XML. The XML location is 'file [D:\IdeaProjects\jsd2209-csmall-product-teacher\target\classes\mapper\BrandMapper.xml]'. Cause: java.lang.IllegalArgumentException: Result Maps collection already contains value for cn.tedu.csmall.product.mapper.AlbumMapper.StandardResultMapCaused by: java.lang.IllegalArgumentException: Result Maps collection already contains value for cn.tedu.csmall.product.mapper.AlbumMapper.StandardResultMap

16. 关于SLF4j日志

在开发实践中,通常是禁止使用System.out.println()这种语句输出信息的,主要原因有:

  • 不易控:在开发过程中,可能会通过这样的语句输出某些变量值,或输出程序的执行过程,这些输出的内容可能仅适用于开发过程中,当项目实际部署上线,就不再应该甚至不允许输出这些内容,因为输出的变量值可能是敏感信息,或程序的执行过程不应该被暴露出来
  • 执行效率低下

在Spring Boot项目中,基础依赖项(spring-boot-starter)中已经包含了日志相关的依赖项,可以直接使用日志框架来输出信息!

在添加了Lombok后,在任何类的声明之前,添加@Slf4j注解,则编译期会自动声明一个名为log的变量,所以,可以在类中通过此变量来输出日志。

如果没有使用Lombok,可以自行声明日志变量,例如:

private static Logger logger = LoggerFactory.getLogger(Slf4jTests.class);

在日志框架中,根据日志信息的重要程度,从不太重要,到非常重要,划分了几个等级,依次是:

  • trace:跟踪信息
  • debug:调试信息
  • info:一般信息
  • warn:警告信息
  • error:错误信息

在调用log变量的方法来输出日志时,有以上5个级别对应的方法,调用不同的方法,就会输出不同级别的日志。

在没有加载Spring Boot的情况下,日志的默认显示级别是debug,只会显示此级别及更加重要的级别的日志,如果加载了Spring Boot(例如在测试类上添加了@SpringBootTest注解),日志的默认显示级别是info,则只会显示infowarnerror级别的日志,不会显示tracedebug级别的日志。

在Spring Boot项目中,可以在配置文件(application.properties / application.yml / 相关Profile配置)中配置logging.level.包名属性,以指定某个包下的所有类的默认日志显示级别,此属性的值为5个级别中的某1个。

例如,在application-dev.yml添加配置(另外,你可以在其它Profile配置中为日志的显示级别配置不同的值):

logging:level:cn.tedu.csmall: trace

**提示:**当配置包名时,不必把包名配置得特别具体,可以作用于其子孙包下所有的类,但是,必须至少配置1级包名,例如配置到cn这一级包,不可以完全不配置包名,需要注意,当前项目中添加的所有依赖项,也是当前项目的一部分,不应该将包名配置得过于简单,例如只使用cn作为配置的包名。

**提示:**Mybatis框架会生成各Mapper接口的对象,这些对象在执行SQL语句时,也会输出日志,是traceinfo级别的日志,当把日志的显示级别设置为较低的级别时,可以看到这些日志。

在调用日志的方法时,如果输出的信息中包含变量值,推荐使用trace(String message, Object... args)方法(其它级别也是同样参数列表的方法),在String message参数中,可以使用{}作为占位符,表示此处是一个变量值,然后,通过Object... args依次传入各占位符对应的值,例如:

int x = 1;
int y = 2;
System.out.println("x = " + x + ", y = " + y + ", x + y = " + (x + y)); // 传统做法
log.info("x = {}, y = {}, x + y = {}", x , y , x + y); // 使用日志输出变量的做法

以上使用日志输出时,不会涉及到字符串的拼接,所以,执行效率会更高,并且,以上方法的第1个参数是字符串常量,是在内存中的字符串常量池中的,也可以一定程度上提高执行效率。

另外,SLF4j其实是一种日志框架的标准,并没有具体实现日志的输出功能,通常,是使用log4jlogback这些日志框架来实现具体功能的!请添加图片描述

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

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

相关文章

【一】kubernetes集群部署

一、docker环境搭建 1、移除以前docker相关包 sudo yum remove docker docker-client docker-client-latest docker-common docker-latest docker-latest-logrotate docker-logrotate docker-engine2、配置yam源 sudo yum install -y yum-utilssudo yum-config-manager --ad…

C++进阶--二叉树编程题

文章目录力扣606. 根据二叉树创建字符串力扣102. 二叉树的层序遍历力扣236. 二叉树的最近公共祖先JZ36 二叉搜索树与双向链表力扣105--通过前序和中序遍历构造二叉树力扣144--二叉树的前序遍历&#xff08;非递归&#xff09;力扣94--二叉树的中序遍历&#xff08;非递归&#…

虹科新闻|虹科与iX systems正式建立合作伙伴关系

近日&#xff0c;虹科与美国iXsystems公司达成战略合作&#xff0c;虹科正式成为iXsystems公司在中国区域的认证授权代理商。未来&#xff0c;虹科将携手iXsystems&#xff0c;共同致力于提供企业级存储解决方案。虹科及iXsystems双方的高层领导人员都对彼此的合作有很大的信心…

操作系统基础教程

目录 第二章&#xff1a;处理器管理 概览 进程调度的层次 进程的调度方式&#xff1a; 调度的评价标准&#xff1a; 典型的调度算法&#xff1a; 第三章&#xff1a;同步、通信和死锁 什么是进程同步&#xff1f; 什么是进程互斥&#xff1f; 进程同步的实现方式 进程…

JVM总结

1. 内存结构 线程私有区 程序计算器 作用&#xff1a;是一块较小的内存空间&#xff0c;存储的是当前线程所执行的字节码文件的序号特点&#xff1a;线程私有&#xff0c;不会出现内存空间溢出 虚拟机栈 虚拟机栈是管理JAVA方法执行的内存模型&#xff0c;每个方法执行时都…

贴吧顶贴软件《今日/更新》

贴吧顶贴软件《今日/更新》百收贴吧工具箱&#xff0c;贴吧顶帖软件&#xff0c;贴吧推广引流神器#贴吧顶帖#贴吧推广 hello&#xff0c;大家好&#xff0c;我是软件的作者百收编辑狂潮老师。本次的视频讲解是作为一个百度顶贴的自动化脚本的视频安装教程和使用教程。你作为新…

SpringCloud(五)MQ消息队列

MQ概念常见消息模型helloworld案例实现实现spring AMQP发送消息实现spring AMQP接收消息工作消息队列实现发布订阅模型Fanout Exchange实现DirectExchange实现TopicExchange实现DirectExchange 和FanoutExchange的差异DirectExchange 和TopicExchange的差异基于RabbitListener注…

钉钉产品体验报告

一、调研的目的了解企业社交软件&#xff0c;借写竞品分析来帮助自己整理思路&#xff0c;看清市场的发展趋势&#xff1b;体验这类企业设计软件&#xff0c;掌握产品核心业务流程和产品结构&#xff0c;把握需求对应的功能点和界面结构&#xff0c;并侧面了解用户习惯&#xf…

用Python做数据分析有哪些优势?

众所周知&#xff0c;可以用作数据分析的语言有很多&#xff0c;包含Python、R语言等&#xff0c;而且Python被誉为数据分析的一大利器&#xff0c;更是该领域的首选语言&#xff0c;那么用Python做数据分析有哪些优势呢?跟着蛋糕往下看。 第一、Python语言自身的优势 Pytho…

ShardingSphere水平、垂直分库、分表和公共表

目录一、ShardingSphere简介二、ShardingSphere-分库分表1、垂直拆分&#xff08;1&#xff09;垂直分库&#xff08;2&#xff09;垂直分表2、水平拆分&#xff08;1&#xff09;水平分库&#xff08;2&#xff09;水平分表三、水平分库操作1、创建数据库和表2、配置分片的规则…

BigGAN

1、BIGGAN 解读1.1、作者 Andrew Brock、Jeff Donahue、Karen Simonyan 1.2、摘要 尽管最近在生成图像建模方面取得了进展&#xff0c;但从 ImageNet 等复杂数据集中 成功生成高分辨率、多样化的样本仍然是一个难以实现的目标。为此&#xff0c;我们以迄 今为止最大的规模训练生…

fastadmin:在新增页面,打开弹窗单选,参数回传

样式&#xff1a;核心代码&#xff1a;一、弹窗的控制器中&#xff1a;// 定义一个公共函数select()&#xff0c;如果这个请求是Ajax&#xff0c;则返回index()函数&#xff0c;否则返回view对象的fetch()函数。 public function select() {if ($this->request->isAjax(…

【软件测试】测试老鸟的迷途,进军高级自动化测试测试......

目录&#xff1a;导读前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09;前言 很多从业几年的选手…

【阿旭机器学习实战】【37】电影推荐系统---基于矩阵分解

【阿旭机器学习实战】系列文章主要介绍机器学习的各种算法模型及其实战案例&#xff0c;欢迎点赞&#xff0c;关注共同学习交流。 电影推荐系统 目录电影推荐系统1. 问题介绍1.1推荐系统矩阵分解方法介绍1.2 数据集&#xff1a;ml-100k2. 推荐系统实现2.1 定义矩阵分解函数2.2 …

消息中间件的概念

中间件(middleware)是基础软件的一大类&#xff0c;属于可复用的软件范畴。中间件在操作系统软件&#xff0c;网络和数据库之上&#xff0c;应用软件之下&#xff0c;总的作用是为处于自己上层的应用软件提供运行于开发的环境&#xff0c;帮助用户灵活、高效的开发和集成复杂的…

ICA简介:独立成分分析

1. 简介 您是否曾经遇到过这样一种情况&#xff1a;您试图分析一个复杂且高度相关的数据集&#xff0c;却对信息量感到不知所措&#xff1f;这就是独立成分分析 (ICA) 的用武之地。ICA 是数据分析领域的一项强大技术&#xff0c;可让您分离和识别多元数据集中的底层独立来源。 …

PPP简介,PPP分层体系架构,PPP链路建立过程及PPP的帧格式

PPP&#xff08;Point-to-Point Protocol&#xff09;是一种用于在两个网络节点之间传输数据的通信协议。它最初是为在拨号网络上进行拨号连接而开发的&#xff0c;现在已经被广泛应用于各种网络环境中&#xff0c;例如在宽带接入、虚拟专用网&#xff08;VPN&#xff09;等场景…

【JAVA】一个项目如何预先加载数据?

这里写目录标题需求实现AutowiredPostConstruct实例CommandLineRunner实例ApplicationListener实例参考需求 一般我们可能会有一些在应用启动时加载资源的需求&#xff0c;局部或者全局使用&#xff0c;让我们来看看都有哪些方式实现。 实现 Autowired 如果是某个类里需求某…

[1]MyBatis+Spring+SpringMVC+SSM整合

一、MyBatis 1、MyBatis简介 1.1、MyBatis历史 MyBatis最初是Apache的一个开源项目iBatis, 2010年6月这个项目由Apache Software Foundation迁移到了Google Code。随着开发团队转投Google Code旗下&#xff0c; iBatis3.x正式更名为MyBatis。代码于2013年11月迁移到Github。…

Vue中如何利用websocket实现实时通讯

首先我们可以先做一个简单的例子来学习一下简单的websocket模拟聊天对话的功能 原理很简单&#xff0c;有点像VUE中的EventBus&#xff0c;用emit和on传来传去 首先我们可以先去自己去用node搭建一个本地服务器 步骤如下 1.新建一个app.js&#xff0c;然后创建pagejson.js文…