AbstractRoutingDataSource实现多数据源切换以及事务中无法切换问题

一、AbstractRoutingDataSource实现多数据源切换

        为了实现数据源的动态切换,我们采用了AbstractRoutingDataSource结合AOP+反射来自定义注解。通过这种机制,我们可以在运行时根据自定义注解来选择不同的数据源,从而实现灵活高效的数据访问策略。

        具体来说,我们首先创建了一个继承自AbstractRoutingDataSource的动态数据源类DynamicDataSource,该类能够管理多个数据源并根据线程上下文中的特定键值来选择使用哪一个数据源。接着,我们定义了一个自定义注解,用于标记需要切换数据源的方法。然后,利用AOP技术,我们在方法执行之前和之后添加了环绕通知,在这个通知中,我们通过反射获取到当前方法的自定义注解信息,并据此设置线程上下文中的数据源键值,从而引导数据源切换逻辑。

       这样,当一个被标注了自定义注解的方法被调用时,系统会自动根据注解指定的数据源名称切换到对应的数据源,完成对数据库的操作。这种方式不仅提高了代码的复用性和可维护性,而且通过解耦业务逻辑与数据源选择,增强了系统的可扩展性和灵活性。

1、自定义注解DataSource
import java.lang.annotation.*;

@Target({ElementType.METHOD})//注解方法
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface DataSource {
    String name() default "";
}
2、切面类DataSourceAspect
import com.zcloud.sunshine.common.constant.DatabaseType;
import com.zcloud.sunshine.config.DynamicDataSource;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.stereotype.Component;

import java.lang.reflect.Method;

@Aspect
@Component
@Slf4j
public class DataSourceAspect {

    @Pointcut("@annotation(com.zcloud.sunshine.annotation.DataSource)")
    public void dataSourcePointCut() {

    }

    @Around("dataSourcePointCut()")
    public Object around(ProceedingJoinPoint point) throws Throwable {
        
       //获取被拦截方法的签名,该签名包含了方法的名称、参数类型等信息
        MethodSignature signature = (MethodSignature) point.getSignature();
        //从签名中获取了实际的方法对象
        Method method = signature.getMethod();
        //从方法上获取名为DataSource的注解。如果这个方法上有DataSource注解,那么dataSource将不为null,否则为null
        DataSource dataSource = method.getAnnotation(DataSource.class);
        if (dataSource == null) {
            log.info("默认数据源 dataSource1");
            DynamicDataSource.setDataSource(DatabaseType.dataSource1.toString());
        } else {
            log.info("切换数据源: " + dataSource.name());
            DynamicDataSource.setDataSource(dataSource.name());
        }

        try {
            return point.proceed();
        } finally {
        //最后一定要清除,这里使用的ThreadLocal来存储的数据源key,所以为了防止内存泄露一定要清除
        //而且该清除操作也是为了防止该切换操作对后续的切换操作造成影响
            DynamicDataSource.clearDataSource();
        }
    }
}

上面两步主要就是通过aop+反射完成自定义切换数据源注解的功能,实现在该注解修饰的方法上,设置当前方法的数据源。如果未设置数据源,则使用默认的数据源。

3、自定义动态数据源DynamicDataSource类
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;

import javax.sql.DataSource;
import java.util.Map;

public class DynamicDataSource extends AbstractRoutingDataSource {
//使用ThreadLocal来存储当前线程的数据源名称,保证多线程情况下,各自的数据源互不影响
    private static final ThreadLocal<String> contextHolder = new ThreadLocal<>();

    public DynamicDataSource(DataSource defaultTargetDataSource, Map<Object, Object> targetDataSources) {
        //将注册的数据源以及设置的默认数据源设置到父类对应的成员变量中
        super.setDefaultTargetDataSource(defaultTargetDataSource);
        super.setTargetDataSources(targetDataSources);
        super.afterPropertiesSet();
    }

    //返回当前数据源
    @Override
    protected Object determineCurrentLookupKey() {
        return getDataSource();
    }

    public static void setDataSource(String dataSource) {
        contextHolder.set(dataSource);
    }

    public static String getDataSource() {
        return contextHolder.get();
    }

    public static void clearDataSource() {
        contextHolder.remove();
    }
}
4、配置多数据源DynamicDataSourceConfig 
import com.alibaba.druid.spring.boot.autoconfigure.DruidDataSourceBuilder;
import com.zcloud.sunshine.common.constant.DatabaseType;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;
import org.springframework.stereotype.Component;

import javax.sql.DataSource;
import java.util.HashMap;
import java.util.Map;

/**
 * @author zy
 * @desc 配置多数据源
 */
@Configuration
@Component
public class DynamicDataSourceConfig {

    @Bean(value = "dataSource1")
    @ConfigurationProperties(prefix = "spring.datasource.druid.dataSource1")
    public DataSource dataSource1() {  //此处的返回类型DataSource不是我们自定义的注解DataSource,而是java.sql包下的
        return DruidDataSourceBuilder.create().build();
    }

    @Bean(value = "dataSource2")
    @ConfigurationProperties(prefix = "spring.datasource.druid.dataSource2")
    public DataSource dataSource2() {
        return DruidDataSourceBuilder.create().build();
    }

    @Bean
    @Primary
    public DynamicDataSource dataSource(@Qualifier(value = "dataSource1") DataSource dataSource1,
                                        @Qualifier(value = "dataSource2") DataSource dataSource2
                                        ) {
        Map<Object, Object> targetDataSources = new HashMap<>();
        targetDataSources.put(DatabaseType.dataSource1.toString(), dataSource1);
        targetDataSources.put(DatabaseType.dataSource2.toString(), dataSource2);
        //注册所有的数据源,并且将数据源1设置为默认数据源
        return new DynamicDataSource(dataSource1, targetDataSources);
    }
}

5、多数据源枚举类
public enum DatabaseType {

    dataSource1,//数据库1
    dataSource2//数据库2
}
6、配置多数据源数据库连接信息
#数据源1
spring.datasource.druid.dataSource1.url=jdbc:mysql://127.0.0.1:3306/zcloud_user?useUnicode=true&characterEncoding=utf-8&allowMultiQueries=true&useTimezone=true&serverTimezone=GMT%2B8
spring.datasource.druid.dataSource1.username=root
spring.datasource.druid.dataSource1.password=zy521
spring.datasource.druid.dataSource1.driver-class-name=com.mysql.cj.jdbc.Driver
spring.datasource.druid.dataSource1.type=com.alibaba.druid.pool.DruidDataSource

#数据源2
spring.datasource.druid.dataSource2.url=jdbc:mysql://127.0.0.1:3306/zcloud_order?useUnicode=true&characterEncoding=utf-8&allowMultiQueries=true&useTimezone=true&serverTimezone=GMT%2B8
spring.datasource.druid.dataSource2.username=root
spring.datasource.druid.dataSource2.password=zy521
spring.datasource.druid.dataSource2.driver-class-name=com.mysql.cj.jdbc.Driver
spring.datasource.druid.dataSource2.type=com.alibaba.druid.pool.DruidDataSource

3-6步骤中的3-4这两个步骤就是实现动态切换数据源的核心代码。我们在配置类DynamicDataSourceConfig中配置我们需要切换的数据源信息,并且设置默认的数据源。配置类中设置动态数据源的信息实际就是调用我们定义的动态数据源类DynamicDataSource的有参构造方法DynamicDataSource。

从该构造方法中,我们可以得知,我们设置的动态数据源信息也就是设置给了我们继承的父类AbstractRoutingDataSource中的成员变量,并且调用了父类的成员方法afterPropertiesSet()。afterPropertiesSet方法大家是不是感觉很熟悉?是的,看过springbean的生命周期的应该对该方法都不会陌生。afterPropertiesSet方法就是在一个类实现InitializingBean接口必须要重写的方法。 该方法是在bean属性设置完成后执行。在我们这个场景,其实就是在我们设置完AbstractRoutingDataSource中的成员变量后去执行。在这个方法里面我们可以看到,其实他就是对我们之前设置的成员变量,进行了一个处理,再赋值给对应的成员变量。

    public void afterPropertiesSet() {
        if (this.targetDataSources == null) {
            throw new IllegalArgumentException("Property 'targetDataSources' is required");
        } else {
            this.resolvedDataSources = CollectionUtils.newHashMap(this.targetDataSources.size());
            this.targetDataSources.forEach((key, value) -> {
                Object lookupKey = this.resolveSpecifiedLookupKey(key);
                DataSource dataSource = this.resolveSpecifiedDataSource(value);
                this.resolvedDataSources.put(lookupKey, dataSource);
            });
            if (this.defaultTargetDataSource != null) {
                this.resolvedDefaultDataSource = this.resolveSpecifiedDataSource(this.defaultTargetDataSource);
            }

        }
    }

 这些都是在项目启动的时候执行的,目的说白了就是将所有的动态数据源信息存储到一个map集合resolvedDataSources中,并设置默认的数据源resolvedDefaultDataSource。

而真正的设置数据源是使用我们自定义的动态数据源类DynamicDataSource的setDataSource()方法实现的。我们设置的数据源能够生效的原因就是,我们在继承AbstractRoutingDataSource类的时候,实现了该抽象方法determineCurrentLookupKey。该方法其实就是我们能够动态设置数据源的核心方法,说白了也就是AbstractRoutingDataSource给我们留的一个口子。

我们实现该方法的逻辑并不难,就是返回我们设置的数据源。 然后,我们看AbstractRoutingDataSource使用该方法的返回值做了什么。追溯到AbstractRoutingDataSource类,找到这个方法determineTargetDataSource中调用了我们实现的determineCurrentLookupKey方法(其实这里也就是使用了模板方法设计模式)。

protected DataSource determineTargetDataSource() {
        Assert.notNull(this.resolvedDataSources, "DataSource router not initialized");
        //调用我们设置的数据源名称
        Object lookupKey = this.determineCurrentLookupKey();
        //从我们在项目启动的时候注册到map集合resolvedDataSources中的数据源信息中get
        //该名称的数据源
        DataSource dataSource = (DataSource)this.resolvedDataSources.get(lookupKey);
        //没有的话就使用默认数据源
        if (dataSource == null && (this.lenientFallback || lookupKey == null)) {
            dataSource = this.resolvedDefaultDataSource;
        }
        //为空抛出异常
        if (dataSource == null) {
            throw new IllegalStateException("Cannot determine target DataSource for lookup key [" + lookupKey + "]");
        } else { 
            return dataSource;
        }
    }

再追溯determineTargetDataSource()方法,可以看到我们再使用getConnection()方法获取数据库连接的时候,就是调用了该方法determineTargetDataSource来获取数据源,然后根据数据源来获取数据库连接的。

至此,我们就应该明白为什么我们能够通过 AbstractRoutingDataSource来实现动态切换数据源了。

二、事务问题

1、问题描述

       在使用spring编程式事务的时候,切换数据源dataSource2会出现无法切换的情况。导致在事务中进行数据库操作的时候使用的默认数据源dataSource1,这样就会导致sql报错。因为默认数据源中并没有sql语句中进行操作的表。

事务代码:

 //开启事务
TransactionStatus transaction = transactionManager.getTransaction(transactionDefinition);
            try {
                if (!CollectionUtils.isEmpty(list)) {
                    //插入运行记录
                    traceRouteResultMapper.addDialRunRecord2(dialRunRecord);
                    //插入执行结果
                    traceRouteResultMapper.insertBatch(list);
                    //插入告警
                    if(!CollectionUtils.isEmpty(alarmListAdd)){
                        dialTestAlarmMapper.insertBatch(alarmListAdd);
                    }
                    //更新告警最新触发时间
                    if(!CollectionUtils.isEmpty(alarmUpdateTime)){
                        dialTestAlarmMapper.updateLastAlarmTime(alarmUpdateTime);
                    }
                }
                //提交事务
                transactionManager.commit(transaction);
            } catch (Exception e) {
                //回滚
                transactionManager.rollback(transaction);
                log.error("TraceRouteServiceImpl.getIndexInfo traceroute失败!", e);
            }

 map层代码:

@Mapper
public interface DialTestAlarmMapper {

    @DataSource(name = "dataSource2")
    List<DialTestAlarmPartInfoDto> queryAllTraceRouteAlarm();

    @DataSource(name = "dataSource2")
    void insertBatch(@Param("dtoList") List<DialTestAlarmDto> alarmListAdd);

    @DataSource(name = "dataSource2")
    void updateLastAlarmTime(@Param("dtoList")  List<String> alarmUpdateTimeIds);
}
2、问题原因

     为什么在事务内无法切换数据源呢?想要弄清这个原因,我们就要看看开启事务的时候,做了什么。追溯开始事务的源码,可以发现一个关键方法doBegin()。

protected void doBegin(Object transaction, TransactionDefinition definition) {
        DataSourceTransactionManager.DataSourceTransactionObject txObject = (DataSourceTransactionManager.DataSourceTransactionObject)transaction;
        Connection con = null;

        try {
            if (!txObject.hasConnectionHolder() || txObject.getConnectionHolder().isSynchronizedWithTransaction()) {
                Connection newCon = this.obtainDataSource().getConnection();
                if (this.logger.isDebugEnabled()) {
                    this.logger.debug("Acquired Connection [" + newCon + "] for JDBC transaction");
                }

                txObject.setConnectionHolder(new ConnectionHolder(newCon), true);
            }
            //获取数据库连接
            txObject.getConnectionHolder().setSynchronizedWithTransaction(true);
            con = txObject.getConnectionHolder().getConnection();
            Integer previousIsolationLevel = DataSourceUtils.prepareConnectionForTransaction(con, definition);
            txObject.setPreviousIsolationLevel(previousIsolationLevel);
            txObject.setReadOnly(definition.isReadOnly());
            if (con.getAutoCommit()) {
                txObject.setMustRestoreAutoCommit(true);
                if (this.logger.isDebugEnabled()) {
                    this.logger.debug("Switching JDBC Connection [" + con + "] to manual commit");
                }

                con.setAutoCommit(false);
            }

            this.prepareTransactionalConnection(con, definition);
            txObject.getConnectionHolder().setTransactionActive(true);
            int timeout = this.determineTimeout(definition);
            if (timeout != -1) {
                txObject.getConnectionHolder().setTimeoutInSeconds(timeout);
            }
             // 把当前的数据源的Connection与线程进行绑定
            if (txObject.isNewConnectionHolder()) {
                TransactionSynchronizationManager.bindResource(this.obtainDataSource(), txObject.getConnectionHolder());
            }

        } catch (Throwable var7) {
            if (txObject.isNewConnectionHolder()) {
                DataSourceUtils.releaseConnection(con, this.obtainDataSource());
                txObject.setConnectionHolder((ConnectionHolder)null, false);
            }

            throw new CannotCreateTransactionException("Could not open JDBC Connection for transaction", var7);
        }
    }

从该方法中,我们可以看到,在我们开启事务的时候,就完成了数据库连接connection和线程的绑定。此时,由于还未进行数据源的切换,所绑定的连接自然来自于默认的数据源。这意味着,在随后的事务处理过程中,只要事务尚未提交,所有针对数据库的操作都将通过这个已经建立好的连接来进行。因此,如果尝试在事务进行中切换数据源,则不会生效。

3、解决办法

      既然我们已经知道了问题的根本原因,解决办法也就变得相对直接明了。我们可以在开启事务之前,先手动的将数据源切换到dataSource2。这样,在随后打开事务时,将会使用来dataSource2的连接与当前线程进行绑定。通过这种预先设置的方式,我们确保了事务中使用的连接来源于我们指定的数据源,从而解决了在事务中切换数据源的问题。

            //手动设置数据源为dataSource2
            DynamicDataSource.setDataSource(DatabaseType.dataSource2.toString());
            TransactionStatus transaction = transactionManager.getTransaction(transactionDefinition);
            try {
                if (!CollectionUtils.isEmpty(list)) {
                    //插入运行记录
                    traceRouteResultMapper.addDialRunRecord2(dialRunRecord);
                    //插入执行结果
                    traceRouteResultMapper.insertBatch(list);
                    //插入告警
                    if(!CollectionUtils.isEmpty(alarmListAdd)){
                        dialTestAlarmMapper.insertBatch(alarmListAdd);
                    }
                    //更新告警最新触发时间
                    if(!CollectionUtils.isEmpty(alarmUpdateTime)){
                        dialTestAlarmMapper.updateLastAlarmTime(alarmUpdateTime);
                    }
                }
                //提交事务
                transactionManager.commit(transaction);
            } catch (Exception e) {
                //回滚
                transactionManager.rollback(transaction);
                log.error("TraceRouteServiceImpl.getIndexInfo traceroute失败!", e);
            }finally {
                //最后清除设置的数据源
                DynamicDataSource.clearDataSource();
            }

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

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

相关文章

C++内存分布 new和delete介绍

目录 C/C内存分布 栈区 堆区 静态区 常量区 C new和delete 分配空间形式对比 new delete与malloc free的区别 可不可以串着使用new和free呢 C/C内存分布 C的内存分布&#xff0c;大体上分为栈区 堆区 静态区 常量区 栈区 栈区是用于存储函数调用时的局部变量 函…

C语言中,如何判断两个数组是否包含相同元素?

在C语言中判断两个数组是否包含相同元素可以采用多种方法&#xff0c;其中最常见的方法是使用排序和比较两个数组的元素。在解释这个问题之前&#xff0c;我们需要了解一下C语言中的数组、排序算法和比较方法。 数组 数组是C语言中一种基本的数据结构&#xff0c;它是一系列相…

mysql的DDL语言和DML语言

DDL语言&#xff1a; 操作数据库&#xff0c;表等&#xff08;创建&#xff0c;删除&#xff0c;修改&#xff09;&#xff1b; 操作数据库 1&#xff1a;查询 show databases 2:创建 创建数据库 create database 数据库名称 创建数据库&#xff0c;如果不存在就创建 crea…

Linux论坛搭建

1.安装httpd服务 1.1安装httpd软件 [rootlocalhost yum.repos.d]# dnf install httpd 1.2.修改httpd的配置 [rootlocalhost yum.repos.d]# vim /etc/httpd/conf/httpd.conf 1.3.启动这个httpd服务,并查看它的状态 [rootlocalhost yum.repos.d]# systemctl start httpd [ro…

前端调用DRI后端API出现跨域资源共享(CORS)问题解决办法

目录 1. 引言2. 跨源资源共享和实现方法3. 在Django项目中配置django-cors-headers库Reference 1. 引言 在进行后端API开发时&#xff0c;有时会遇到“跨域资源共享 (CORS) 请求…被阻止“的错误&#xff0c;如图1所示。本文讲解如何在使用DRF&#xff08;Django REST Framewo…

什么是ISP,为什么跨境推荐ISP?

ISP&#xff0c;全称Internet Service Provider&#xff0c;即“互联网服务提供商”。它是为个人或企业提供访问、使用或参与互联网服务的组织&#xff0c;主要为用户提供互联网接入业务、信息业务和增值业务。ISP是经国家主管部门批准的正式运营企业&#xff0c;享受国家法律保…

多模光纤标准:OM1、OM2、OM3、OM4和OM5

【摘要】 在当今信息时代&#xff0c;光纤通信作为一种高速、高带宽的数据传输方式&#xff0c;已经成为现代通信网络的重要基石。而在光纤通信系统中&#xff0c;多模光纤因其适用于短距离传输和相对低成本而备受青睐。本文瑞哥将带大家好好了解多模光纤中的不同标准&#xff…

Error: contextBridge API can only be used when contextIsolation is enabled

在electron项目中preload.js文件使用下面的方法时报错 const { contextBridge, ipcRenderer } require(electron); contextBridge.exposeInMainWorld(electronApi, {});node:electron/js2c/renderer_init:2 Unable to load preload script: D:\Vue\wnpm\electron\preload.js …

大型企业高效内部协同,向日葵SDK私有化部署案例解析

大型集团企业的内部&#xff0c;沟通协作的重要性不言而喻&#xff0c;我们时常能听到关于所谓“大企业病”的吐槽&#xff0c;多数也是源于企业内部沟通协作效率低&#xff0c;进而导致内耗加重。甚至我们可以这么说&#xff0c;越是发展壮大的集团企业&#xff0c;其内部的沟…

java:Http协议和Tomcat

HTTP协议 Hyper Text Transfer Protocol 超文本传输协议,规定了浏览器和服务器之间数据传输的规则 特点: 基于TCP协议,面向连接,安全 基于请求响应模型:一次请求对应一次响应 HTTP协议是无状态协议,对事务的处理没有记忆能力,每次请求-响应都是独立的. 优点 速度较快 …

图论基础知识 深度优先(Depth First Search, 简称DFS),广度优先(Breathe First Search, 简称DFS)

图论基础知识 学习记录自代码随想录 dfs 与 bfs 区别 dfs是沿着一个方向去搜&#xff0c;不到黄河不回头&#xff0c;直到搜不下去了&#xff0c;再换方向&#xff08;换方向的过程就涉及到了回溯&#xff09;。 bfs是先把本节点所连接的所有节点遍历一遍&#xff0c;走到下…

平安城市 停车场 景区对讲SV-6201-T IP网络非可视对讲报警柱简介

平安城市 停车场 景区对讲SV-6201-T IP网络非可视对讲报警柱简介 18123651365微信 功能特点&#xff1a; 全金属外壳&#xff0c;户外防风雨&#xff0c;坚固耐用&#xff0c;易于识别单键呼叫&#xff0c;可通过软件指定呼叫目标&#xff0c;双向对讲广播喊话终端内置扬声器…

LabVIEW连接PostgreSql

一、安装ODBC 下载对应postgreSQL版本的ODBC 下载网址&#xff1a;http://ftp.postgresql.org/pub/odbc/versions/msi/ 下载好后默认安装就行&#xff0c;这样在ODBC数据源中才能找到。 二、配置系统DSN 实现要新建好要用的数据库&#xff0c;这里的用户名&#xff1a;postg…

SpringCloud 之 服务消费者

前提 便于理解我修改了本地域名》这里!!! 127.0.0.1 eureka7001.com 127.0.0.1 eureka7002.com 127.0.0.1 eureka7003.comRest学习实例之消费者 创建一个消费者去消费 消费者模块展示 1、导入依赖 <!-- 实体类api自己创建的模块 Web 部分依赖展示--><depe…

57-VPX电路设计

视频链接 VPX连接器电路设计01_哔哩哔哩_bilibili VPX接口电路设计 1、VPX介绍 1.1、VPX简介 VPX总线是VITA(VME International Trade Association, VME国际贸易协会)组织于2007年在其VME总线基础上提出的新一代高速串行总线标准。VPX总线的基本规范、机械结构和总线信号等…

Golang特殊init函数

介绍 init()函数是一个特殊的函数&#xff0c;存在一下特性 不能被其它函数调用&#xff0c;而是子main()函数之前自动调用不能作为参数传入不能有传入参数和返回值 作用&#xff1a; 对变量进行初始化检查/修复程序状态注册运行一次计算 以下是<<the way to go>>…

(开源版)企业AI名片S2B2C商城系统商业计划书

团队使命 擎动人工智能跃迁&#xff0c;融技术与商业之行 项目背景 话说2022年12月7日那天&#xff0c;国务院大大发布了个重磅消息&#xff0c;宣布咱们国家的三年抗疫大战终于告一段落&#xff0c;全面放开啦&#xff01;这意味着咱们的市场经济要重新焕发生机啦&#xff…

【EI会议|稳定检索】2024年航空航天、空气动力学与自动化工程国际会议(ICAAAE 2024)

2024 International Conference on Aerospace, Aerodynamics, and Automation Engineering 一、大会信息 会议名称&#xff1a;2024年航空航天、空气动力学与自动化工程国际会议 会议简称&#xff1a;ICAAAE 2024 收录检索&#xff1a;提交Ei Compendex,CPCI,CNKI,Google Schol…

短袖面料有哪些种类?质量好的短袖品牌实测推荐

夏天确实是一个让人感到炎热的季节&#xff0c;而选择合适的短袖对于提高穿着的舒适度和避免不必要的困扰非常重要。所以为了让大家能够选到一些合适自己而且质量好短袖&#xff0c;今天为大家总结了几点关键的选购技巧或一些知识科普&#xff01;并且结合我近期测评过的众多短…

【C++ STL序列容器】list 双向链表

文章目录 【 1. 基本原理 】【 2. list 的创建 】2.1 创建1个空的 list2.2 创建一个包含 n 个元素的 list&#xff08;默认值&#xff09;2.3 创建一个包含 n 个元素的 list&#xff08;赋初值&#xff09;2.4 通过1个 list 初始化另一个 list2.5 拷贝其他类型容器的指定元素创…