【mybatis-plus】自定义多数据源,动态切换数据源事务失效问题

news2025/1/13 10:30:10

背景

做了一个和navicat一样的工具,web版工具,然后数据库链接信息都是存在一个主数据库表的里,所以这里涉及到了动态切换数据源,以及一些事务等。今天说下多数据源切换时,事务失效。

目录

 一、常见的事务失效

@Transactional

1、@Transactional 应用在非 public 修饰的方法上

2、@Transactional 注解属性 rollbackFor 设置错误

3、同一个类中方法调用,导致@Transactional失效

4、异常被你的 catch“吃了”导致@Transactional失效

5、数据库引擎不支持事务

6、开启多线程任务时,事务管理会受到影响

二、案例代码

数据库表

代码一

代码二

我的解决代码


 

 一、常见的事务失效

@Transactional

1、@Transactional 应用在非 public 修饰的方法上

事务拦截器在目标方法执行前后进行拦截,内部会调用方法来获取Transactional 注解的事务配置信息,调用前会检查目标方法的修饰符是否为 public,不是 public则不会获取@Transactional 的属性配置信息。

2、@Transactional 注解属性 rollbackFor 设置错误

rollbackFor 可以指定能够触发事务回滚的异常类型。Spring默认抛出了未检查unchecked异常(继承自 RuntimeException 的异常)或者 Error才回滚事务;其他异常不会触发回滚事务。如果在事务中抛出其他类型的异常,但却期望 Spring 能够回滚事务,就需要指定rollbackFor属性。

3、同一个类中方法调用,导致@Transactional失效

开发中避免不了会对同一个类里面的方法调用,比如有一个类Test,它的一个方法A,A再调用本类的方法B(不论方法B是用public还是private修饰),但方法A没有声明注解事务,而B方法有。则外部调用方法A之后,方法B的事务是不会起作用的。这也是经常犯错误的一个地方。
那为啥会出现这种情况?其实这还是由于使用Spring AOP代理造成的,因为只有当事务方法被当前类以外的代码调用时,才会由Spring生成的代理对象来管理。

4、异常被你的 catch“吃了”导致@Transactional失效

如果你手动的catch捕获这个异常并进行处理,事务管理器会认为当前事务应该正常commit,就会导致注解失效,如果非要捕获且不失效,就必须在代码块内throw new Exception抛出异常。

5、数据库引擎不支持事务

开启事务的前提就是需要数据库的支持,我们一般使用的Mysql引擎时支持事务的,所以一般不会出现这种问题。

6、开启多线程任务时,事务管理会受到影响

因为线程不属于spring托管,故线程不能够默认使用spring的事务,也不能获取spring注入的bean在被spring声明式事务管理的方法内开启多线程,多线程内的方法不被事务控制。
如下代码,线程内调用insert方法,spring不会把insert方法加入事务就算在insert方法上加入@Transactional注解,也不起作用。

被外部调用的公共方法A有两个进行了数据操作的子方法B和子方法C的事务注解说明:

1.被外部调用的公共方法A未声明事务@Transactional,子方法B和C若是其他类的方法且各自声明事务,则事务由子方法B和C各自控制

2.被外部调用的公共方法A未声明事务@Transactional,子方法B和C若是本类的方法,则无论子方法B和C是否声明事务,事务均不会生效

3.被外部调用的公共方法A声明事务@Transactional,无论子方法B和C是不是本类的方法,无论子方法B和C是否声明事务,事务均由公共方法A控制

4.被外部调用的公共方法A声明事务@Transactional,子方法运行异常,但运行异常被子方法自己 try-catch 处理了,则事务回滚是不会生效的!

二、案例代码

数据库表

 

以上是目前数据库的数据,我接下来要做的是修改uuid为1的数据,和新增两条数据,以及删除uuid为2和3的,同时操作来演示事务。

代码一

@Api(tags = "动态数据源管理")
@RestController
@RequestMapping("/hvit/dataResource/")
public class SysDataResourceController {

    @Autowired
    private SysDataResourceDataService sysDataResourceDataService;

    @ApiOperation("数据集查看-->插入/编辑/删除行")
    @PostMapping("/insertRowDataToDataTable")
    public ResponseEntity insertRowDataToDataTable(@RequestBody RowDataReq rowDataReq) {
        return ResponseEntity.ok(sysDataResourceDataService.insertRowDataToDataTable(rowDataReq));
    }
@Slf4j
@Service
public class SysDataResourceDataService {

    @Autowired
    private SysDataDirectoryDataService sysDataDirectoryDataService;
    @Autowired
    private SysUserService sysUserService;
    @Autowired
    private SysTableStructureService sysTableStructureService;
    @Autowired
    private DBService dbService;
    @Autowired
    private SysDataResourceDataService sysDataResourceDataService; 
/***
     * 数据集查看-->插入/编辑/删除行
     * @param rowDataReq
     * @return
     * 被外部调用的公共方法A未声明事务@Transactional,子方法B和C若是其他类的方法且各自声明事务,则事务由子方法B和C各自控制
     */
    public R insertRowDataToDataTable(RowDataReq rowDataReq) {
        //try {
        if (StringUtils.isAnyEmpty(rowDataReq.getTableName(), rowDataReq.getPoolName())) {
            return R.error("缺少参数!");
        }
        //先获取表的主键
        SysDataResource sysDataResource = sysDataResourceService.getById(rowDataReq.getPoolName());
        String prikey = "";
        if (sysDataResource != null) {
            prikey = dbService.getDataTablePrimaryKey(rowDataReq.getTableName(), sysDataResource.getDataResourceName());
        }
        insertAndUpdateAndDeleteDataTableData(rowDataReq, prikey);
        return R.ok();
    }


    @Transactional(rollbackFor = Exception.class)
    public void insertAndUpdateAndDeleteDataTableData(RowDataReq rowDataReq, String prikey) {
        //处理插入行
        if (!CollectionUtils.isEmpty(rowDataReq.getAddRowData())) {
            rowDataReq.getAddRowData().forEach(x -> {
                StringBuffer keyString = new StringBuffer();
                StringBuffer valueString = new StringBuffer();
                x.forEach((key, value) -> {
                    keyString.append(key + ",");
                    valueString.append("'" + value + "'" + ",");
                });
                String k = keyString.toString().substring(0, keyString.length() - 1);
                String v = valueString.toString().substring(0, valueString.length() - 1);
                sysDataResourceDataService.insertDataTableData(rowDataReq.getTableName(), k, v);
            });
        }
        //处理编辑行
        if (!CollectionUtils.isEmpty(rowDataReq.getUpdateRowData())) {
            if (StringUtils.isNotEmpty(prikey)) {
                String finalPrikey = prikey;
                rowDataReq.getUpdateRowData().forEach(x -> {
                    StringBuffer updateString = new StringBuffer();
                    StringBuffer conditionStr = new StringBuffer();
                    x.forEach((key, value) -> {
                        if (key.equals(finalPrikey)) {
                            conditionStr.append(key + "=" + "'" + value + "'");
                        }
                        updateString.append(key + "=" + "'" + value + "'" + ",");
                    });
                    String u = updateString.toString().substring(0, updateString.length() - 1);
                    sysDataResourceDataService.updateDataTableData(rowDataReq.getTableName(), u, conditionStr.toString());
                });
            }
        }
        //处理删除
        if (!CollectionUtils.isEmpty(rowDataReq.getDeleteRowData())) {
            if (StringUtils.isNotEmpty(prikey)) {
                String finalPrikey = prikey;
                rowDataReq.getDeleteRowData().forEach(x -> {
                    StringBuffer conditionStr = new StringBuffer();
                    x.forEach((key, value) -> {
                        if (key.equals(finalPrikey)) {
                            conditionStr.append(key + "=" + "'" + value + "'");
                        }
                    });
                    sysDataResourceDataService.deleteDataTableData(rowDataReq.getTableName(), conditionStr.toString());
                });
            }
        }
    }

/***
     * 动态插入表数据
     * @param tableName
     * @param columnsName
     * @param values
     */
    @ChangeDB
    public void insertDataTableData(String tableName, String columnsName, String values) {
        sysDataResourceMapper.insertDataTableData(tableName, columnsName, values);
    }

    /***
     * 获取数据库主键
     * @param tableName
     * @param dataResourceName
     * @return
     */
    @ChangeDB
    public String getDataTablePrimaryKey(String tableName, String dataResourceName) {
        return sysDataResourceMapper.getDataTablePrimaryKey(tableName, dataResourceName);
    }

    /***
     * 修改表数据
     * @param tableName
     * @param updateString
     * @param condition
     */
    @ChangeDB
    public void updateDataTableData(String tableName, String updateString, String condition) {
        sysDataResourceMapper.updateDataTableData(tableName, updateString, condition);
    }

    /***
     * 删除表数据
     * @param tableName
     * @param condition
     */
    @ChangeDB
    public void deleteDataTableData(String tableName, String condition) {
        sysDataResourceMapper.deleteDataTableData(tableName, condition);
    }

}
@ChangeDB这个是自定义注解,用于动态切换数据源。这个下一期我会说,如何aop+自定义注解来动态切换数据源。

我们来看下这个代码,不知道大家有没有发现为什么类内的方法不直接调用?而是自己把自己注入进spring中,然后再调用?大家可以思考下!

接下来进入正题,大家可以一眼发现这个调用其实事务是不会生效的,因为主方法并没有使用@Transactional(rollbackFor = Exception.class),即使子方法使用了,但是依旧不会生效。这符合上面说的:被外部调用的公共方法A未声明事务@Transactional,子方法B和C若是本类的方法,则无论子方法B和C是否声明事务,事务均不会生效。

代码二

@Slf4j
@Service
public class SysDataResourceDataService {

    @Autowired
    private SysDataDirectoryDataService sysDataDirectoryDataService;
    @Autowired
    private SysUserService sysUserService;
    @Autowired
    private SysTableStructureService sysTableStructureService;
    @Autowired
    private DBService dbService;
    @Autowired
    private SysDataResourceDataService sysDataResourceDataService; 
/***
     * 数据集查看-->插入/编辑/删除行
     * @param rowDataReq
     * @return
     * 被外部调用的公共方法A未声明事务@Transactional,子方法B和C若是其他类的方法且各自声明事务,则事务由子方法B和C各自控制
     */
    public R insertRowDataToDataTable(RowDataReq rowDataReq) {
        //try {
        if (StringUtils.isAnyEmpty(rowDataReq.getTableName(), rowDataReq.getPoolName())) {
            return R.error("缺少参数!");
        }
        //先获取表的主键
        SysDataResource sysDataResource = sysDataResourceService.getById(rowDataReq.getPoolName());
        String prikey = "";
        if (sysDataResource != null) {
            prikey = dbService.getDataTablePrimaryKey(rowDataReq.getTableName(), sysDataResource.getDataResourceName());
        }
        insertAndUpdateAndDeleteDataTableData(rowDataReq, prikey);
        return R.ok();
    }

   
    public void insertAndUpdateAndDeleteDataTableData(RowDataReq rowDataReq, String prikey) {
        //处理插入行
        if (!CollectionUtils.isEmpty(rowDataReq.getAddRowData())) {
            rowDataReq.getAddRowData().forEach(x -> {
                StringBuffer keyString = new StringBuffer();
                StringBuffer valueString = new StringBuffer();
                x.forEach((key, value) -> {
                    keyString.append(key + ",");
                    valueString.append("'" + value + "'" + ",");
                });
                String k = keyString.toString().substring(0, keyString.length() - 1);
                String v = valueString.toString().substring(0, valueString.length() - 1);
                dbService.insertDataTableData(rowDataReq.getTableName(), k, v);
            });
        }
        //处理编辑行
        if (!CollectionUtils.isEmpty(rowDataReq.getUpdateRowData())) {
            if (StringUtils.isNotEmpty(prikey)) {
                String finalPrikey = prikey;
                rowDataReq.getUpdateRowData().forEach(x -> {
                    StringBuffer updateString = new StringBuffer();
                    StringBuffer conditionStr = new StringBuffer();
                    x.forEach((key, value) -> {
                        if (key.equals(finalPrikey)) {
                            conditionStr.append(key + "=" + "'" + value + "'");
                        }
                        updateString.append(key + "=" + "'" + value + "'" + ",");
                    });
                    String u = updateString.toString().substring(0, updateString.length() - 1);
                    dbService.updateDataTableData(rowDataReq.getTableName(), u, conditionStr.toString());
                });
            }
        }
        //处理删除
        if (!CollectionUtils.isEmpty(rowDataReq.getDeleteRowData())) {
            if (StringUtils.isNotEmpty(prikey)) {
                String finalPrikey = prikey;
                rowDataReq.getDeleteRowData().forEach(x -> {
                    StringBuffer conditionStr = new StringBuffer();
                    x.forEach((key, value) -> {
                        if (key.equals(finalPrikey)) {
                            conditionStr.append(key + "=" + "'" + value + "'");
                        }
                    });
                    dbService.deleteDataTableData(rowDataReq.getTableName(), conditionStr.toString());
                });
            }
        }
    }


}

dbService:

/***
     * 动态插入表数据
     * @param tableName
     * @param columnsName
     * @param values
     */
    @ChangeDB
    @Transactional(rollbackFor = Exception.class)
    public void insertDataTableData(String tableName, String columnsName, String values) {
        sysDataResourceMapper.insertDataTableData(tableName, columnsName, values);
    }

    /***
     * 获取数据库主键
     * @param tableName
     * @param dataResourceName
     * @return
     */
    @ChangeDB
    public String getDataTablePrimaryKey(String tableName, String dataResourceName) {
        return sysDataResourceMapper.getDataTablePrimaryKey(tableName, dataResourceName);
    }

    /***
     * 修改表数据
     * @param tableName
     * @param updateString
     * @param condition
     */
    @ChangeDB
    @Transactional(rollbackFor = Exception.class)
    public void updateDataTableData(String tableName, String updateString, String condition) {
        sysDataResourceMapper.updateDataTableData(tableName, updateString, condition);
    }

    /***
     * 删除表数据
     * @param tableName
     * @param condition
     */
    @ChangeDB
    @Transactional(rollbackFor = Exception.class)
    public void deleteDataTableData(String tableName, String condition) {
        sysDataResourceMapper.deleteDataTableData(tableName, condition);
    }

 案例二的代码是SysDataResourceDataService内方法调用其他类的加了事务的方法。可以看到dbservice的几个方法都加了事务。

看代码这种方法报错后,其实也不会事务回滚的,因为它属于方法调用了两个其他类事务的方法,简单点说也就是各管各的了,所以再insertRowDataToDataTable方法内调用报错是无法使用事务回滚的,这种正好属于上面说的:被外部调用的公共方法A未声明事务@Transactional,子方法B和C若是其他类的方法且各自声明事务,则事务由子方法B和C各自控制

我的解决代码


     * 数据集查看-->插入/编辑/删除行
     * @param rowDataReq
     * @return
     * 被外部调用的公共方法A未声明事务@Transactional,子方法B和C若是其他类的方法且各自声明事务,则事务由子方法B和C各自控制
     */
    public R insertRowDataToDataTable(RowDataReq rowDataReq) {
        //try {
        if (StringUtils.isAnyEmpty(rowDataReq.getTableName(), rowDataReq.getPoolName())) {
            return R.error("缺少参数!");
        }
        //先获取表的主键
        SysDataResource sysDataResource = sysDataResourceService.getById(rowDataReq.getPoolName());
        String prikey = "";
        if (sysDataResource != null) {
            prikey = dbService.getDataTablePrimaryKey(rowDataReq.getTableName(), sysDataResource.getDataResourceName());
        }
        dbService.insertAndUpdateAndDeleteDataTableData(rowDataReq, prikey);
        return R.ok();
    }

可以看到主方法没有加 @Transactional,我们直接调用的是dbService中insertAndUpdateAndDeleteDataTableData,没错,我们把方法提取到另一个类中了。

dbService:

    @ChangeDB
    @Transactional(rollbackFor = Exception.class)
    public void insertAndUpdateAndDeleteDataTableData(RowDataReq rowDataReq, String prikey) {
        //处理插入行
        if (!CollectionUtils.isEmpty(rowDataReq.getAddRowData())) {
            rowDataReq.getAddRowData().forEach(x -> {
                StringBuffer keyString = new StringBuffer();
                StringBuffer valueString = new StringBuffer();
                x.forEach((key, value) -> {
                    keyString.append(key + ",");
                    valueString.append("'" + value + "'" + ",");
                });
                String k = keyString.toString().substring(0, keyString.length() - 1);
                String v = valueString.toString().substring(0, valueString.length() - 1);
                sysDataResourceMapper.insertDataTableData(rowDataReq.getTableName(), k, v);
            });
        }
        //处理编辑行
        if (!CollectionUtils.isEmpty(rowDataReq.getUpdateRowData())) {
            if (StringUtils.isNotEmpty(prikey)) {
                String finalPrikey = prikey;
                rowDataReq.getUpdateRowData().forEach(x -> {
                    StringBuffer updateString = new StringBuffer();
                    StringBuffer conditionStr = new StringBuffer();
                    x.forEach((key, value) -> {
                        if (key.equals(finalPrikey)) {
                            conditionStr.append(key + "=" + "'" + value + "'");
                        }
                        updateString.append(key + "=" + "'" + value + "'" + ",");
                    });
                    String u = updateString.toString().substring(0, updateString.length() - 1);
                    sysDataResourceMapper.updateDataTableData(rowDataReq.getTableName(), u, conditionStr.toString());
                });
            }
        }
        //处理删除
        if (!CollectionUtils.isEmpty(rowDataReq.getDeleteRowData())) {
            if (StringUtils.isNotEmpty(prikey)) {
                String finalPrikey = prikey;
                rowDataReq.getDeleteRowData().forEach(x -> {
                    StringBuffer conditionStr = new StringBuffer();
                    x.forEach((key, value) -> {
                        if (key.equals(finalPrikey)) {
                            conditionStr.append(key + "=" + "'" + value + "'");
                        }
                    });
                    sysDataResourceMapper.deleteDataTableData(rowDataReq.getTableName(), conditionStr.toString());
                });
            }
        }
    }

改成如上代码后,经过测试,修改操作报错了,新增的部分同时也会滚了。

这只是我在开发中遇到的问题。如果雷同,可以参考下。

下期:我们说下怎么通过aop+自定义注解来配置动态数据源,以及动态切换数据源。 

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

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

相关文章

大数据必学Java基础(一百一十二):开发案例之登录验证

文章目录 开发案例之登录验证 一、准备实体类 二、准备一些页面和静态资源

Spring boot day1

Maven 进阶 spring boot 框架 spring boot 相比于SSM框架的优缺点 优点: 创建独立的sping应用程序内嵌的tomcat,无需部署war文件简化的maven配置自动配置spring提供生产就绪型功能,如指标,健康检查和外部配置 特性:…

3.1 卷积神经网络基础

文章目录计算机视觉的发展历程卷积神经网络卷积(Convolution)卷积计算填充(padding)步幅(stride)感受野(Receptive Field)多输入通道、多输出通道和批量操作飞桨卷积API介绍卷积算子…

碰焊机触摸屏软件开发

1 软件需求 人机界面为MCGS 10寸触摸屏1062KW,接口为RS232/RS484/以太网,供电电源为24V。机触摸屏的功能为机器人的基本状态信息显示和对机器人的控制及参数给定。 目前设计的小车分成两栏共10格,需要将对应的触摸屏大致分成相应的10大格&a…

【嵌入式】AI落地部署技能

针对不同平台对生成的模型进行转换,也就是俗称的parse、convert,即前端解释器针对转化后的模型进行优化在特定的平台(嵌入端或者服务端)成功运行已经转化好的模型在模型可以运行的基础上,保证模型的速度、精度和稳定性用C、cuda写算子(预处理…

AWS-Eventbridge-事件总线实践

aws eventbridge 功能主要使用场景为,针对账号发生各种事件,发生告警,可以发送sns告警,也可以配置发送到当前区域或者其他账号当前区域的事件总线上汇总,再做下一步规划eventbridge 事件总线类似与整个事件的一个目录集…

机器学习——03决策树算法

机器学习——03决策树算法 参考资料 AIlearningMachine-Learning-in-Action庞善民.西安交通大学机器学习导论2022春PPT 具体算法的原理步骤请参考本人的另一篇博文:[机器学习导论]—— 第四课——决策树 一、信息熵与信息增益 🔥信息熵 信息熵使信息…

中小企业OA系统的设计与实现

开发工具(eclipse/idea/vscode等): 数据库(sqlite/mysql/sqlserver等): 功能模块(请用文字描述,至少200字): 模块划分:通知类型模块、通知信息模块、部门模块、员工模块、考勤模块、工资模块、奖惩类型、奖惩信息、请假…

USACO开赛!这份参赛指引必看!

美国信息学奥赛USACO 2022 – 2023赛季首场比赛于明日(12月16日)开始!这份参赛指引,一定要认真看噢~ 第一步:打开 USACO 官网:http://www.usaco.org/,点击【注册新账号】,开始注册账…

宽凳科技完成超亿元B1轮融资 率先突破高精地图量产落地

近日,国内领先的高精地图及其智能应用综合解决方案服务商宽凳科技宣布完成B1轮超亿元融资。本轮融资由聚焦于新能源汽车产业链投资及新兴技术产业投资的紫峰资本与信益资本联合领投,崇业投资跟投,同时本轮资本引入了德清政府战略投资&#xf…

WebDAV之葫芦儿·派盘+Xplore

Xplore 支持WebDAV方式连接葫芦儿派盘。 手机文件太多、太乱,本地目录中找不想要的文件,怎么办?推荐使用Xplore将手机中的文件以不同的文件方式罗列出来,并展示给用户。文件管理器以图片、音乐、视频、文档、压缩包及安装包等类型进行分类,使手机中的文件一目了然的分列…

离线地图开发包

相关教程: 1、如何搭建离线地图开发环境 视频教程 2、下载离线地图数据(金字塔瓦片数据) 视频教程 3、下载离线地图地形数据库(实现地表高低起伏) 4、添加离线地图数据到本地服务器 (含3D) 视频…

【身份证识别】BP神经网络身份证号码识别【含Matlab源码 1344期】

⛄一、身份证号码识别简介(附课题作业报告) 1 引言 当今是一个信息高度发达的时代,对于每个公民而言身份证那一连串的数字体现了个人信息的唯一性,出于保障公民合法权益和社会治安的考虑,越来越多的行业都开始建立自己…

数据价值深度挖掘,分析服务上线“探索”能力

近日,华为分析服务6.9.0版本发布,正式上线探索能力。开发者可自由定义与配置分析模型,支持报告实时预览,数据洞察体验更加灵活与便捷。 新上线的探索能力中,有漏斗分析、事件归因、会话路径分析三个高级分析模型。在原…

45-Jenkins-Sidebar Link插件实现添加侧边栏

Sidebar Link插件实现添加侧边栏前言安装插件使用插件自由风格项目使用Pipeline项目使用前言 本篇来学习下使用Sidebar Link插件在项目侧边栏添加自定义功能按钮链接 安装插件 Manage Jenkins --> Mangage Plugins --> 可选插件 --> 输出框输入 Sidebar Links 使…

【猿如意】中的『Bluefish』工具详情介绍

目录 一、工具名称 二、下载安装渠道 2.1 什么是猿如意? 2.2 如何下载猿如意? 2.3 如何在猿如意中下载开发工具? 三、Bluefish工具功能简介 四、Bluefish的下载和安装 4.1下载Bluefish 4.2安装Bluefish 五、Bluefish的基本使用 5.1…

Linux学习02-主机管理与磁盘分区

1 Linux与硬件的搭配 实际上,在Linux服务器中,内存的重要性比CPU重要。因为内存不够就会使用到硬盘的内存交换分区。 显卡对于不需要X-Windows的服务器来哦说,是最不重要的一个组件。 各设备在Linux中的文件命名。 在Linux系统中&#xf…

Linux常用调试工具

编译阶段 Linux入门 nm 获取二进制文件包含的符号信息 strings 获取二进制文件包含的字符串常量 strip 去除二进制文件包含的符号 readelf 显示目标文件详细信息 objdump 尽可能反汇编出源代码 addr2line 根据地址查找代码行 运行阶段 gdb 强大的调试工具 ldd 显示程序需…

大咖说|云端即时渲染:下一代互联网的算力基座?

阿里云【大咖说】子系列【计算讲谈社】第十五讲播出! 下一代互联网是什么?其算力基座又是什么? 14:00-15:30 全网播出:【计算讲谈社】第十五讲,蔚领时代创始人兼CEO郭建君、蔚领时代数字人事业部总经理费元华、蔚领时…

云工作站这5大新功能不来体验一下吗?

哈喽,大家周五好哇!赞奇云工作站又有更新大动作啦,此次更新包括子账号登录设备限制、内客户端控制台视觉优化、新增子账号删除、客户端支持工作区网络的接入方式、桌面名称功能,下面就一起来看看更新的具体内容吧—— 为了提高企业…