Spring源码阅读:Spring事务执行流程

news2025/1/12 11:59:12

一、概述

Spring事务执行的流程如下图,我们将按照下面的执行顺序,介绍Spring的事务。

 

二、代码分析

(一)核心流程

TransactionAspectSupport#invokeWithinTransaction 事务处理核心伪代码

// 通过事务调用
protected Object invokeWithinTransaction(Method method, @Nullable Class<?> targetClass,
      final InvocationCallback invocation) throws Throwable {

   // 获取事务属性源对象
   TransactionAttributeSource tas = getTransactionAttributeSource();
   // 通过事务属性源对象获取到当前方法的事务属性信息
   final TransactionAttribute txAttr = (tas != null ? tas.getTransactionAttribute(method, targetClass) : null);
   // 获取配置的事务管理器对象DataSourceTransactionManager
   final TransactionManager tm = determineTransactionManager(txAttr);

   // TransactionManager 转换为  PlatformTransactionManager
   PlatformTransactionManager ptm = asPlatformTransactionManager(tm);
   // 获取连接点的唯一标识  类名+方法名
   final String joinpointIdentification = methodIdentification(method, targetClass, txAttr);

   // 创建事务信息TransactionInfo
   TransactionInfo txInfo = createTransactionIfNecessary(ptm, txAttr, joinpointIdentification);

   Object retVal;
   try {
        // 执行被增强方法,调用具体的处理逻辑
        retVal = invocation.proceedWithInvocation();
   }
   catch (Throwable ex) {
        // 异常回滚
        completeTransactionAfterThrowing(txInfo, ex);
        throw ex;
    }
   finally {
        //清除事务信息,恢复线程私有的老的事务信息
        cleanupTransactionInfo(txInfo);
   }

    // 成功后提交,会进行资源储量,连接释放,恢复挂起事务等操作
    commitTransactionAfterReturning(txInfo);
    return retVal;

}

  1、 getTransactionAttributeSource() 获取事务属性源对象,获取的是在IOC容器启动过程中,创建的NameMatchTransactionAttributeSource对象,内含属性nameMap,可通过方法名称获取方法对应的事务属性。

// key: 方法名称;value:方法的事务属性
2 private Map<String, TransactionAttribute> nameMap = new HashMap<>();

 2、determineTransactionManager(txAttr) 获取事务管理器,返回在IOC容器启动过程中,创建的DataSourceTransactionManager事务管理器对象。

  3、asPlatformTransactionManager(txAttr) 将DataSourceTransactionManager转换为PlatformTransactionManager。

  4、methodIdentification(method, targetClass, txAttr) 获取被增强方法的类名 + 方法名

  5、获取连接、开启事务 (包含事务传播特性的处理,执行insert、update、delete等被增强方法前的处理)

  6、执行被增强方法,数据库新增、删除、更新操作

  7、是否异常

   被增强方法执行异常,completeTransactionAfterThrowing(txInfo, ex) 事务回滚操作

   被增强方法执行无异常, commitTransactionAfterReturning(txInfo) 事务提交操作

(二)创建事务信息

protected TransactionInfo createTransactionIfNecessary(@Nullable PlatformTransactionManager tm,
      @Nullable TransactionAttribute txAttr, final String joinpointIdentification) {

   // If no name specified, apply method identification as transaction name.
   // 如果没有名称指定则使用方法唯一标识,并使用DelegatingTransactionAttribute封装txAttr
   if (txAttr != null && txAttr.getName() == null) {
      txAttr = new DelegatingTransactionAttribute(txAttr) {
         @Override
         public String getName() {
            return joinpointIdentification;
         }
      };
   }

   // 获取TransactionStatus事务状态信息,开启事务、获取jdbc连接
   TransactionStatus status = tm.getTransaction(txAttr);

   // 根据指定的属性与status准备TransactionInfo,
   return prepareTransactionInfo(tm, txAttr, joinpointIdentification, status);
}

1、封装事务属性对象

  用DelegatingTransactionAttribute对象封装事务属性对象TransactionAttribute,若事务属性对象的name属性为空,重写DelegatingTransactionAttribute的getName方法,将被增强方法的类名+方法名返回;

2、事务状态 - TransactionStatus

AbstractPlatformTransactionManager#getTransaction 获取事务状态信息核心伪代码

public final TransactionStatus getTransaction(@Nullable TransactionDefinition definition)
      throws TransactionException {

   // 若不指定事务定义,则使用默认的事务定义StaticTransactionDefinition
   TransactionDefinition def = (definition != null ? definition : TransactionDefinition.withDefaults());

   // 获取数据源对象 DataSourceTransactionObject
   Object transaction = doGetTransaction();

   // 若已存在事务,按已存在事务流程执行
   if (isExistingTransaction(transaction)) {
      return handleExistingTransaction(def, transaction, debugEnabled);
   }

   // Mandatory支持当前事务,如果不存在就抛异常
   if (def.getPropagationBehavior() == TransactionDefinition.PROPAGATION_MANDATORY) {
      throw new IllegalTransactionStateException(
            "No existing transaction found for transaction marked with propagation 'mandatory'");
   }
   // 事务不存在,创建新事务
   else if (def.getPropagationBehavior() == TransactionDefinition.PROPAGATION_REQUIRED ||
         def.getPropagationBehavior() == TransactionDefinition.PROPAGATION_REQUIRES_NEW ||
         def.getPropagationBehavior() == TransactionDefinition.PROPAGATION_NESTED) {
      SuspendedResourcesHolder suspendedResources = suspend(null);

      try {
         // 开启事务(获取连接、开启事务)
         return startTransaction(def, transaction, debugEnabled, suspendedResources);
      }
      catch (RuntimeException | Error ex) {
         resume(null, suspendedResources);
         throw ex;
      }
   }
   else {
      // 创建一个空事务信息,用作事务同步
      boolean newSynchronization = (getTransactionSynchronization() == SYNCHRONIZATION_ALWAYS);
      return prepareTransactionStatus(def, null, true, newSynchronization, debugEnabled, null);
   }
}

1、获取数据源事务对象DataSourceTransactionObject,含是否允许当前事务设置保存点、jdbc连接持有对象,此处有一个复用JDBC连接逻辑。

  TransactionSynchronizationManager.getResource(obtainDataSource()) 获取数据源对应的jdbc连接 核心伪代码

private static Object doGetResource(Object actualKey) {
    // 获取线程私有事务资源
    Map<Object, Object> map = resources.get();
    // 第一次创建事务,线程私有事务资源为空,后续创建,返回null
    if (map == null) {
        return null;
    }
    // 获取线程私有事务资源中数据源对应的jdbc连接
    Object value = map.get(actualKey);
    // 如果value是ResourceHolder类型,并且已经解绑
    if (value instanceof ResourceHolder && ((ResourceHolder) value).isVoid()) {
        // 将该数据源从线程私有事务资源中的元素map里移除
        map.remove(actualKey);
        // 线程私有事务资源的map为空,清空线程私有事务资源

        if (map.isEmpty()) {
            resources.remove();
        }
        // 返回null
        value = null;
    }
    // 返回数据源对应的jdbc连接
    return value;
}

2、isExistingTransaction(transaction) 若已存在事务,根据被增强方法的传播特性,判断是抛异常、新建事务、还是支持原有事务。

3、根据被增强方法的传播特性判断是抛异常、创建事务、按非事务方式执行。

  当然我们最关心的还是按照事务的方式来执行被增强方法,下面来看看被增强方法的事务是如何处理的。

(三)开启事务

  AbstractPlatformTransactionManager#startTransaction 开启事务核心伪代码

// 开启事务
private TransactionStatus startTransaction(TransactionDefinition definition, Object transaction,
      boolean debugEnabled, @Nullable SuspendedResourcesHolder suspendedResources) {

   // 是否需要新同步
   boolean newSynchronization = (getTransactionSynchronization() != SYNCHRONIZATION_NEVER);
   // 创建新事务状态对象
   DefaultTransactionStatus status = newTransactionStatus(
         definition, transaction, true, newSynchronization, debugEnabled, suspendedResources);
   // 获取连接、开启事务
   doBegin(transaction, definition);
   // 新同步事务的设置,针对于当前线程的设置
   prepareSynchronization(status, definition);
   return status;
}

1、获取事务是否需要同步标识

  用作后续prepareSynchronization对线程私有变量做初始化操作的判断;

2、创建事务状态对象

  AbstractPlatformTransactionManager#newTransactionStatus 创建事务状态对象

// 创建事务状态对象
protected DefaultTransactionStatus newTransactionStatus(TransactionDefinition definition, @Nullable Object transaction, boolean newTransaction,
       boolean newSynchronization, boolean debug, @Nullable Object suspendedResources) {

   // 重置同步标识,只有当前事务需要同步,并且未被激活,同步标识为true
  boolean actualNewSynchronization = newSynchronization && !TransactionSynchronizationManager.isSynchronizationActive();
   // 创建DefaultTransactionStatus事务状态对象
  return new DefaultTransactionStatus(transaction, newTransaction, actualNewSynchronization, definition.isReadOnly(), debug, suspendedResources);
}

3、获取连接、开启事务

  DataSourceTransactionManager#doBegin 获取连接、开启事务核心伪代码:

// 获取连接、开启事务
protected void doBegin(Object transaction, TransactionDefinition definition) {
   // 强制转化事务对象
   DataSourceTransactionObject txObject = (DataSourceTransactionObject) transaction;
   Connection con = null;

   try {
      // 判断事务对象没有数据库连接持有器
      if (!txObject.hasConnectionHolder() ||
            txObject.getConnectionHolder().isSynchronizedWithTransaction()) {
         // 通过数据源获取一个数据库连接对象
         Connection newCon = obtainDataSource().getConnection();

         // 把数据库连接包装成一个ConnectionHolder对象 然后设置到txObject对象中去
         txObject.setConnectionHolder(new ConnectionHolder(newCon), true);
      }

      // 标记当前的连接是一个同步事务
      txObject.getConnectionHolder().setSynchronizedWithTransaction(true);
      // 获取JDBC连接
      con = txObject.getConnectionHolder().getConnection();

      // 为当前的事务设置隔离级别
      Integer previousIsolationLevel = DataSourceUtils.prepareConnectionForTransaction(con, definition);
      // 设置先前隔离级别
      txObject.setPreviousIsolationLevel(previousIsolationLevel);
      // 设置是否只读
      txObject.setReadOnly(definition.isReadOnly());

      // 关闭自动提交
      if (con.getAutoCommit()) {
         //设置需要恢复自动提交
         txObject.setMustRestoreAutoCommit(true);
         // 关闭自动提交
         con.setAutoCommit(false);
      }

      // 判断是否设置为只读事务
      prepareTransactionalConnection(con, definition);
      // 标记激活事务
      txObject.getConnectionHolder().setTransactionActive(true);

      // 设置事务超时时间
      int timeout = determineTimeout(definition);
      if (timeout != TransactionDefinition.TIMEOUT_DEFAULT) {
         txObject.getConnectionHolder().setTimeoutInSeconds(timeout);
      }

      // 绑定数据源和连接到同步管理器上,把数据源作为key,数据库连接作为value 设置到线程变量中
      if (txObject.isNewConnectionHolder()) {
         // 将当前获取到的连接绑定到当前线程
         TransactionSynchronizationManager.bindResource(obtainDataSource(), txObject.getConnectionHolder());
      }
   }

   catch (Throwable ex) {
      if (txObject.isNewConnectionHolder()) {
         // 释放数据库连接
         DataSourceUtils.releaseConnection(con, obtainDataSource());
         txObject.setConnectionHolder(null, false);
      }
      throw new CannotCreateTransactionException("Could not open JDBC Connection for transaction", ex);
   }
}

3.1、获取jdbc连接

  第一次创建事务,数据源属性对象中的数据库jdbc连接为空,通过数据源获取jdbc数据源连接。

3.2、填充数据源属性对象属性

  1、设置连接持有器中的事务为同步事务,设置数据源属性对象之前的隔离级别、设置只读属性;

  2、关闭jdbc连接的自动提交,开启事务;

  3、标记当前事务为激活状态,可用作后续prepareSynchronization对线程私有变量做初始化操作的判断;

3.3、绑定新的jdbc连接至当前线程的事务资源中

  若当前连接是一个新的jdbc连接,需要将当前jdbc连接绑定到线程当前线程私有的事务资源属性中TransactionSynchronizationManager#bindResource 核心伪代码如下:

// 绑定jdbc连接至当前线程的事务资源中
public static void bindResource(Object key, Object value) throws IllegalStateException {
   // 获取数据源
   Object actualKey = TransactionSynchronizationUtils.unwrapResourceIfNecessary(key);
   Assert.notNull(value, "Value must not be null");
   // 获取线程私有事务资源中jdbc连接
   Map<Object, Object> map = resources.get();
   // 第一次创建事务,事务资源未创建,初始化线程私有事务资源
   if (map == null) {
      map = new HashMap<>();
      resources.set(map);
   }
   // 将数据源作为key,jdbc连接作为value,绑定到线程私有的事务资源中
   Object oldValue = map.put(actualKey, value);

   // 若当前线程的事务资源中原有的value为ResourceHolder类型,并且已经解绑,将原有的事务资源value置空
   if (oldValue instanceof ResourceHolder && ((ResourceHolder) oldValue).isVoid()) {
       oldValue = null;
   }

   // 若当前线程的事务资源已经被绑定,抛异常
   if (oldValue != null) {
      throw new IllegalStateException("Already value [" + oldValue + "] for key [" +
            actualKey + "] bound to thread [" + Thread.currentThread().getName() + "]");
   }

}

  当外层事务的传播特性为REQUIRED,内层的传播特性也为REQUIRED时,内层可复用外层创建的jdbc连接。通过数据源获取的jdbc连接是同一个。

(四)事务信息

  事务信息准备核心伪代码:

// 准备事务信息
protected TransactionInfo prepareTransactionInfo(@Nullable PlatformTransactionManager tm,
  @Nullable TransactionAttribute txAttr, String joinpointIdentification,
    @Nullable TransactionStatus status) {

  // 创建事务信息
  TransactionInfo txInfo = new TransactionInfo(tm, txAttr, joinpointIdentification);



  // 设置新事务状态
  txInfo.newTransactionStatus(status);

   // 事务信息绑定到当前线程
  txInfo.bindToThread();

   // 返回事务信息
  return txInfo;
}

1、创建事务信息

  创建TransactionInfo对象,封装事务管理器,事务注解属性,被增强方法类名+方法名,事务状态对象

2、事务信息绑定到当前线程

// 持有当前被增强方法的事务状态,支持多个被增强方法间的事务处理
private static final ThreadLocal<TransactionInfo> transactionInfoHolder =
      new NamedThreadLocal<>("Current aspect-driven transaction");

// 将事务信息绑定到当前线程
private void bindToThread() {
    // 获取事务信息持有器原来的事务信息,并设置到oldTransactionInfo属性
    this.oldTransactionInfo = transactionInfoHolder.get();
    // 将当前线程的事务属性设置进事务持有器中
    transactionInfoHolder.set(this);
}

(五)事务信息

  TransactionAspectSupport#commitTransactionAfterReturning 调用事务管理器提交事务概览伪代码

// 提交事务
protected void commitTransactionAfterReturning(@Nullable TransactionInfo txInfo) {
   // 事务信息不为空,并且事务信息对象持有事务状态不为空
   if (txInfo != null && txInfo.getTransactionStatus() != null) {
      // 调用事务信息对象中事务管理器的事务提交方法
      txInfo.getTransactionManager().commit(txInfo.getTransactionStatus());
   }
}

1、提交处理

  AbstractPlatformTransactionManager#commit 事务提交伪代码

public final void commit(TransactionStatus status) throws TransactionException {
  // 获取事务状态对象
  DefaultTransactionStatus defStatus = (DefaultTransactionStatus) status;
  // 处理事务提交
  processCommit(defStatus);
}

  AbstractPlatformTransactionManager#processCommit 处理事务提交伪代码

private void processCommit(DefaultTransactionStatus status) throws TransactionException {
   try {
     //当前状态是新事务
     if (status.isNewTransaction()) {
        // 若当前事务为新事务,事务提交
        doCommit(status);
     }
   } finally {
      //根据条件,完成后数据清除,和线程的私有资源解绑,重置连接自动提交,隔离级别,是否只读,释放连接,恢复挂起事务等
      cleanupAfterCompletion(status);
   }
}

  DataSourceTransactionManager#doCommit 事务提交核心伪代码

protected void doCommit(DefaultTransactionStatus status) {
    // 获取事务状态中的数据源事务对象
    DataSourceTransactionObject txObject = (DataSourceTransactionObject) status.getTransaction();
    // 获取数据源事务对象中连接持有器的jdbc连接
    Connection con = txObject.getConnectionHolder().getConnection();
    try {
        // jdbc连接事务提交
        con.commit();
    }
catch (SQLException ex) {
        throw new TransactionSystemException("Could not commit JDBC transaction", ex);
    }
}

  AbstractPlatformTransactionManager#cleanupAfterCompletion 提交完成时的清理核心伪代码

private void cleanupAfterCompletion(DefaultTransactionStatus status) {
   // 将当前事务设置为完成状态
   status.setCompleted();
   // 新事务完成事务提交,清理线程私有事务属性资源
   if (status.isNewSynchronization()) {
      TransactionSynchronizationManager.clear();
   }
   // 恢复jdbc连接相关操作
   if (status.isNewTransaction()) {
      doCleanupAfterCompletion(status.getTransaction());
   }
   // 恢复挂起事务
   if (status.getSuspendedResources() != null) {
      Object transaction = (status.hasTransaction() ? status.getTransaction() : null);
      resume(transaction, (SuspendedResourcesHolder) status.getSuspendedResources());
   }
}

  TransactionSynchronizationManager#clear() 清除当前线程私有的事务同步信息核心代码

// 事务同步
private static final ThreadLocal<Set<TransactionSynchronization>> synchronizations = new NamedThreadLocal<>("Transaction synchronizations");

// 当前事务的名称
private static final ThreadLocal<String> currentTransactionName = new NamedThreadLocal<>("Current transaction name");

// 当前事务是否只读
private static final ThreadLocal<Boolean> currentTransactionReadOnly = new NamedThreadLocal<>("Current transaction read-only status");

// 当前事务的隔离级别
private static final ThreadLocal<Integer> currentTransactionIsolationLevel = new NamedThreadLocal<>("Current transaction isolation level");

// 实际事务是否激活
private static final ThreadLocal<Boolean> actualTransactionActive = new NamedThreadLocal<>("Actual transaction active");

   public static void clear() {
       synchronizations.remove();
       currentTransactionName.remove();
       currentTransactionReadOnly.remove();
       currentTransactionIsolationLevel.remove();
       actualTransactionActive.remove();
   }

  DataSourceTransactionManager#doCleanupAfterCompletion jdbc连接相关操作核心伪代码。  jdbc连接解绑、属性恢复、连接释放、连接关闭及数据源事务对象持有jdbc连接释放操作。

// 事务提交后jdbc连接相关处理
protected void doCleanupAfterCompletion(Object transaction) {
   // 获取数据源事务对象
   DataSourceTransactionObject txObject = (DataSourceTransactionObject) transaction;

   // 新事务,将数据库连接从当前线程中解除绑定
   if (txObject.isNewConnectionHolder()) {
      TransactionSynchronizationManager.unbindResource(obtainDataSource());
   }

   // 获取连接
   Connection con = txObject.getConnectionHolder().getConnection();
   // 关闭事务,恢复数据库连接的自动提交属性
   if (txObject.isMustRestoreAutoCommit()) {
     con.setAutoCommit(true);
   }
   // 重置数据库jdbc连接, 恢复原有隔离级别、是否只读属性
   DataSourceUtils.resetConnectionAfterTransaction(con, txObject.getPreviousIsolationLevel(), txObject.isReadOnly());

   // 当前事务是独立的新创建的事务,在事务完成时释放数据库连接并关闭连接
   if (txObject.isNewConnectionHolder()) {
      DataSourceUtils.releaseConnection(con, this.dataSource);
   }

   // 清除数据源事务对象中的连接持有器
   txObject.getConnectionHolder().clear();
}

  AbstractPlatformTransactionManager#resume 恢复外层原挂起事务核心代码

// 恢复外层挂起事务
protected final void resume(@Nullable Object transaction, @Nullable SuspendedResourcesHolder resourcesHolder)
      throws TransactionException {
   // 若存在挂起事务,恢复挂起事务
   if (resourcesHolder != null) {
      // 获取挂起事务的资源属性
      Object suspendedResources = resourcesHolder.suspendedResources;
      // 挂起事务的资源属性不为空,将事务属性资源绑定到线程私有的事务属性资源中
      if (suspendedResources != null) {
         doResume(transaction, suspendedResources);
      }
   }
}

2、回滚处理

  TransactionAspectSupport#completeTransactionAfterThrowing 事务回滚概览伪代码:

// 异常时事务处理
protected void completeTransactionAfterThrowing(@Nullable TransactionInfo txInfo, Throwable ex) {
   if (txInfo != null && txInfo.getTransactionStatus() != null) {
      // 事务信息中事务属性不为空,并且事务属性中回滚
      if (txInfo.transactionAttribute != null && txInfo.transactionAttribute.rollbackOn(ex)) {
        // 进行回滚
        txInfo.getTransactionManager().rollback(txInfo.getTransactionStatus());
      }
      else {
         // 回滚标识rollBackOnly为 false,执行提交
         txInfo.getTransactionManager().commit(txInfo.getTransactionStatus());
      }
   }
}

  DefaultTransactionAttribute#rollbackOn 是否回滚判断标识

// 未捕获异常的回滚操作
public boolean rollbackOn(Throwable ex) {
   return (ex instanceof RuntimeException || ex instanceof Error);
}

  AbstractPlatformTransactionManager#processRollback 回滚核心伪代码

// 处理回滚
private void processRollback(DefaultTransactionStatus status, boolean unexpected) {
   try {
      // 当前事务是一个新事务
      if (status.isNewTransaction()) {
        // 进行回滚
        doRollback(status);
      }
   }
   finally {
      // 回滚完成时的处理
      cleanupAfterCompletion(status);
   }
}

  DataSourceTransactionManager#doRollback 事务回滚核心伪代码

@Override
protected void doRollback(DefaultTransactionStatus status) {
   // 获取事务状态中的数据源事务对象
   DataSourceTransactionObject txObject = (DataSourceTransactionObject) status.getTransaction();
   // 获取数据源事务对象中连接持有器的jdbc连接
   Connection con = txObject.getConnectionHolder().getConnection();
   try {
      // jdbc的回滚
      con.rollback();
   }catch (SQLException ex) {
      throw new TransactionSystemException("Could not roll back JDBC transaction", ex);
   }
}

参考文章:

Spring 事务源码分析_spring事务源码分析_J3code的博客-CSDN博客

Spring事务源码分析_spring事务源码深度解析_程序员小潘的博客-CSDN博客

https://www.cnblogs.com/wt20/p/10957371.html

https://www.cnblogs.com/RunningSnails/p/17015808.html

Spring事务流程源码剖析+传播行为运用案例+心得 - 知乎

十八、spring 事务之事务执行流程 - 简书

https://www.cnblogs.com/RunningSnails/p/17015808.html

https://www.cnblogs.com/RunningSnails/p/17017176.html

https://www.cnblogs.com/RunningSnails/p/17020456.html

https://www.cnblogs.com/RunningSnails/p/17023323.html

https://www.cnblogs.com/RunningSnails/p/17025931.html

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

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

相关文章

第12届蓝桥杯Scratch选拔赛真题集锦

第12届蓝桥杯Scratch选拔赛真题集锦 编程题 第 1 题 问答题 吃豆人游戏 题目说明 编程实现: 吃豆人游戏: 通过方向键控制吃豆人吃到水果&#xff0c;当水果全部被吃掉后游戏结束。具体要求: 1).点击绿旗&#xff0c;舞台上出现要求的角色与背景;每个格子内有一个水果和一个幽…

chatgpt赋能Python-python_ip摄像头

Python IP摄像头介绍 IP摄像头技术的快速发展&#xff0c;使得越来越多的人开始使用IP摄像头来保护家庭、企业和公共场所的安全。Python语言的简洁、优雅和易于学习&#xff0c;使得它成为越来越多人选择的编程语言。在本文中&#xff0c;我们将介绍如何使用Python编程来控制I…

java循环依赖

java循环依赖 什么叫循环依赖&#xff1f; Java循环依赖是指两个或多个类之间存在相互依赖的情况&#xff0c;造成它们互相持有对方的实例而无法正确地加载和初始化。这种问题通常发生在使用Spring等依赖注入框架时&#xff0c;因为这些框架会自动管理类的实化和注入。 我们看…

深度学习基础入门篇[10]:序列模型-词表示{One-Hot编码、Word Embedding、Word2Vec、词向量的一些有趣应用}

【深度学习入门到进阶】必看系列&#xff0c;含激活函数、优化策略、损失函数、模型调优、归一化算法、卷积模型、序列模型、预训练模型、对抗神经网络等 专栏详细介绍&#xff1a;【深度学习入门到进阶】必看系列&#xff0c;含激活函数、优化策略、损失函数、模型调优、归一化…

机器学习神经网络——Adaboost分离器算法

系列文章目录 机器学习之SVM分类器介绍——核函数、SVM分类器的使用 机器学习的一些常见算法介绍【线性回归&#xff0c;岭回归&#xff0c;套索回归&#xff0c;弹性网络】 机器学习相关概念思维导图 文章目录 系列文章目录 前言 Adaboost算法的简单介绍 Adaboost算法相…

Unity之使用Photon PUN开发多人游戏教程

前言 Photon是一个网络引擎和多人游戏平台,可以处理其服务器上的所有请求,我们可以在 Unity(或其他游戏引擎)中使用它,并快速把游戏接入Photon的网络中,而我们就可以专注于在项目中添加逻辑,专注于游戏玩法和功能了。 PUN(Photon Unity Networking)是一种开箱即用的解…

Linux服务器安装启动卸载宝塔面板

宝塔面板简介 宝塔面板是提升运维效率的服务器管理软件&#xff0c;支持一键LAMP/LNMP/集群/监控/网站/FTP/数据库/JAVA等100多项服务器管理功能&#xff0c;支持Windows和Linux系统&#xff0c;服务器可以通过网络轻松管理&#xff0c;提高运维效率。例如&#xff1a;创建和管…

极米H6搭载光学变焦打造无损4K,带来更沉浸观影体验

近年来&#xff0c;科技飞速发展&#xff0c;我国涌现出很多新兴科技企业&#xff0c;如家用智能投影、无人机、扫地机器人等行业发展迅速&#xff0c;国际竞争力逐年增强。其中&#xff0c;家用智能投影行业成长快速&#xff0c;正展现出蓬勃的发展生机。根据IDC自2015到2022年…

rollup打包vue组件并发布到npm

个人认为rollup在打包组件的实现方式比用webpack方便&#xff0c;webpack应该是比较适合打包大型的工程项目&#xff0c;而rollup适合打包一些平时在项目中的组件 rollup中文文档&#xff1a;简介 | rollup.js 中文文档 | rollup.js中文网 (rollupjs.com) 这次打包一个滚动加…

python操作mysql

Python 中使用什么第三方库来操作 MySQL &#xff1f;如何安 装&#xff1f; 使用第三方库为&#xff1a; pymysql 安装&#xff1a; pip install pymysql如何获取链接对象&#xff1f; • from pymysql import Connection 导包 • Connection( 主机 , 端口 , 账户 , 密码 ) 即…

Java基础面试题突击系列4

&#x1f469;&#x1f3fb; 作者&#xff1a;一只IT攻城狮 &#xff0c;关注我不迷路 ❤️《java面试核心知识》突击系列&#xff0c;持续更新… &#x1f490; 面试必知必会学习路线&#xff1a;Java技术栈面试系列SpringCloud项目实战学习路线 &#x1f4dd;再小的收获x365天…

Docker Compose 和 ChatGpt 私有部署

1. 下载 Docker Compose 的二进制文件&#xff1a; Docker Compose 的Github下载链接地址是&#xff1a; Releases docker/compose GitHub 注&#xff1a;这里没有讲解docker的安装&#xff0c;因为已经写好了&#xff0c;在这里可以去看一下&#xff0c;先安装好docker在其弄…

京东购物车如何提升30%性能 | 京东云技术团队

1、背景 购物车面临的挑战&#xff1a; 1&#xff09;新业务&#xff1a;随着业务形态的丰富&#xff0c;购物车在不断支持各种新业务&#xff0c;依赖的外部接口也随之增加&#xff1b; 2&#xff09;下沉&#xff1a;一些前端调用的接口下沉到购物车中台&#xff1b; 3&am…

Linux systemd

这里写目录标题 一:配置文件内容二:配置文件位置三:开机启动配置四:systemctl使用 一:配置文件内容 systemd服务配置文件存放在/usr/lib/systemd目录下&#xff0c;有系统system和用户user之分&#xff0c;需要开机不登录就能运行的程序&#xff0c;放在/usr/lib/systemd/syst…

项目跟踪的好处有哪些?项目经理必须了解

项目跟踪要跟踪什么呢&#xff1f;主要针对计划、任务和项目成员三个方面&#xff0c;是为了了解项目的实际进展情况而进行。 在跟踪过程中我们会发现项目计划的不当之处&#xff0c;促使我们去改进和完善计划&#xff1b;发现项目风险&#xff0c;及时解决问题&#xff1b;了…

软件设计之UML用例图大白话教程

1、为什么要使用UML用例图&#xff1f; 对一个复杂问题或者现象的分析&#xff0c;好的方式方法往往能带来事半功倍的效果。比如在软件开发领域&#xff0c;参与的人员角色各种各样&#xff0c;比如软件开发工程师、产品经理、客户、运营人员、老板、用户、B端客户等等&#x…

chatgpt赋能Python-python_kargs

Python中的*kargs&#xff1a;详解 在Python编程中&#xff0c;经常需要为函数传递参数。除了常规的参数传递&#xff0c;Python还支持通过关键字参数传递参数。这种方法比较灵活&#xff0c;可以快速地修改函数的输入参数。其中&#xff0c;kargs是Python中非常常见的一个参数…

管理git项目:003

首先我们得创建一个git管理仓库 进入桌面路径&#xff0c;待会创建的文件可以在桌面找到 cd ..【创建git管理仓库】&#xff1a;创建一个.git 管理仓库&#xff0c; 这个文件被创建后是一个隐藏文件。这个.git文件的作用是&#xff1a;指定当前文件夹作为git管理仓库 git init …

今天,我问了ChatGPT一个难以启齿的问题,内容实录

文 / 高扬&#xff08;微信公众号&#xff1a;量子论&#xff09; 七个月前&#xff0c;我是一个害怕失业的漂泊上海的小青年&#xff0c;每天在合租房里刷着LeetCode。 现在我已经在一家人工智能公司从事着自己喜欢的关于机器深度学习的研发工作。 你想知道我的生活发生了什么…

学习git

文章目录 02-为什么要学习Git软件&#xff1f;03 概念&#xff1a;版本控制04 概念&#xff1a;版本控制软件基础功能05 概念&#xff1a;集中式、分布式版本控制系统、多人协作开发5.1 文件冲突问题5.2集中式版本控制&#xff08;CVS&#xff0c;SVN&#xff09;5.3 分布式版本…