当前位置: 首页 > news >正文

Spring 事务管理详解及使用

在这里插入图片描述

✅作者简介:2022年博客新星 第八。热爱国学的Java后端开发者,修心和技术同步精进。
🍎个人主页:Java Fans的博客
🍊个人信条:不迁怒,不贰过。小知识,大智慧。
💞当前专栏:SSM 框架从入门到精通
✨特色专栏:国学周更-心性养成之路
🥭本文内容:一文吃透 Spring 中的IOC和DI

文章目录

  • Spring 事务管理接口
    • 1、事务管理器接口 PlatformTransactionManager
    • 2、事务定义接口 TransactionDefinition
  • Spring 事务管理的实现方法
    • 1、没有事务管理的情况分析
    • 2、通过配置 XML 实现事务管理
    • 3、利用注解实现事务管理
    • 4、在业务层实现事务管理

在这里插入图片描述

  事务(Transaction)是访问数据库的一个操作序列,这些操作要么都做,要么都不做,是一个不可分割的工作单元。通过事务,数据库能将逻辑相关的一组操作绑定在一起,以便保持数据的完整性。

事务有4个重要特性,简称 ACID

  • A:Automicity原子性,即事务中的所有操作要么全部执行,要么全部不执行。
  • C:Consistency一致性,事务执行的结果必须是使数据库从一个一致状态变到另一个一致状态。
  • I:Isolation隔离性,即一个事务的执行不能被另一个事务影响。
  • D:Durabillity持久性,即事务提交后将被永久保存。

  在Java EE开发中,事务原本属于 Dao 层中的范畴,但一般情况下需要将事务提升到业务层(Service层),以便能够使用事务的特性来管理具体的业务。

Spring 事务管理接口

  Spring 的事务管理,主要用到两个事务相关的接口。

1、事务管理器接口 PlatformTransactionManager

  事务管理器接口 PlatformTransactionManager 主要用于完成事务的提交、回滚,及获取事务的状态信息。PlatformTransactionManager 接口有两个常用的实现类:

  • DataSourceTransactionManager实现类:使用JDBC或MyBatis进行数据持久化时使用。
  • HibernateTransactionManager实现类:使用Hibernate进行数据持久化时使用。

  关于Spring的事务提交与回滚方式,默认是:发生运行时异常时回滚,发生受检查异常时提交, 也就是说程序抛出runtime异常的时候才会进行回滚,其他异常不回滚。

2、事务定义接口 TransactionDefinition

  事务定义接口 TransactionDefinition 中定义了事务描述相关的三类常量:事务隔离级别常量事务传播行为常量事务默认超时时限常量,及对它们的操作。

【1】事务隔离级别常量

  在应用程序中,多个事务并发运行,操作相同的数据,可能会引起脏读,不可重复读,幻读等问题 。

  • 脏读(Dirty reads):第一个事务访问并改写了数据,尚未提交事务,这时第二个事务进来了,读取了刚刚改写的数据,如果这时第一个事务回滚了,这样第二个事务读取到的数据就是无效的“脏数据”。
  • 不可重复读(Nonrepeatable read):第一个事务在其生命周期内多次查询同一个数据,在两次查询之间,第二个事务访问并改写了该数据,导致第一个事务两次查询同一个数据得到的结果不一样。
  • 幻读(Phantom read)——幻读与不可重复读类似。它发生在第一个事务在生命周期进行了两次按同一查询条件查询数据,第一次按该查询条件读取了几行数据,这时第二个事务进来了,插入或删除了一些数据时,然后第一个事务再次按同一条件查询,发现多了一些原本不存在的记录或者原有的记录不见了。

  为了解决并发问题,TransactionDefinition 接口定义了5个事务隔离常量如下:

  • DEFAULT:采用数据库 默认 的事务隔离级别。不同数据库不一样,MySql的默认为 REPEATABLE_READ(可重复读);Oracle默认为READ_COMMITTED(读已提交)。
  • READ_UNCOMMITTED读未提交。允许另外一个事务读取到当前事务未提交的数据,隔离级别最低,未解决任何并发问题,会产生脏读,不可重复读和幻像读。
  • READ_COMMITTED读已提交,被一个事务修改的数据提交后才能被另外一个事务读取,另外一个事务不能读取该事务未提交的数据。解决脏读,但还存在不可重复读与幻读。
  • REPEATABLE_READ可重复读。解决了脏读、不可重复读,但还存在幻读。
  • SERIALIZABLE串行化。按时间顺序一一执行多个事务,不存在并发问题,最可靠,但性能与效率最低。

【2】事务传播行为常量

  事务传播行为是指处于不同事务中的方法在相互调用时,执行期间事务的维护情况。例如,当一个事务方法B调用另一个事务方法A时,应当明确规定事务如何传播,比方可以规定A方法继续在B方法的现有事务中运行,也可以规定A方法开启一个新事务,在新事务中运行,现有事务先挂起,等A方法的新事务执行完毕后再恢复。TransactionDefinition 接口一共定义了 七种 传播行为常量说明如下。

  • PROPAGATION_ REQUIRED:指定的方法必须在事务内执行。若当前存在事务,就加入到当前事务中;若当前没有事务,则创建一个新事务。这种传播行为是最常见的选择,也是 Spring 默认的事务传播行为。如该传播行为加在actionB ()方法上,该方法将被actionA ()调用,若actionA ()方法在执行时就是在事务内的,则actionB ()方法的执行也加入到该事务内执行。若actionA ()方法没有在事务内执行,则actionB ()方法会创建一个事务,并在其中执行。
  • PROPAGATION_ SUPPORTS:指定的方法支持当前事务,但若当前没有事务,也可以以非事务方式执行。
  • PROPAGATION_ MANDATORY:指定的方法必须在当前事务内执行,若当前没有事务,则直接抛出异常。
  • PROPAGATION_ REQUIRES_NEW:总是新建一个事务,若当前存在事务,就将当前事务挂起,直到新事务执行完毕。
  • PROPAGATION_ NOT_SUPPORTED:指定的方法不能在事务环境中执行,若当前存在事务,就将当前事务挂起。
  • PROPAGATION_ NEVER:指定的方法不能在事务环境下执行,若当前存在事务,就直接抛出异常。
  • PROPAGATION_ NESTED:指定的方法必须在事务内执行。若当前存在事务,则在嵌套事务内执行;若当前没有事务,则创建一个新事务。

【3】默认事务超时时限

  常量 TIMEOUT_DEFAULT 定义了事务底层默认的超时时限,及不支持事务超时时限设置的none值。该值一般使用默认值即可。

Spring 事务管理的实现方法

  Spring 支持编程式事务声明式事务

  编程式事务 直接在主业务代码中精确定义事务的边界,事务以硬编码的方式嵌入到了主业务代码里面,好处是能提供更加详细的事务管理,但由于编程式事务主业务与事务代码混在一起,不易分离,耦合度高,不利于维护与重用。

  声明式事务 则基于 AOP 方式,能将主业务操作与事务规则进行解耦。能在不影响业务代码的具体实现情况下实现事务管理。所以比较常用的是声明式事务。声明式事务又有两种具体的实现方式:基于XML配置文件的方式基于注解的方式

1、没有事务管理的情况分析

  项目案例: 模拟支付宝转账,张三、李四原本各有账户余额 2000 元,张三转账 500 元给李四,但转账过程中出现了异常。

  实现步骤:

【1】在 MySQL 中创建数据库表,代码如下:

create table alipay(aliname varchar (60),amount double
);

【2】在 dao 层创建 IAccountDao 接口,代码如下

package com.hh.dao;public interface AlipayDao {public void transfer(String fromA,String toB,int amount);
}

【3】创建 IAccountDao 接口的实现类 IAccountDaoImpl,代码如下:

package com.hh.dao;import org.springframework.jdbc.core.JdbcTemplate;public class AlipayDaoImpl implements AlipayDao{JdbcTemplate jdbcTemplate;	public JdbcTemplate getJdbcTemplate() {return jdbcTemplate;}public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {this.jdbcTemplate = jdbcTemplate;}@Overridepublic void transfer(String fromA, String toB, int amount) {jdbcTemplate.update("update alipay set amount=amount-? where aliname=?",amount,fromA);Integer.parseInt("a");jdbcTemplate.update("update alipay set amount=amount+? where aliname=?",amount,toB);}
}

  这个 transfer 方法主要实现两个操作:

  操作一:转出操作,张三的账户钱减少;
  操作二:转入操作,里斯的账户钱增加;

  但两个操作中间模拟出现了差错(异常),这将导致张三的钱少了,而李四的钱却没有增加。

【4】添加 Spring 配置文件,代码如下:

<?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:aop="http://www.springframework.org/schema/aop"xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/aophttp://www.springframework.org/schema/aop/spring-aop.xsd">	<!-- 配置数据源 --><bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource"><property name="driverClassName"><value>com.mysql.jdbc.Driver</value></property><property name="url"><value>jdbc:mysql://localhost:3306/usersdb </value></property><property name="username"><value>root</value></property><property name="password"><value>root</value></property></bean>	<!-- 配置jdbcTemplate模板 注入dataSource --><bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"><property name="dataSource" ref="dataSource" />	</bean>	<!-- 配置DAO,注入jdbcTemplate属性值 --><bean id="userDao" class="com.hh.dao.UserDaoImpl"><property name="jdbcTemplate" ref="jdbcTemplate"/></bean>	<!-- 配置DAO,注入jdbcTemplate属性值 --><bean id="alipayDao" class="com.hh.dao.AlipayDaoImpl"><property name="jdbcTemplate" ref="jdbcTemplate"/></bean></beans>

【5】添加测试类 TestAlipay

package com.hh.test;import java.util.List;import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import com.hh.dao.AlipayDao;public class TestAlipay {public static void main(String[] args) {ApplicationContext context=new ClassPathXmlApplicationContext("applicationContext.xml");AlipayDao alipayDao=(AlipayDao) context.getBean("alipayDao");alipayDao.transfer("张三", "李四", 500);}
}

2、通过配置 XML 实现事务管理

  下面进行事务管理方面的改进,目标是把类 AlipayDaoImpl 里的整个 transfer( ) 方法作为事务管理,这样 transfer( ) 里的所有操作(包括转出/转入操作)都纳入同一个事务,从而使 transfer( ) 里的所有操作要么一起成功,要么一起失败。这里利用了 Spring 的事务管理机制进行处理。

项目案例: 模拟支付宝转账,张三、李四原本各有账户余额 2000 元,张三转账 500 元给李四,但转账过程中间出现异常,导致数据不一致 ,现应用 Spring 的事务管理,配置 XML,避免不一致的情况。

实现步骤:

【1】修改 Spring 配置文件,内容如下:

在这里插入图片描述

<?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:aop="http://www.springframework.org/schema/aop"xmlns:tx="http://www.springframework.org/schema/tx"xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsdhttp://www.springframework.org/schema/aophttp://www.springframework.org/schema/aop/spring-aop.xsd">	<!-- 配置数据源 --><bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource"><property name="driverClassName"><value>com.mysql.jdbc.Driver</value></property><property name="url"><value>jdbc:mysql://localhost:3306/usersdb </value></property><property name="username"><value>root</value></property><property name="password"><value>root</value></property></bean>	<!-- 配置jdbcTemplate模板 注入dataSource --><bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"><property name="dataSource" ref="dataSource" />	</bean>	<!-- 配置DAO,注入jdbcTemplate属性值 --><bean id="alipayDao" class="com.lifeng.dao.AlipayDaoImpl"><property name="jdbcTemplate" ref="jdbcTemplate"/></bean><!-- 定义事务管理器 --><bean id="txManager"class="org.springframework.jdbc.datasource.DataSourceTransactionManager"><property name="dataSource" ref="dataSource" /></bean><!-- 编写事务通知 --><tx:advice id="txAdvice" transaction-manager="txManager"><tx:attributes><tx:method name="*" propagation="REQUIRED" 	isolation="DEFAULT"	 read-only="false" />			<!-- <tx:method name="save*" propagation="REQUIRED" /><tx:method name="add*" propagation="REQUIRED" /><tx:method name="insert*" propagation="REQUIRED" /><tx:method name="delete*" propagation="REQUIRED" /><tx:method name="update*" propagation="REQUIRED" /><tx:method name="search*" propagation="SUPPORTS"  read-only="true"/><tx:method name="select*" propagation="SUPPORTS"  read-only="true"/><tx:method name="find*" propagation="SUPPORTS"  read-only="true"/><tx:method name="get*" propagation="SUPPORTS"  read-only="true"/> -->	</tx:attributes></tx:advice>	<!-- 编写AOP,让spring自动将事务切入到目标切点 --><aop:config><!-- 定义切入点 --><aop:pointcut id="txPointcut"expression="execution(* com.lifeng.dao.*.*(..))" /><!-- 将事务通知与切入点组合 --><aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut" /></aop:config>	</beans>

  这里可以把事务功能理解为切面,通过aop配置实现事务(切面)自动切入到切入点(目标方法),从而将目标方法(切入点)纳入事务管理,而目标方法本身可以不用管事务,专心做自已的主业务功能就行了

【2】其它程序代码不变,运行测试。

  测试时尽管转账中间出现了异常,但是张三、李四的钱都没变化,保持了一致性,这样就达到了目的,证明了 transfer 方法中的两个操作都纳入了同一个事务。发生异常时,事务回滚,保证了数据的一致性。

  上面配置中:

<tx:method name="*" propagation="REQUIRED" isolation="DEFAULT"	 read-only="false" />

  表示匹配的切点方法都进行事务管理,这里*表示匹配所有切点方法,propagation="REQUIRED"表示匹配的切点方法必须在事务内执行,isolation="DEFAULT"表示事务隔离级别默认,对于MySQL数据库,隔离级别为REPEATABLE_READ(可重复读)。read-only="false"表示非只读。

  这个配置粒度太大,所有方法都同一种事务管理模式,要想不同的方法实现不一样的事务管理,还得细化配置。项目中常见的细化配置如下面代码所示。

	<!-- 编写通知 --><tx:advice id="txAdvice" transaction-manager="txManager"><tx:attributes><tx:method name="save*" propagation="REQUIRED" /><tx:method name="add*" propagation="REQUIRED" /><tx:method name="insert*" propagation="REQUIRED" /><tx:method name="delete*" propagation="REQUIRED" /><tx:method name="update*" propagation="REQUIRED" /><tx:method name="search*" propagation="SUPPORTS"  read-only="true"/><tx:method name="select*" propagation="SUPPORTS"  read-only="true"/><tx:method name="find*" propagation="SUPPORTS"  read-only="true"/><tx:method name="get*" propagation="SUPPORTS"  read-only="true"/></tx:attributes></tx:advice>

  这样,不同的方法匹配不同的事务管理模式。

  <tx:method name=“save*” propagation=“REQUIRED” />表示凡是以save开头的切点方法必须在事务内执行,其他增删改都一样的意思,查的话就不同<tx:method name=“select*” propagation=“SUPPORTS” read-only=“true”/>表示心是以select开头的切点方法支持当前事务,但若当前没有事务,也可以以非事务方式执行,read-only="true"表示只读,其他几个类似。

3、利用注解实现事务管理

  上面是利用XML配置文件实现事务管理的办法,下面来学习用注解实现事务管理。

  使用@Transactional注解在类或方法上,即可实现事务管理。 @Transactional注解的属性有下面这些(可选):

  • propagation:用于设置事务传播的属性,该属性类型为propagation枚举,默认值为Propagation.REQUIRED。
  • isolation:用于设置事务的隔离级别,该属性类型为Isolation枚举,默认值为Isolation.DEFAULT。
  • readOnly:用于设置该方法对数据库的操作是否是只读的,该属性为boolean,默认值false。
  • timeout:用于设置本操作与数据库连接的超时时限。单位为秒,类型为int,默认值为-1,即没有时限。
  • rollbackFor:指定需要回滚的异常类,类型为 Class[],默认值为空数组。当然,若只有一个异常类时,可以不使用数组。
  • rollbackForClassName:指定需要回滚的异常类类名,类型为 String[],默认值为空数组。当然,若只有一个异常类时,可以不使用数组。
  • noRollbackFor:指定不需要回滚的异常类。类型为Class[],默认值为空数组。当然,若只有一个异常类时,可以不使用数组。
  • noRollbackForClassName:指定不需要回滚的异常类类名。类型为String[],默认值为空数组。当然,若只有一个异常类时,可以不使用数组。

  需要注意的是,@Transactional 若用在方法上,只能用于public方法上。对于其他非 public方法,如果加上了注解 @Transactional,虽然 Spring 不会报错,但不会将指定事务织入到该方法中。因为Spring会忽略掉所有非public方法上的 @Transaction 注解。若 @Transaction 注解在类上,则表示该类上所有的方法均将在执行时织入事务。

项目案例: 模拟支付宝转账,张三李四原本各有账户余额2000元,张三转账500元给李四,但转账过程中间出现异常,应用spring的事务管理,使用注解,避免不一致的情况。

实现步骤:

【1】修改 Spring 配置文件如下:

	<!-- 定义事务管理器 --><bean id="txManager"class="org.springframework.jdbc.datasource.DataSourceTransactionManager"><property name="dataSource" ref="dataSource" /></bean><!-- 开启事务注解驱动 --><tx:annotation-driven transaction-manager="txManager"/></beans>

  可以发现,配置文件比之前简化了很多,事务方面,只需定义好事务管理器,再开启事务注解驱动就行了。其他的交给注解来解决。

【2】利用 @Transactional 注解修改转账方法。

  @Transactional 既可以用来修饰类,也可以修饰方法,如果修饰类,则表示事务的设置对整个类的所有方法都起作用,如果修饰在方法上,则只对该方法起作用,关键代码如下。

public class AlipayDaoImpl implements AlipayDao{
@Override@Transactional(propagation=Propagation.REQUIRED,isolation=Isolation.DEFAULT,readOnly=false)public void transfer(String fromA, String toB, int amount) {jdbcTemplate.update("update alipay set amount=amount-? where aliname=?",amount,fromA);Integer.parseInt("a");jdbcTemplate.update("update alipay set amount=amount+? where aliname=?",amount,toB);}

  将transfer方法注解为事务。

【3】运行测试,发现数据库同样没改变,所以注解事务起到作用了。

4、在业务层实现事务管理

  上面的案例是在DAO层实现事务管理,相对简单一些,但实际上开发时需要在业务层实现事务管理,而不是在DAO层,为此,项目修改如下,特别要注意在业务层的事务管理实现。

项目案例: 模拟支付宝转账,张三李四原本各有账户余额2000元,张三转账500元给李四,但转账过程中间出现异常,在业务层应用spring的事务管理,配置xml,避免不一致的情况。

实现步骤:

【1】修改DAO层,转出转入分拆成两个方法。

	@Overridepublic void tranferFrom(String fromA,int amount){jdbcTemplate.update("update alipay set amount=amount-? where aliname=?",amount,fromA);}@Overridepublic void tranferTo(String toB,int amount){jdbcTemplate.update("update alipay set amount=amount+? where aliname=?",amount,toB);}

【2】新建包com.hh.service,创建业务层AlipayService.java类,代码如下:

public class AlipayService {private AlipayDao alipayDao;public void transfer(String fromA, String toB, int amount) {alipayDao.tranferFrom(fromA, amount);Integer.parseInt("a");alipayDao.tranferTo(toB, amount);}
}

  上述代码相当于把有异常问题的 transfer( ) 方法迁移到业务层中来。

【3】修改配置文件。关键配置如下:

	<!-- 定义事务管理器 --><bean id="txManager"class="org.springframework.jdbc.datasource.DataSourceTransactionManager"><property name="dataSource" ref="dataSource" /></bean><!-- 编写事务通知 --><tx:advice id="txAdvice" transaction-manager="txManager"><tx:attributes><tx:method name="*" propagation="REQUIRED" 	isolation="DEFAULT"	 read-only="false" />		</tx:attributes></tx:advice>	<!-- 编写AOP,让spring自动将事务切入到目标切点 --><aop:config><!-- 定义切入点 --><aop:pointcut id="txPointcut"expression="execution(* com.lifeng.service.*.*(..))" /><!-- 将事务通知与切入点组合 --><aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut" /></aop:config>

【4】修改测试类,代码如下:

public class TestAlipay {public static void main(String[] args) {ApplicationContext context=new ClassPathXmlApplicationContext("applicationContext.xml");AlipayService alipayService=(AlipayService) context.getBean("alipayService");alipayService.transfer("张三", "李四", 500);}
}

  测试结束,数据库的数据保持不变,证明事务管理成功。


  码文不易,本篇文章就介绍到这里,如果想要学习更多Java系列知识点击关注博主,博主带你零基础学习Java知识。与此同时,对于日常生活有困扰的朋友,欢迎阅读我的第四栏目:《国学周更—心性养成之路》,学习技术的同时,我们也注重了心性的养成。

在这里插入图片描述

相关文章:

Spring 事务管理详解及使用

✅作者简介&#xff1a;2022年博客新星 第八。热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;Java Fans的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 &#x1f49e;当前专栏…...

LeetCode 232.用栈实现队列

请你仅使用两个栈实现先入先出队列。队列应当支持一般队列支持的所有操作&#xff08;push、pop、peek、empty&#xff09;&#xff1a;实现 MyQueue 类&#xff1a;void push(int x) 将元素 x 推到队列的末尾int pop() 从队列的开头移除并返回元素int peek() 返回队列开头的元…...

go面向对象思想封装继承多态

go貌似都没有听说过继承&#xff0c;当然这个继承不像c中通过class类的方式去继承&#xff0c;还是通过struct的方式&#xff0c;所以go严格来说不是面向对象编程的语言&#xff0c;c和java才是&#xff0c;不过还是可以基于自身的一些的特性实现面向对象的功能&#xff0c;面向…...

【网络原理9】HTTP响应篇

在前两篇文章当中&#xff0c;已经分别介绍了HTTP是什么&#xff0c;以及常见的请求头当中的属性。【网络原理7】认识HTTP_革凡成圣211的博客-CSDN博客HTTP抓包&#xff0c;Fiddler的使用https://blog.csdn.net/weixin_56738054/article/details/129148515?spm1001.2014.3001.…...

SpringCloud之Seata(二)

4.Seata如何应用于项目&#xff1f; 安装seata及修改配置 4.1 官网下载Seata安装包 4.2 修改seata/config.txt 4.2.1 修改存储方式 store.db.dbTypemysql store.db.driverClassNamecom.mysql.jdbc.Driver store.db.urljdbc:mysql://你的IP:3306/seata?useUnicodetrue sto…...

【Redis-入门阶段】基本数据结构

Redis支持多种数据结构&#xff0c;包括字符串、列表、哈希、集合和有序集合。这些数据结构在Redis中被称为键值对&#xff0c;其中键是一个字符串&#xff0c;值可以是一个字符串、列表、哈希、集合或有序集合。接下来&#xff0c;我们将详细介绍这些数据结构的使用方法。字符…...

BACnet协议详解————MS/TP物理层,数据链路层和网络层

文章目录写在前面1 物理层2 数据链路层MSTP的流程如下noteMS/TP帧格式3 网络层写在前面 这周加更一篇&#xff0c;来弥补一下之前落下的进度。简单的说两句&#xff0c;之前讲应用层的时候&#xff0c;只是跟官方的手册来同步一下&#xff0c;但是从个人理解来说&#xff0c;自…...

Tomcat

Tomcat 1 简介 1.1 什么是Web服务器 Web服务器是一个应用程序&#xff08;软件&#xff09;&#xff0c;对HTTP协议的操作进行封装&#xff0c;使得程序员不必直接对协议进行操作&#xff0c;让Web开发更加便捷。主要功能是"提供网上信息浏览服务"。 Web服务器是安…...

创客匠人直播:构建公域到私域的用户增长模型

进入知识付费直播带货时代&#xff0c;很多拥有知识技能经验的老师和培训机构吃到了流量红利。通过知识付费直播&#xff0c;老师们可以轻松实现引流、变现&#xff0c;还可以突破时间、地域的限制&#xff0c;为全国各地的学员带来优质的教学服务&#xff0c;因此越来越受到教…...

机试指南

文章目录零、绪论和IDE安装int取值范围常犯的编程小错误一、枚举和模拟 (暴力求解)(一) 枚举1.Reverse函数 求 反序数2.程序出错的原因1.编译错误 (compile)&#xff1a;基本语法错误2.链接错误 (link)&#xff1a;函数名写错了3.运行错误 (run)&#xff1a;结果与预期不符&…...

Android CTA认证设定首选网络类型

需求 硬件只支持4G,过CTA认证时打网络电话,会出现3G网络的选择,会导致过不了,需要禁用3G网络选择功能。 Android 8.1.0 分析 可adb命令查看当前的网络类型 getprop | grep “network” 打印如下: [gsm.network.type]: [LTE,LTE] [ro.telephony.default_network]: [9] …...

Android 动态切换应用图标方案

经常听到大家讨论类似的需求&#xff0c;怀疑大厂是不是用了此方案&#xff0c;据我个人了解&#xff0c;多数头部 app 其实都是发版来更新节假日的 icon。当然本方案也是一种可选的方案&#xff0c;以前我也调研过&#xff0c;存在问题和作者所述差不多&#xff0c;此外原文链…...

SMART PLC斜坡函数功能块(梯形图代码)

斜坡函数Ramp的具体应用可以参看下面的文章链接: PID优化系列之给定值斜坡函数(PLC代码+Simulink仿真测试)_RXXW_Dor的博客-CSDN博客很多变频器里的工艺PID,都有"PID给定值变化时间"这个参数,这里的给定值变化时间我们可以利用斜坡函数实现,当然也可以利用PT1…...

不那么认真的linux复习

这是个不那么认真的linux总结&#xff0c;可能有一些错误 1、linuxkernel&#xff08;内核&#xff09;shell&#xff08;外壳&#xff09;fs&#xff08;文件系统&#xff09;pro/uti/tol&#xff08;应用程序&#xff09; 2、ls&#xff08;列出文件&#xff09; -a&#xf…...

Redis系列文章总纲

跟着老万学Redis 前言 从事开发工作这么久&#xff0c;很多核心技术其实都还只是局限在满足日常开发工作中的基础使用&#xff0c;并没有完整的总结研究。今年的目标之一是完成几个技术栈的系列博客&#xff0c;系统的总结一下知识体系&#xff0c;目前计划是从Redis开始。 Re…...

更新丨三大模块升级,助力高效交付商业项目!

功能更新&#xff01;本文将介绍最新升级的步进漫游、行业方案、VR漫游三个模块&#xff0c;让您更快更好的了解系统能力&#xff0c;为您带来更加便捷、高效的使用体验。步进漫游 离线导出步进式漫游系统&#xff0c;是基于全景图自动生成三维建模的解决方案&#xff0c;实现大…...

C++回顾(二)——const和引用

2.1 C中的const 2.1.1 C与C中const的比较 &#xff08;1&#xff09;C语言中的const C语言中 const修饰的变量是一个 常变量&#xff0c;本质还是变量&#xff0c;有自己的地址空间。 &#xff08;2&#xff09;C中的const 1、C中 const 变量声明的是一个真正的常量&#xff…...

MXNet中使用双向循环神经网络BiRNN对文本进行情感分类<改进版>

在上一节的情感分类当中&#xff0c;有些评论是负面的&#xff0c;但预测的结果是正面的&#xff0c;比如&#xff0c;"this movie was shit"这部电影是狗屎&#xff0c;很明显就是对这部电影极不友好的评价&#xff0c;属于负类评价&#xff0c;给出的却是positive。…...

DNS 域名解析

介绍域名 网域名称&#xff08;英语&#xff1a;Domain Name&#xff0c;简称&#xff1a;Domain&#xff09;&#xff0c;简称域名、网域。 域名是互联网上某一台计算机或计算机组的名称。 域名可以说是一个 IP 地址的代称&#xff0c;目的是为了便于记忆。例如&#xff0c…...

Spring MVC 源码- ViewResolver 组件

ViewResolver 组件ViewResolver 组件&#xff0c;视图解析器&#xff0c;根据视图名和国际化&#xff0c;获得最终的视图 View 对象回顾先来回顾一下在 DispatcherServlet 中处理请求的过程中哪里使用到 ViewResolver 组件&#xff0c;可以回到《一个请求响应的旅行过程》中的 …...

【Hello Linux】初识冯诺伊曼体系

作者&#xff1a;小萌新 专栏&#xff1a;Linux 作者简介&#xff1a;大二学生 希望能和大家一起进步&#xff01; 本篇博客简介&#xff1a;简单介绍冯诺伊曼体系 冯诺伊曼体系 冯诺伊曼体系结构的合理性 我们在Linux的第一篇博客中讲解了第一台计算机的发明是为了解决导弹的…...

mysql索引,主从多个核心主题去探索问题。

网上收集不错的优化方案 事务 mvcc 详讲 详讲 索引 索引概念 MySQL官方对索引的定义为:索引(index)是帮助MySQL高效获取数据的数据结构(有序)。在数据之外,数据 库系统还维护者满足特定查找算法的数据结构,这些数据结构以某种方式引用(指向)数据, 这样就可以在这些数 据…...

前端一面必会面试题(边面边更)

哪些情况会导致内存泄漏 以下四种情况会造成内存的泄漏&#xff1a; 意外的全局变量&#xff1a; 由于使用未声明的变量&#xff0c;而意外的创建了一个全局变量&#xff0c;而使这个变量一直留在内存中无法被回收。被遗忘的计时器或回调函数&#xff1a; 设置了 setInterval…...

【Hello Linux】初识操作系统

作者&#xff1a;小萌新 专栏&#xff1a;Linux 作者简介&#xff1a;大二学生 希望能和大家一起进步&#xff01; 本篇博客简介&#xff1a;简单介绍下操作系统的概念 操作系统 操作系统是什么&#xff1f; 操作系统是管理软硬件资源的软件 为什么要设计操作系统 为什么要设…...

完美的vue3动态渲染菜单路由全程

前言&#xff1a; 首先&#xff0c;我们需要知道&#xff0c;动态路由菜单并非一开始就写好的&#xff0c;而是用户登录之后获取的路由菜单再进行渲染&#xff0c;从而可以起到资源节约何最大程度的保护系统的安全性。 需要配合后端&#xff0c;如果后端的值不匹配&#xff0…...

2023年CDGA考试模拟题库(301-400)

2023年CDGA考试模拟题库(301-400) 300.无附加价值的信息通常也不会被删除,因为:[1分] A.它不应该被移除,所有数据都是有价值的 B.我们可能在以后的某个阶段需更这些信息 C.规程中不明确是否应该保留 D.数据是一种资产它很可能在未来被认为是有价值的 E.规程中不明确哪些是…...

Linux-常见命令

&#x1f69c;关注博主&#xff1a;翻斗花园代码手牛爷爷 &#x1f699;Gitee仓库&#xff1a;牛爷爷爱写代码 目录&#x1f692;xshell热键&#x1f697;Linux基本命令&#x1f697;ls指令&#x1f695;pwd指令&#x1f696;cd指令&#x1f68c;touch指令&#x1f68d;mkdir指…...

2.25测试对象分类

一.按照测试对象划分1.界面测试又称UI测试,按照界面的需求(一般是ui设计稿)和界面的设计规则,对我们软件界面所展示的全部内容进行测试和检查.对于非软件来说:颜色,大小,材质,整体是否美观对于软件来说:输入框,按钮,文字,图片...的尺寸,颜色,形状,整体适配,清晰度等等,2.可靠性…...

【Zabbix实战之部署篇】Zabbix客户端的安装部署方法

【Zabbix实战之部署篇】Zabbix客户端的安装部署方法 一、Zabbix-agent2介绍1.Zabbix-agent2简介2.Zabbix-agent2优点3.主动模式和被动模式二、环境规划1.Zabbix服务器部署链接2.IP规划三、配置客户端系统环境1.关闭selinux2.放行端口或关闭防火墙四、安装zabbix-agent21.下载za…...

【CSS】CSS 层叠样式表 ② ( CSS 引入方式 - 内嵌样式 )

文章目录一、CSS 引入方式 - 内嵌样式1、内嵌样式语法2、内嵌样式示例3、内嵌样式完整代码示例4、内嵌样式运行效果一、CSS 引入方式 - 内嵌样式 1、内嵌样式语法 CSS 内嵌样式 , 一般将 CSS 样式写在 HTML 的 head 标签中 ; CSS 内嵌样式 语法如下 : <head><style …...