JDBC 结构优化2
JDBC 结构优化2
文章目录
- JDBC 结构优化2
- 结构优化2 - ATM系统(存,取,转,查)
- 1 Service
- 2 事务
- 3 ThreadLocal
- 4 事务的封装
结构优化2 - ATM系统(存,取,转,查)
1 Service
什么是业务?
- 代表用户完成的一个业务功能,可以由一个或多个DAO的调用组成。
- 软件所提供的一个功能叫业务。
核心思想:
-
用户角度: 功能即业务
- 转账这一功能就是一个业务
-
服务者角度: 业务包含多个步骤
- 从A账户扣钱
- 给B账户加钱
-
程序设计角度: 每一个步骤都是一个方法
-
AccountDao
- save() //存钱
- take() //取钱
-
AccountService
-
transfer(){dao.take(A);dao.save(B); }
-
-
**目的:**DAO提高数据操作的重用性,Service提高业务功能的重用性。
2 事务
概念:
- 事务是一个原子操作,由一个或多个SQL语句组成。
- 在同一个事务当中,所有的SQL语句要么全部执行成功,要么全部失败。
事务的边界:
- 在JDBC中,先获得Connection对象。
- 开始事务:conn.setAutoCommit(false); #set autocommit=0;
- 手动提交事务:conn.commit(); #commit
- 手动回滚事务:conn.rollback(); #rollback;
在Service中,调用了多次DAO操作。每一个业务功能都要控制事务。
在Service层中控制事务:
问题: 当程序发生异常,事务回滚并没有成功。
原因:Service控制事务和DAO访问数据库时的连接对象,并非同一个
注意: 在一个业务流程中,要保证多个数据库访问操作的连接对象是同一个。
解决方法:
- 可以将整个线程中(单线程)中,存储一个共享值。
- 线程拥有一个类似Map的属性,键值对结构<ThreadLocal对象,值>。
3 ThreadLocal
ThreadLocal(线程本地变量):
- 每一个线程通过ThreadLocal绑定一个连接对象。
- 在整个流程中任一环节可以存值或取值。
ThreadLocal应用流程:
ThreadLocal应用:
-
在释放资源的方法中,连接对象关闭后,提供threadLocal.remove()。
-
将关闭后的连接从当前ThreadLocal中移除。
4 事务的封装
现有问题:
- 事务的开启、提交、回滚的代码都写在了Service中。
- Service的主要职责是模拟现实业务功能。要将事务的控制封装在工具类中。
工具类封装事务控制方法:
- 开启事务:begin();
代码演示:
// 4 开启事务
public static void begin() throws SQLException{Connection connection = DBUtils.getConnection();if (connection != null) {connection.setAutoCommit(false);}
}
- 提交事务:commit();
代码演示:
// 5 提交事务
public static void commit() throws SQLException{Connection connection = DBUtils.getConnection();if (connection != null) {connection.commit();}
}
- 回滚事务:rollback();
代码演示:
// 6 回滚事务
public static void rollback() throws SQLException{Connection connection = DBUtils.getConnection();if (connection != null) {connection.rollback();}
}
- 关闭连接close(); 解除绑定连接
代码演示:
// 3 释放资源
public static void closeAll(Connection connection, Statement statement, ResultSet resultSet) {try {if (resultSet != null) {resultSet.close();}if (statement != null) {statement.close();}if (connection != null) {//判断是否开启事务, 如果没有开启事务, 要关闭//若开启事务, 则在事务结束后手动关闭if (connection.getAutoCommit()) {connection.close();//从线程中移除threadLocal.remove();}}} catch (SQLException e) {System.out.println("资源释放失败: "+e.getMessage());throw new RuntimeException(e);}
}
// 7 关闭事务连接
public static void close() throws SQLException{Connection connection = DBUtils.getConnection();if (connection != null) {connection.close();threadLocal.remove();}
}
ATM系统
代码演示:
Dao层
Dao接口:
public interface AccountDao {//查询Account select(String cardNo, String password);Account select(String cardNo);//更新//更新存钱void updateSave(String cardNo, BigDecimal money);//更新取钱void updateTake(String cardNo, BigDecimal money);//更新密码void updatePassword(String cardNo, String password);
}
DaoImpl:
public class AccountDaoImpl implements AccountDao {//登录@Overridepublic Account select(String cardNo, String password) {ResultSet resultSet = null;PreparedStatement preparedStatement = null;Connection connection = null;try {// 1 获取连接connection = DBUtils.getConnection();// 2 创建预编译命令String sql = "SELECT * FROM companydb.account where cardNo = ? And password = ?";preparedStatement = connection.prepareStatement(sql);// 给'?'占位符赋值preparedStatement.setObject(1, cardNo);preparedStatement.setObject(2, password);// 3 执行命令resultSet = preparedStatement.executeQuery();Account account = null;// 4 处理if (resultSet.next()) {int id = resultSet.getInt("id");String name = resultSet.getString("name");BigDecimal money = resultSet.getBigDecimal("money");account = new Account(id, name, money, cardNo, password);}return account;} catch (SQLException e) {throw new RuntimeException(e);} finally {// 5 释放资源DBUtils.closeAll(connection, preparedStatement, resultSet);}}//通过卡号查询@Overridepublic Account select(String cardNo) {ResultSet resultSet = null;PreparedStatement preparedStatement = null;Connection connection = null;try {// 1 获取连接connection = DBUtils.getConnection();// 2 创建预编译命令String sql = "SELECT * FROM companydb.account where cardNo = ?";preparedStatement = connection.prepareStatement(sql);// 参数赋值preparedStatement.setObject(1, cardNo);// 3 执行命令resultSet = preparedStatement.executeQuery();Account account = null;// 4 处理(获取结果集中的数据)if (resultSet.next()) {int id = resultSet.getInt("id");String name = resultSet.getString("name");BigDecimal money = resultSet.getBigDecimal("money");String password = resultSet.getString("password");account = new Account(id, name, money, cardNo, password);}return account;} catch (SQLException e) {throw new RuntimeException(e);} finally {//关闭资源DBUtils.closeAll(connection, preparedStatement, resultSet);}}//存钱@Overridepublic void updateSave(String cardNo, BigDecimal money) {Connection connection = null;PreparedStatement preparedStatement = null;try {//1 获取连接connection = DBUtils.getConnection();//2 创建预编译SQL命令String sql = "UPDATE companydb.account SET account.money = account.money + ? WHERE account.cardNo = ?";preparedStatement = connection.prepareStatement(sql);//2.1 参数赋值preparedStatement.setObject(1, money);preparedStatement.setObject(2, cardNo);//3 执行命令preparedStatement.executeUpdate();} catch (SQLException e) {throw new RuntimeException(e);} finally {//4 释放资源DBUtils.closeAll(connection, preparedStatement, null);}}//取钱@Overridepublic void updateTake(String cardNo, BigDecimal money) {Connection connection = null;PreparedStatement preparedStatement = null;try {//1 获取连接connection = DBUtils.getConnection();//2 创建预编译SQL命令String sql = "UPDATE companydb.account SET account.money = account.money - ? WHERE account.cardNo = ?";preparedStatement = connection.prepareStatement(sql);//2.1 参数赋值preparedStatement.setObject(1, money);preparedStatement.setObject(2, cardNo);//3 执行命令/处理preparedStatement.executeUpdate();} catch (SQLException e) {throw new RuntimeException(e);} finally {//释放资源DBUtils.closeAll(connection, preparedStatement, null);}}//修改密码@Overridepublic void updatePassword(String cardNo, String password) {Connection connection = null;PreparedStatement preparedStatement = null;try {//1 获取连接connection = DBUtils.getConnection();String sql = "UPDATE companydb.account SET account.password = ? WHERE account.cardNo = ?";//2 创建预编译SQL命令preparedStatement = connection.prepareStatement(sql);//2.1 参数赋值preparedStatement.setObject(1, password);preparedStatement.setObject(2, cardNo);//3 执行命令/处理preparedStatement.executeUpdate();} catch (SQLException e) {throw new RuntimeException(e);} finally {//4 释放资源DBUtils.closeAll(connection, preparedStatement, null);}}
}
entity
Account:
public class Account {private Integer id;private String name;private BigDecimal money;private String cardNo;private String password;public Account() {}public Account(Integer id, String name, BigDecimal money, String cardNo, String password) {this.id = id;this.name = name;this.money = money;this.cardNo = cardNo;this.password = password;}public Integer getId() {return id;}public void setId(Integer id) {this.id = id;}public String getName() {return name;}public void setName(String name) {this.name = name;}public BigDecimal getMoney() {return money;}public void setMoney(BigDecimal money) {this.money = money;}public String getCardNo() {return cardNo;}public void setCardNo(String cardNo) {this.cardNo = cardNo;}public String getPassword() {return password;}public void setPassword(String password) {this.password = password;}
}
service层
service接口:
public interface AccountService {//登录Account login(String cardNo, String password);//存钱void save(String cardNo, BigDecimal money);//取钱void take(String cardNo, BigDecimal money);//转账void trans(String from, String to, BigDecimal money);//查询余额Account queryMoney(String cardNo, String password);//修改密码void modifyPassword(String cardNo, String oldPwd, String newPwd);
}
serviceImpl:
public class AccountServiceImpl implements AccountService {//声明数据访问对象, 以调用Dao层操作private static final AccountDaoImpl accountDao = new AccountDaoImpl();//登录public Account login(String cardNo, String password) {//通过卡号和密码查询数据, 都正确时放回数据, 登录成功Account account = accountDao.select(cardNo, password);if (account == null) {throw new RuntimeException("账户或密码错误");}return account;}//存钱public void save(String cardNo, BigDecimal money) {//添加限制//因money是BigDecimal类型, 使用compareTo方法进行比较//通过相减的方式进行比较判断if (money.compareTo(new BigDecimal(0))<=0) {throw new RuntimeException("存款金额必须大于0");}if (money.compareTo(new BigDecimal(50000))>0) {throw new RuntimeException("存款金额必须小于50000");}accountDao.updateSave(cardNo, money);}//取款public void take(String cardNo, BigDecimal money) {if (money.compareTo(new BigDecimal(0))<=0) {throw new RuntimeException("取款金额必须大于0");}if (money.compareTo(new BigDecimal(20000))>0) {throw new RuntimeException("取款金额必须小于20000");}//通过卡号查询当前账户的余额Account account = accountDao.select(cardNo);if (money.compareTo(account.getMoney())>0) {throw new RuntimeException("账户余额不足...");}accountDao.updateTake(cardNo, money);}//转账public void trans(String from, String to, BigDecimal money) {if (money.compareTo(new BigDecimal(0))<=0) {throw new RuntimeException("转账金额必须大于0");}if (money.compareTo(new BigDecimal(20000))>0) {throw new RuntimeException("转账金额必须小于20000");}//通过卡号查询转账账户的余额Account account = accountDao.select(from);if (money.compareTo(account.getMoney())>0) {throw new RuntimeException("账户余额不足...");}//判断转账双方是否是同一人if (from.equals(to)) {throw new RuntimeException("不能给自己转账...");}//通过卡号查询判断接收方是否存在if (accountDao.select(to) == null) {throw new RuntimeException("对方账户不存在...");}//转账try {//开启事务, 调用线程中的连接DBUtils.begin();//使用同一连接accountDao.updateTake(from, money);accountDao.updateSave(to, money);DBUtils.commit();//手动提交} catch (Exception e) {try {DBUtils.rollback();//异常回滚} catch (SQLException ex) {e.printStackTrace();}throw new RuntimeException(e);} finally {//手动关闭try {DBUtils.close();//关闭事务连接} catch (SQLException e) {e.printStackTrace();}}}//查询余额public Account queryMoney(String cardNo, String password) {AccountDao accountDao = new AccountDaoImpl();Account account = accountDao.select(cardNo, password);if (account==null) {throw new RuntimeException("密码有误");}return account;}//修改密码public void modifyPassword(String cardNo, String oldPwd, String newPwd) {AccountDao accountDao = new AccountDaoImpl();Account account = accountDao.select(cardNo,oldPwd);if (account == null) {throw new RuntimeException("原始密码有误...");}if (oldPwd.equals(newPwd)) {throw new RuntimeException("新密码不能与原始密码相同");}accountDao.updatePassword(cardNo,newPwd);}
}
test:
ATMSystem:
public class ATMSystem {public static void main(String[] args) {Scanner scanner = new Scanner(System.in);System.out.println("请输入卡号");String cardNo = scanner.next();System.out.println("请输入密码");String password = scanner.next();//创建业务对象AccountService accountService = new AccountServiceImpl();try {Account account = accountService.login(cardNo, password);boolean flag = true;System.out.println("登录成功, 欢迎: "+account.getName());do {System.out.println("1 存钱, 2 取钱, 3 转账, 4 查询余额, 5 修改密码, 0 退出");System.out.println("请选择:");int choose = scanner.nextInt();switch (choose) {case 1:try {System.out.println("请输入存钱金额:");BigDecimal money = scanner.nextBigDecimal();accountService.save(account.getCardNo(), money);System.out.println("存钱成功...");} catch (Exception e) {System.out.println(e.getMessage());}break;case 2:try {System.out.println("请输入取款金额:");BigDecimal money = scanner.nextBigDecimal();accountService.take(account.getCardNo(), money);System.out.println("取款成功...");} catch (Exception e) {System.out.println(e.getMessage());}break;case 3:System.out.println("请输入对方的卡号:");String toCard = scanner.next();System.out.println("请输入转账金额:");BigDecimal money = scanner.nextBigDecimal();try {accountService.trans(account.getCardNo(),toCard,money);System.out.println("转账成功...");} catch (Exception e) {System.out.println(e.getMessage());}break;case 4:System.out.println("请输入密码:");String pwd = scanner.next();try {System.out.println("当前账户余额为: "+ accountService.queryMoney(account.getCardNo(), pwd).getMoney());} catch (Exception e) {System.out.println(e.getMessage());}break;case 5:System.out.println("请输入原始密码");String oldPwd = scanner.next();System.out.println("请输入新密码");String newPwd = scanner.next();System.out.println("请再次输入新密码");String againPwd = scanner.next();if (newPwd.equals(againPwd)) {try {accountService.modifyPassword(account.getCardNo(), oldPwd, newPwd);System.out.println("密码修改成功...");} catch (Exception e) {System.out.println(e.getMessage());}} else {System.out.println("两次密码不一致...");}break;case 0:flag = false;break;default:System.out.println("输入有误...");}} while (flag);System.out.println("欢迎下次光临...");} catch (Exception e) {System.out.println(e.getMessage());}}
}
util工具类
DbUtils:
public class DBUtils {private static String url;private static String user;private static String pwd;private static final ThreadLocal<Connection> threadLocal = new ThreadLocal<>();// 1 注册驱动static {try {// 读取属性配置文件Properties properties = new Properties();FileInputStream fis = new FileInputStream("Properties/db.properties");properties.load(fis);fis.close();// 变量赋值String driver = properties.getProperty("driver");url = properties.getProperty("url");user = properties.getProperty("user");pwd = properties.getProperty("pwd");// 获取驱动对象Class.forName("com.mysql.jdbc.Driver");} catch (Exception e) {System.out.println("注册驱动失败: "+e.getMessage());}}// 2 获取连接public static Connection getConnection() {try {//从线程中获取连接Connection connection = threadLocal.get();//判断线程中是否有连接, 没有则从数据库中获取if (connection == null) {connection = DriverManager.getConnection(url,user,pwd);//将连接放入线程中threadLocal.set(connection);}//若线程中有连接, 则直接返回return connection;} catch (SQLException e) {throw new RuntimeException(e);}}// 3 释放资源public static void closeAll(Connection connection, Statement statement, ResultSet resultSet) {try {if (resultSet != null) {resultSet.close();}if (statement != null) {statement.close();}if (connection != null) {//判断是否开启事务, 如果没有开启事务, 要关闭//若开启事务, 则在事务结束后手动关闭if (connection.getAutoCommit()) {connection.close();//从线程中移除threadLocal.remove();}}} catch (SQLException e) {System.out.println("资源释放失败: "+e.getMessage());throw new RuntimeException(e);}}// 4 开启事务public static void begin() throws SQLException{Connection connection = DBUtils.getConnection();if (connection != null) {connection.setAutoCommit(false);}}// 5 提交事务public static void commit() throws SQLException{Connection connection = DBUtils.getConnection();if (connection != null) {connection.commit();}}// 6 回滚事务public static void rollback() throws SQLException{Connection connection = DBUtils.getConnection();if (connection != null) {connection.rollback();}}// 7 关闭事务连接public static void close() throws SQLException{Connection connection = DBUtils.getConnection();if (connection != null) {connection.close();threadLocal.remove();}}
}
相关文章:

JDBC 结构优化2
JDBC 结构优化2 文章目录 JDBC 结构优化2结构优化2 - ATM系统(存,取,转,查)1 Service2 事务3 ThreadLocal4 事务的封装 结构优化2 - ATM系统(存,取,转,查) 1 Service 什么是业务? 代表用户完成的一个业务功能,可以由一个或多个DAO的调用组成。软件所提供的一个功…...
大模型相关术语
AGI(Artificial General Intelligence) 指通用人工智能,专注于研制像人一样思考、像人一样从事多种用途的机器。它与一般的特定领域智能(如机器视觉、语音识别等)相区分。 AIGC(AI-Generated Content&…...
数据库之九 流程控制、存储过程和函数
【零】数据准备 【1】创建用户信息表 (1)创建表 id:编号name:用户名sex:性别,默认男balance:余额register_time:注册时间 drop table if exists user; create table user( id in…...

DolphinDB学习(2):增删改查数据表(分布式表的基本操作)
文章目录 创建数据表1. 创建数据表全流程2. 核心:创建table3. 在已有的数据表中追加新的数据 数据表自身的操作1. 查询有哪些数据表2. 删除某张数据表3. 修改数据表的名称 博客里只介绍最常见的分区表(createPartitionedTable)的创建方法&…...

100天精通Python(实用脚本篇)——第114天:基于smtplib与email模块实现收发邮件(附上多个案例代码)
文章目录 专栏导读案例说明一、smtplib模块是什么?1.1 模块介绍1.2 SMTP参数说明1.3 SMTP常用方法 二、email模块是什么?1.1 模块介绍1.2 常用类说明 三、案例实战3.1 获取授权码3.2 代码步骤3.3 发送文本格式邮件3.4 发送图片格式邮件3.5 发送指定文件夹…...

redisTemplate.opsForValue()
redisTemplate 在Spring Data Redis中,redisTemplate 是一个非常重要的组件,它为开发者提供了各种操作 Redis 的方法。对于 opsForValue() 方法,它是用来获取一个操作字符串值的操作对象。这意味着你可以使用它来执行各种字符串相关的操作…...

多线程事务如何回滚?
背景介绍 1,最近有一个大数据量插入的操作入库的业务场景,需要先做一些其他修改操作,然后在执行插入操作,由于插入数据可能会很多,用到多线程去拆分数据并行处理来提高响应时间,如果有一个线程执行失败&am…...

医院如何筛选安全合规的内外网文件交换系统?
医院内外网文件交换系统是专为医疗机构设计的,用于在内部网络(内网)和外部网络(外网)之间安全、高效地传输敏感医疗数据和文件的解决方案。这种系统对于保护患者隐私、遵守医疗数据保护法规以及确保医疗服务的连续性和…...

C51 单片机学习(一):基础外设
参考 51单片机入门教程 1. 单片机简介 1.1 定义 单片机(Micro Controller Unit,简称 MCU) 内部集成了 CPU、RAM、ROM、定时器、中断系统、通讯接口等一系列电脑的常用硬件功能单片机的任务是信息采集(依靠传感器)、处…...

Docker容器引擎镜像创建
目录 一、镜像的创建 (一)基于现有镜像创建 1.启动一个镜像,在容器里做修改 2.将修改后的容器提交为新的镜像 (二)基于本地模板创建 (三)基于Dockerfile 创建 1.联合文件系统(…...

布尔逻辑与逻辑门
计算机为什么使用二进制: 计算机的元器件晶体管只有 2 种状态,通电(1)& 断电(0),用二进制可直接根据元器件的状态来设计计算机。而且,数学中的“布尔代数”分支,可以…...

opencv-python计算视频光流
光流基本概念 光流表示的是相邻两帧图像中每个像素的运动速度和运动方向。具体:光流是空间运动物体在观察成像平面上的像素运动的瞬时速度,是利用图像序列中像素在时间域上的变化以及相邻帧之间的相关性来找到上一帧跟当前帧之间存在的对应关系…...

Spring 中获取 Bean 对象的三种方式
目录 1、根据名称获取Bean 2、根据Bean类型获取Bean 3、根据 Bean 名称 Bean 类型来获取 Bean(好的解决方法) 假设 Bean 对象是 User,并存储到 Spring 中,注册到 xml 文件中 public class User {public String sayHi(){retur…...

centos系统安装Ward服务器监控工具
简介 Ward是一个简约美观多系统支持的服务器监控面板 安装 1.首先安装jdk yum install java-1.8.0-openjdk-devel.x86_64 2.下载jar wget 3.启动 java -jar ward-1.8.8.jar 体验 浏览器输入 http://192.168.168.110:4000/ 设置服务名设置为:myserver 端口号:5000 点击…...

计算机网络-数据交换方式(电路交换 报文交换 分组交换及其两种方式 )
文章目录 为什么要数据交换?总览电路交换电路交换的各个阶段建立连接数据传输释放连接 电路交换的特点电路交换的优缺点 报文交换报文交换流程报文交换的优缺点 分组交换分组交换流程分组交换的优缺点 数据交换方式的选择分组交换的两种方式数据报方式数据报方式的特…...

【C++入门到精通】特殊类的设计 | 单例模式 [ C++入门 ]
阅读导航 引言一、设计模式概念(了解)二、单例模式1. 饿汉模式(1)概念(2)模拟实现(3)优缺点(4)适用场景 2. 懒汉模式(1)概念ÿ…...

【创建vue项目的两种方式】
Vue环境搭建 NodeJs安装包安装淘宝镜像 环境搭建webpack安装全局安装vue/cli查看模板创建项目1.webpack2. vue-cli NodeJs安装包 下载链接:官网链接 下载下来后,直接傻瓜式的安装即可。 通过在cmd控制台输入以下命令查看是否安装成功 node -v因为适配某…...

2. HarmonyOS应用开发DevEcoStudio准备-1
2. HarmonyOS应用开发DevEcoStudio准备-1 下载 DevEco Studio 进入HUAWEI DevEco Studio产品页产品页。 单击下载列表右侧的按钮,下载 DevEco Studio。 安装 DevEco Studio 下载完成后,双击下载的 deveco-studio-xxxx.exe,进入 DevEco St…...

《二叉树》——3(层序遍历)
目录 前言: 层序遍历: 解析: 前言: 本文主讲链式二叉树的层序遍历,在前面的张篇blog我们初步实现了链式二叉树递归部分的内容,对于递归算法的学习和思维方式我们仍然需要不断加强,所以将对链式二叉树进行…...

HarmonyOS应用开发者基础认证考试答案
HarmonyOS应用开发者基础认证考试答案 一、判断题 1.Ability是系统调度应用的最小单元,是能够完成一个独立功能的组件。一个应用可以包含一个或多个Ability。 正确(True) 2.所有使用Component修饰的自定义组件都支持onPageShow,onBackPress和onPageHide…...

简易版抽奖活动的设计技术方案
1.前言 本技术方案旨在设计一套完整且可靠的抽奖活动逻辑,确保抽奖活动能够公平、公正、公开地进行,同时满足高并发访问、数据安全存储与高效处理等需求,为用户提供流畅的抽奖体验,助力业务顺利开展。本方案将涵盖抽奖活动的整体架构设计、核心流程逻辑、关键功能实现以及…...
STM32+rt-thread判断是否联网
一、根据NETDEV_FLAG_INTERNET_UP位判断 static bool is_conncected(void) {struct netdev *dev RT_NULL;dev netdev_get_first_by_flags(NETDEV_FLAG_INTERNET_UP);if (dev RT_NULL){printf("wait netdev internet up...");return false;}else{printf("loc…...

如何将联系人从 iPhone 转移到 Android
从 iPhone 换到 Android 手机时,你可能需要保留重要的数据,例如通讯录。好在,将通讯录从 iPhone 转移到 Android 手机非常简单,你可以从本文中学习 6 种可靠的方法,确保随时保持连接,不错过任何信息。 第 1…...

SiFli 52把Imagie图片,Font字体资源放在指定位置,编译成指定img.bin和font.bin的问题
分区配置 (ptab.json) img 属性介绍: img 属性指定分区存放的 image 名称,指定的 image 名称必须是当前工程生成的 binary 。 如果 binary 有多个文件,则以 proj_name:binary_name 格式指定文件名, proj_name 为工程 名&…...
Go 语言并发编程基础:无缓冲与有缓冲通道
在上一章节中,我们了解了 Channel 的基本用法。本章将重点分析 Go 中通道的两种类型 —— 无缓冲通道与有缓冲通道,它们在并发编程中各具特点和应用场景。 一、通道的基本分类 类型定义形式特点无缓冲通道make(chan T)发送和接收都必须准备好࿰…...

无人机侦测与反制技术的进展与应用
国家电网无人机侦测与反制技术的进展与应用 引言 随着无人机(无人驾驶飞行器,UAV)技术的快速发展,其在商业、娱乐和军事领域的广泛应用带来了新的安全挑战。特别是对于关键基础设施如电力系统,无人机的“黑飞”&…...

mac 安装homebrew (nvm 及git)
mac 安装nvm 及git 万恶之源 mac 安装这些东西离不开Xcode。及homebrew 一、先说安装git步骤 通用: 方法一:使用 Homebrew 安装 Git(推荐) 步骤如下:打开终端(Terminal.app) 1.安装 Homebrew…...
华为OD最新机试真题-数组组成的最小数字-OD统一考试(B卷)
题目描述 给定一个整型数组,请从该数组中选择3个元素 组成最小数字并输出 (如果数组长度小于3,则选择数组中所有元素来组成最小数字)。 输入描述 行用半角逗号分割的字符串记录的整型数组,0<数组长度<= 100,0<整数的取值范围<= 10000。 输出描述 由3个元素组成…...

沙箱虚拟化技术虚拟机容器之间的关系详解
问题 沙箱、虚拟化、容器三者分开一一介绍的话我知道他们各自都是什么东西,但是如果把三者放在一起,它们之间到底什么关系?又有什么联系呢?我不是很明白!!! 就比如说: 沙箱&#…...
【把数组变成一棵树】有序数组秒变平衡BST,原来可以这么优雅!
【把数组变成一棵树】有序数组秒变平衡BST,原来可以这么优雅! 🌱 前言:一棵树的浪漫,从数组开始说起 程序员的世界里,数组是最常见的基本结构之一,几乎每种语言、每种算法都少不了它。可你有没有想过,一组看似“线性排列”的有序数组,竟然可以**“长”成一棵平衡的二…...