SpringMVC Day 07 : 表单验证

前言

表单验证在Web开发中是非常常见和重要的一部分,它用于确保用户提交的数据符合预期的规则和限制。

通过表单验证,我们可以有效地捕获并处理用户输入中的错误或不正确的数据,从而提高应用程序的数据质量和用户体验。在本教程中,我们将学习如何使用Spring MVC的表单验证功能来验证用户提交的表单数据,并处理验证结果。

在接下来的内容中,我们将深入了解Spring MVC表单验证的各个方面,包括如何定义验证规则、如何显示验证错误信息以及如何处理验证结果等。我们还将探讨常见的表单验证场景,并提供实际示例来演示如何实现这些验证逻辑。

一、前期准备

1、新建项目,结构如下

 2、导入依赖 
    <dependencies><!-- springmvc 依赖,会将spring的核心包一并添加进来 --><!-- https://mvnrepository.com/artifact/org.springframework/spring-webmvc --><dependency><groupId>org.springframework</groupId><artifactId>spring-webmvc</artifactId><version>5.3.23</version></dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><version>1.18.30</version></dependency><dependency><groupId>ch.qos.logback</groupId><artifactId>logback-classic</artifactId><version>1.4.5</version></dependency><!-- 集成 hibernata bean 验证器 --><dependency><groupId>org.hibernate.validator</groupId><artifactId>hibernate-validator</artifactId><version>6.0.13.Final</version></dependency></dependencies>

hibernate-validator是一个基于Hibernate Validator的Java Bean验证框架。它提供了一组用于验证Java Bean属性的注解,并具有灵活、可扩展的验证机制。

使用hibernate-validator可以实现对Java Bean对象进行数据验证和校验,确保对象符合预期的规则和约束。它可以用于以下方面:

  1. 参数校验:在方法参数上添加注解,对传入的参数进行验证,以确保参数满足指定的条件。例如,可以验证参数是否为非空、是否满足正则表达式、是否在指定的范围内等。

  2. 表单验证:在表单提交时,对表单中的字段进行验证,以确保用户输入的数据符合要求。例如,可以验证用户名、密码、邮箱等字段的格式和有效性。

  3. 实体类验证:在保存或更新实体对象时,对实体类中的字段进行验证,以确保数据的完整性和一致性。例如,可以验证实体对象的属性是否为空、长度是否符合要求、是否满足特定的业务规则等。

  4. 自定义验证:除了使用内置的验证注解外,还可以自定义验证注解和验证器,以实现更复杂的验证逻辑。通过自定义注解和验证器,可以灵活地定义和扩展验证规则。

通过引入hibernate-validator的依赖,可以在项目中使用其提供的注解来对Java Bean对象进行验证。在验证过程中,如果发现验证不通过,将会抛出相关的异常或返回验证结果,从而方便开发者处理验证失败的情况。

 3、配置 web.xml 
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"version="4.0"><servlet><servlet-name>dispatcher</servlet-name><servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class><load-on-startup>1</load-on-startup></servlet><servlet-mapping><servlet-name>dispatcher</servlet-name><url-pattern>/</url-pattern></servlet-mapping>
</web-app>

用于配置 Servlet 的映射和加载。在 Spring MVC 中,它用于配置 DispatcherServlet 的初始化和请求映射。

具体来说,这段配置的作用如下:

  1. 定义了一个名为 "dispatcher" 的 Servlet,并指定了 org.springframework.web.servlet.DispatcherServlet 作为其处理类。
  2. 设置了 load-on-startup 属性为 1,表示在应用启动时就加载该 Servlet。
  3. 使用 <servlet-mapping> 元素将 "dispatcher" Servlet 映射到所有的请求路径上(即 <url-pattern>/</url-pattern>),意味着所有的请求都会经过该 Servlet 进行处理。

 这段配置的作用是将所有的请求交给 DispatcherServlet 处理,并让它成为应用的核心控制器。DispatcherServlet 将根据请求的 URL 和其他配置信息,将请求分发给相应的处理器方法进行处理,然后返回响应结果。

4、在 resources 包下新建一个 message.properties
user.userName.notEmpty = 请输入用户名
user.age.notNull = 请输入年龄
user.age.min = 年龄不能小于18
user.birth.notNull = 请输入出生年月日
user.email.notEmpty = 请输入邮箱地址
user.email.legal = 请填写合法的email邮箱

这里面的值会在实体类中获取。 

5、新建 User 实体类

@Data
public class User {/*** 验证空字符串*/@NotEmpty(message = "{user.userName.notEmpty}")private String userName;/*** 验证空值*/@NotNull(message = "{user.age.notNull}")/*** 最小值范围*/@Min(value = 18,message = "{user.age.min}")private Integer age;@NotNull(message = "{user.birth.notNull}")//@DateTimeFormat(pattern = "yyyy-MM-dd")private Date brith;@NotEmpty(message = "{user.email.notEmpty}")@Email(message = "{user.email.legal}")private String email;}

这段代码是一个User类的定义,使用了一些注解来进行数据验证。下面是每个注解的作用:

  1. @NotEmpty(message = "{user.userName.notEmpty}"):该注解用于验证字符串属性值是否非空。如果userName属性为空字符串,则会产生一个指定消息的验证错误。

  2. @NotNull(message = "{user.age.notNull}"):该注解用于验证属性值是否非空。如果age属性为空值(null),则会产生一个指定消息的验证错误。

  3. @Min(value = 18, message = "{user.age.min}"):该注解用于验证数值属性值是否大于等于指定的最小值。在这个例子中,如果age属性的值小于18,则会产生一个指定消息的验证错误。

  4. @NotNull(message = "{user.birth.notNull}"):该注解用于验证属性值是否非空。如果birth属性为空值(null),则会产生一个指定消息的验证错误。

  5. //@DateTimeFormat(pattern = "yyyy-MM-dd"):这是一个注释,用于指定日期类型属性的格式化规则。在这个例子中,如果取消注释并指定了合适的格式化规则,可以对brith属性的日期进行格式验证。

  6. @NotEmpty(message = "{user.email.notEmpty}"):该注解用于验证字符串属性值是否非空。如果email属性为空字符串,则会产生一个指定消息的验证错误。

  7. @Email(message = "{user.email.legal}"):该注解用于验证字符串属性值是否符合Email的格式要求。如果email属性不是合法的Email格式,则会产生一个指定消息的验证错误。

通过在User类的属性上添加这些注解,可以对属性值进行验证,确保满足特定的条件和格式要求。当验证失败时,会生成相应的错误消息供使用者查看和处理。

6、新建一个 ResultVO
@Data
public class ResultVO<T> {// 响应状态码,默认 200private Integer code = HttpStatus.OK.value();private String message;private T data;}

泛型类ResultVO,用于封装接口的响应结果。下面是每个属性的作用:

  1. private Integer code = HttpStatus.OK.value():该属性表示响应的状态码,默认为200,表示成功。HttpStatus.OK.value()是Spring提供的常量,表示HTTP状态码200。

  2. private String message:该属性表示响应的消息,用于描述响应的详细信息。根据具体的业务需求,可以设置不同的消息内容。

  3. private T data:该属性表示响应的数据,是一个泛型类型,可以根据实际情况指定具体的数据类型。根据接口的需求,可以将查询结果、返回对象等放在这个属性中。

通过定义这个ResultVO类,可以将接口的响应结果统一封装并返回给前端。它包含了响应的状态码、消息和数据,可以方便地进行操作和处理。在实际使用中,可以根据业务需要对ResultVO类进行扩展,添加更多的属性和方法。

二、使用 @Valid 验证

1、编写 controller 控制器

@RestController
public class UserController {/*** 注册日期格式化器* @param binder*/@InitBinderpublic void regFormatter (WebDataBinder binder){binder.addCustomFormatter(new DateFormatter("yyyy-MM-dd"));}/*** @Valid 注解:标注的参数实体要参与 Bean 验证* 通常放在实体上** @param user* @param result 这个结果集存放了验证信息,如果这个校验未通过*               就从这个结果集中获取验证消息* @return*/@PostMapping("/add")public ResultVO add(@Valid User user, BindingResult result) throws JsonProcessingException {ResultVO vo = new ResultVO();// 先验证是否校验通过,如果存在错误消息则表示未通过if ( result.hasErrors() ){// 创建一个 map 来保存这些错误信息Map<String,String> error = new HashMap<>();// 循环遍历所有的错误信息result.getFieldErrors().forEach(fieldError -> {// 以字段名作为 key ,错误信息作为 value 保存到 map 中error.put(fieldError.getField(), fieldError.getDefaultMessage());});// 设置响应状态码vo.setCode(HttpStatus.INTERNAL_SERVER_ERROR.value());// 将 map 序列化成 JSON 字符串String messages = new ObjectMapper().writeValueAsString(error);vo.setMessage(messages);// 直接将 errors 保存到 data 中
//            vo.setData(Keymap);return vo;}return vo;}}

UserController类,用于处理用户相关的请求。下面是每个方法的作用:

  1. regFormatter方法:该方法使用@InitBinder注解,用于注册日期格式化器。在这个例子中,通过调用binder.addCustomFormatter(new DateFormatter("yyyy-MM-dd"))方法,将日期格式化器注册到WebDataBinder中,指定日期的格式为"yyyy-MM-dd"。

  2. add方法:该方法使用@PostMapping("/add")注解,用于处理添加用户的请求。在这个例子中,通过@Valid注解标注了User参数,表示要对User对象进行Bean验证。同时,该方法还接受一个BindingResult参数,用于存放验证结果的信息。

    • 首先,通过创建一个ResultVO对象vo来存放响应结果。
    • 然后,通过result.hasErrors()判断是否存在验证错误消息。如果存在错误,则进行错误处理。
    • 在错误处理中,创建一个Map<String, String>对象error来保存错误信息,循环遍历所有的错误信息,并将字段名作为键,错误信息作为值,保存到error中。
    • 设置响应状态码为500(HttpStatus.INTERNAL_SERVER_ERROR.value()),并将error对象序列化为JSON字符串,赋给vomessage属性。
    • 如果需要,可以将error对象直接保存到vodata属性中。
    • 最后,返回vo对象作为响应结果。

通过定义这个UserController类,并使用注解进行日期格式化注册和Bean验证,可以处理用户添加请求,并根据验证结果返回相应的响应结果。在实际使用中,可以根据业务需求对方法进行扩展和优化。

2、配置 dispatcher-servlet.xml 
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:context="http://www.springframework.org/schema/context"xmlns:mvc="http://www.springframework.org/schema/mvc"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/mvc https://www.springframework.org/schema/mvc/spring-mvc.xsd"><!-- 扫描 --><context:component-scan base-package="edu.nf.ch07"/><!-- mvc 注解驱动 注意:需要重新引用验证器的 id  --><mvc:annotation-driven validator="validator"/><!-- 默认 servlet 处理静态资源 --><mvc:default-servlet-handler/><!-- 内部资源视图解析器 --><bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"><property name="prefix" value="/WEB-INF/jsp/"/><property name="suffix" value=".jsp"/></bean><!-- 配置资源消息 --><bean id="messageSource" class="org.springframework.context.support.ReloadableResourceBundleMessageSource"><!-- 配置资源文件的路径 --><property name="basenames"><list><!-- 指定资源文件的位置,注意,不需要写文件后缀 --><value>classpath:message</value></list></property><!-- 指定编码格式 --><property name="defaultEncoding" value="utf-8"/></bean><!-- 配置校验器 --><bean id="validator" class="org.springframework.validation.beanvalidation.LocalValidatorFactoryBean"><!-- 设置校验器的提供方 --><property name="providerClass" value="org.hibernate.validator.HibernateValidator"/><!-- 引用资源文件 --><property name="validationMessageSource" ref="messageSource"/></bean></beans>

每个属性的作用:

  1. <context:component-scan base-package="edu.nf.ch07"/>:该配置用于告诉Spring扫描指定包下的类,以便自动注册为Spring的组件(如控制器、服务等)。

  2. <mvc:annotation-driven validator="validator"/>:该配置启用Spring MVC对注解驱动的支持,其中validator属性指定用于处理验证的验证器Bean的ID。它会自动将验证错误与错误消息关联起来。

  3. <mvc:default-servlet-handler/>:该配置用于将静态资源的请求交给默认的Servlet处理,而不是通过DispatcherServlet处理。这样可以让静态资源(如CSS、JavaScript等)能够直接访问。

  4. <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">:这是一个内部资源视图解析器的配置,用于解析并返回JSP视图的路径。prefix属性指定JSP文件所在的目录,suffix属性指定JSP文件的后缀名。

  5. <bean id="messageSource" class="org.springframework.context.support.ReloadableResourceBundleMessageSource">:该配置用于配置资源消息,即国际化文本消息的配置。basenames属性指定资源文件的位置,defaultEncoding属性指定资源文件的编码格式。

  6. <bean id="validator" class="org.springframework.validation.beanvalidation.LocalValidatorFactoryBean">:这是一个校验器的配置,用于对验证框架进行配置。providerClass属性指定所使用的验证器的提供者,validationMessageSource属性引用了前面配置的消息资源。

 3、新建一个 reg.html 页面提交数据
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>用户注册</title><script src="js/JQuery文件.txt.js"></script>
</head>
<body>
<!-- 这个 div 用于显示验证的消息 -->
<div id="msg"></div><form id="f1">Name:<input type="text" name="userName"><br>Age:<input type="text" name="age"><br>Birth:<input type="text" name="birth"><br>Email:<input type="text" name="email"><br><input type="button" value="注册"><br>
</form><script>$(function () {$(':button').on('click',function () {$.ajax({url:'../add',type:'post',data:$('#f1').serialize(),success:function ( result ) {if ( result.code === 500 ){$('#msg').empty();let errors = result.message;// 将 json 字符串转换为 json 对象errors = $.parseJSON(errors);// 循环遍历错误消息放到 div 中$.each(errors,function (key,val) {$('#msg').append(val + '<br>')})} else{$('#msg').empty();}}})})})
</script></body>
</html>

这段代码是一个简单的前端页面,用于用户注册。它使用了jQuery库来处理表单提交和异步请求。

在页面中,有一个<form>标签,包含了用户需要填写的字段,如姓名、年龄、出生日期和电子邮件等。其中,每个字段都使用了<input>标签,并通过name属性指定了字段名。

在页面底部的JavaScript代码中,使用了jQuery的$(function(){})方法,确保文档加载完成后执行相应代码。首先,通过$(':button')选择器选中所有<input type="button">元素,并绑定了一个点击事件处理函数。当按钮被点击时,触发了一个AJAX请求。

AJAX请求使用了$.ajax()方法,其中传入了请求的URL、请求类型和数据等参数。请求的URL为../add,表示将请求发送到服务器上的/add路径。请求类型为POST,即使用HTTP POST方法发送请求。数据使用$('#f1').serialize()方法对表单进行序列化,将表单中的输入值转换为URL编码的字符串。

当请求成功返回时,会执行success回调函数。在回调函数中,首先判断响应的result对象中的code属性是否为500(表示服务器内部错误)。如果是500,则说明存在验证错误消息。然后,通过$('#msg')选中页面中的<div>元素,并清空其内容。接下来,将响应的message属性(包含错误信息的JSON字符串)转换为JSON对象,并使用$.each()方法遍历错误信息,将每条错误信息添加到<div>元素中显示出来。

总体而言,这段代码实现了一个简单的用户注册页面,并使用AJAX请求将表单数据发送到服务器进行处理。在服务器端,通过验证结果返回不同的响应。在前端页面中,通过jQuery来处理和显示服务器返回的验证错误信息。

4、运行效果

我们测试的结果,刚开始什么都不写的时候都提示了为空,当一个一个输进去后,就没有了提示,不满足条件的也会提示。 

 三、使用 Springmvc 表单验证的好处

Spring MVC的表单验证功能具有以下几个好处:

  1. 提高数据的有效性和完整性:通过表单验证,可以确保用户提交的数据符合预期的格式、长度、范围等要求。这可以避免因为不合法的数据导致程序出错或产生意外结果。有效的表单验证可以提高数据的质量和可靠性。

  2. 减少后端开发工作量:使用Spring MVC的表单验证,可以在后端自动执行验证逻辑,而无需手动编写大量的验证代码。框架会根据注解配置或自定义验证规则,自动完成验证过程。这样可以极大地减少开发人员的工作量,并提高开发效率。

  3. 支持代码重用和模块化:通过使用Spring MVC的表单验证,可以将验证逻辑从业务逻辑中分离出来,形成独立的验证模块。这样可以提高代码的重用性和可维护性,减少重复代码的编写。同时,验证模块可以针对不同的实体对象进行复用,进一步提高开发效率。

  4. 提供友好的用户反馈:通过表单验证,可以及时地向用户提供错误信息或者警告信息,帮助用户发现并纠正输入错误。这可以提升用户体验,让用户更加轻松地填写正确的数据。同时,通过在前端页面显示验证错误信息,可以帮助开发人员快速定位并修复问题。

  5. 防止安全漏洞和数据损坏:表单验证还可以用于防范安全漏洞,例如输入校验、XSS(跨站脚本攻击)预防等。通过限制输入的数据格式和内容,可以有效地防止恶意用户提交恶意代码或其他危险数据,提高应用程序的安全性。

总之,Spring MVC的表单验证功能在提高数据质量、减少开发工作量、提供友好用户反馈、增强应用程序安全性等方面具有重要的好处。它是开发Web应用程序时不可或缺的一部分,能够提升开发效率和用户体验。

四、gitee 案例

地址:CH07 · qiuqiu/SpringMVC - 码云 - 开源中国 (gitee.com)

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

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

相关文章

Thread

Thread 线程启动线程第一种创建线程线程的第二种创建方式使用匿名内部类完成线程的两种创建 Thread API线程的优先级线程提供的静态方法守护线程用户线程和守护线程的区别体现在进程结束时 多线并发安全问题同步块 线程 启动线程 启动线程:调用线程的start方法,而不是直接调用…

lua-resty-request库写入爬虫ip实现数据抓取

根据提供的引用内容&#xff0c;正确的库名称应该是lua-resty-http&#xff0c;而不是lua-resty-request。使用lua-resty-http库可以方便地进行爬虫&#xff0c;需要先安装OpenResty和lua-resty-http库&#xff0c;并将其引入到Lua脚本中。然后&#xff0c;可以使用lua-resty-h…

【网络协议】聊聊http协议

当我们输入www.baidu.com的时候&#xff0c;其实是先将baidu.com的域名进行DNS解析&#xff0c;转换成对应的ip地址&#xff0c;然后开始进行基于TCP构建三次握手的连接&#xff0c;目前使用的是1.1 默认是开启了keep-Alive。可以在多次请求中进行连接复用。 HTTP 请求的构建…

【k8s】pod详解

一、Pod介绍 1、Pod的基础概念 Pod是kubernetes中最小的资源管理组件&#xff0c;Pod也是最小化运行容器化应用的资源对象&#xff0c;一个pod代表着集群中运行的一个进程。kubernetes中其它大多数组件都是围绕着pod来进行支持和扩展pod功能的。 例如&#xff0c;用于管理po…

计算机基础知识41

前端 # 前端是所有跟用户直接打交道 比如&#xff1a;PC页面、手机页面、汽车显示屏&#xff0c;肉眼可以看见的 # 后端&#xff1a;一堆代码&#xff0c;用户不能够直接看到&#xff0c;不直接与用户打交道 常见的后端&#xff1a;Python、Java、Go等 # 学了前端就可以做全栈…

算法题:二叉树的层序遍历

层序遍历&#xff0c;看似简单&#xff0c;实则陷阱很多&#xff0c;怪不得该题目被认定为中等难度题。此处运用了迭代求解法。&#xff08;完整题目附在了最后&#xff09; # Definition for a binary tree node. class TreeNode:def __init__(self, val0, leftNone, rightNon…

Vectrosity 插件使用

1 下载 https://download.csdn.net/download/moonlightpeng/88490704?spm1001.2014.3001.5503 2 使用&#xff0c;目前在2020.3.3上测试可以 导入时选5.6 再导入demo

FlinkCDC系列:通过skipped.operations参数选择性处理新增、更新、删除数据

在flinkCDC源数据配置&#xff0c;通过debezium.skipped.operations参数控制&#xff0c;配置需要过滤的 oplog 操作。操作包括 c 表示插入&#xff0c;u 表示更新&#xff0c;d 表示删除。默认情况下&#xff0c;不跳过任何操作&#xff0c;以逗号分隔。配置多个操作&#xff…

四、[mysql]索引优化-1

目录 前言一、场景举例1.联合索引第一个字段用范围查询不走索引(分情况&#xff09;2.强制走指定索引3.覆盖索引优化4.in和or在表数据量比较大的情况会走索引&#xff0c;在表记录不多的情况下会选择全表扫描5.like 后% 一般情况都会走索引(索引下推) 二、Mysql如何选择合适的索…

(三)上市企业实施IPD成功案例分享之——五菱

2022年对汽车产业而言是极为不平凡的一年。这一年&#xff0c;企业受到疫情反复、芯片短缺、原材料价格上涨等负面因素影响&#xff0c;汽车产业的变革持续加速。面对变革与挑战&#xff0c;五菱汽车仍逆势交出一份超出市场预期的成绩单。上半年&#xff0c;五菱发布2022年财报…

2023年【熔化焊接与热切割】考试题及熔化焊接与热切割模拟考试

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 2023年【熔化焊接与热切割】考试题及熔化焊接与热切割模拟考试&#xff0c;包含熔化焊接与热切割考试题答案和解析及熔化焊接与热切割模拟考试练习。安全生产模拟考试一点通结合国家熔化焊接与热切割考试最新大纲及熔…

ASO优化之如何制作Google Play的长短描述

应用的描述以及标题和图标是元数据中最关键的元素&#xff0c;可以影响用户是否决定下载我们的应用程序。简短描述的长度限制为80个字符&#xff0c;它提供了更多的有关应用背景信息的机会。 1、简短描述帮助用户快速了解我们应用。 确保内容丰富的同时&#xff0c;保持简洁和…

Redis与Mysql的数据一致性(双写一致性)

双写一致性&#xff1a;当修改了数据库的数据也要同时的更新缓存的数据&#xff0c;使缓存和数据库的数据要保持一致。 一般是在写数据的时候添加延迟双删的策略 先删缓存 再修改数据 延迟一段时间后再次删除缓存 这种方式其实不是很靠谱 一致性要求高 共享锁&#xff1a;读…

Android裁剪图片之后无法加载的问题

适配Android11之后更改了图片保存目录&#xff0c;导致裁剪之后图片一直无法加载&#xff08;fileNotfound&#xff09; 最主要的问题在于保存裁剪文件的目录不能为私有目录&#xff0c;因为裁剪工具是系统工具&#xff0c;无法直接访问项目本身的私有目录。 解决办法&#x…

基于适应度相关算法的无人机航迹规划-附代码

基于适应度相关算法的无人机航迹规划 文章目录 基于适应度相关算法的无人机航迹规划1.适应度相关搜索算法2.无人机飞行环境建模3.无人机航迹规划建模4.实验结果4.1地图创建4.2 航迹规划 5.参考文献6.Matlab代码 摘要&#xff1a;本文主要介绍利用适应度相关算法来优化无人机航迹…

分布式:一文吃透分布式事务和seata事务

目录 一、事务基础概念二、分布式事务概念什么是分布式事务分布式事务场景CAP定理CAP理论理解CAPCAP的应用 BASE定理强一致性和最终一致性BASE理论 分布式事务分类刚性事务柔性事务 三、分布式事务解决方案方案汇总XA规范方案1&#xff1a;2PC第一阶段&#xff1a;准备阶段第二…

阿里云急了,云服务器老用户优惠价格99元一年!

2023阿里云服务器双11优惠价格99元一年经济型e实例&#xff0c;并且续费不涨价&#xff0c;云服务器ECS-经济型e实例2核2G配置、3M带宽、40G ESSD entry系统盘优惠价99元一年&#xff0c;原价956.64元/年&#xff0c;可用于中小型网站建设、开发测试、小程序或app搭建&#xff…

用rust写web服务器笔记(11/1)

文章目录 一、创建一个具有监听链接功能web服务器二、读取请求内容三、编写web服务器返回网页(编写响应)四、编写web服务器有条件的返回网页五、编写多线程的web服务器六、用线程池实现web服务器七、实现线程池清除的web服务器八、读取文件 rust官网文档地址&#xff1a;https:…

windows10下Node.js安装教程

文章目录 windows10下Node.js安装教程下载安装包执行安装检查环境变量测试 windows10下Node.js安装教程 下载安装包 官网 执行安装 检查环境变量 系统已经为Node.js添加了相应的系统环境变量 测试 打开命令窗口 使用命令&#xff1a; END

从小白到大牛:Linux嵌入式系统开发的完整指南

Linux嵌入式系统开发一直是一个激动人心的领域&#xff0c;吸引着越来越多的开发者。无论你是初学者还是已经有一些经验的开发者&#xff0c;本文将为你提供从小白到大牛的完整指南&#xff0c;帮助你掌握Linux嵌入式系统开发的关键概念和技能。我们将深入探讨Linux内核、设备驱…