JDBC 结构优化2

JDBC 结构优化2

文章目录

  • JDBC 结构优化2
    • 结构优化2 - ATM系统(存,取,转,查)
      • 1 Service
      • 2 事务
      • 3 ThreadLocal
      • 4 事务的封装

结构优化2 - ATM系统(存,取,转,查)

1 Service

什么是业务?

  • 代表用户完成的一个业务功能,可以由一个或多个DAO的调用组成。
  • 软件所提供的一个功能叫业务。

核心思想:

  • 用户角度: 功能即业务

    • 转账这一功能就是一个业务
  • 服务者角度: 业务包含多个步骤

    1. 从A账户扣钱
    2. 给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();}}
}

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[NCTF2019]Fake XML cookbook(特详解)

先试了一下弱口令&#xff0c;哈哈习惯了 查看页面源码发现xml function doLogin(){var username $("#username").val();var password $("#password").val();if(username "" || password ""){alert("Please enter the usern…

数据据库八之 视图、触发器、事务

【零】准备数据 【1】创建表 &#xff08;1&#xff09;部门表 d_id是部门的编号d_name是部门的名字 # 确保表不存在 drop table if exists department; # 创建表 create table department( d_id int auto_increment primary key, d_name varchar(6) )auto_increment 501 …

腾讯云部署vue+node项目

文章目录 一、安装宝塔二、vue项目部署三、node项目部署 前言: 关于项目部署,一开始也是找了很多资料,费了点时间,所以记录一下。希望能对各位有所帮助。 一、安装宝塔 1.首先在控制台,进入云服务器的终端界面 2.输入命令和密码获取权限,并且安装宝塔界面 yum install -y w…

如何把png改成jpg格式?分享5个好用的方法!

你是否经常遇到需要将PNG格式的图片转换成JPG格式的情况&#xff1f;无论是因为工作需要&#xff0c;还是为了适应不同的平台和设备&#xff0c;格式转换都是我们经常要面对的问题。那么&#xff0c;如何快速、简单地完成这个任务呢&#xff1f;今天&#xff0c;我们就来为你揭…

计算机网络——IP协议

前言 网络层的主要负责地址分配和路由选择,ip负责在网络中进行数据包的路由和传输。 IPv4报文组成&#xff08;了解&#xff09; IPv4首部&#xff1a;IPv4首部包含了用于路由和传输数据的控制信息&#xff0c;其长度为20个字节&#xff08;固定长度&#xff09;。 版本&#…

图灵之旅--ArrayList顺序表LinkedList链表栈Stack队列Queue

目录 线性表顺序表ArrayList简介ArrayList使用ArrayList的构造ArrayList常见操作ArrayList的遍历ArrayList的扩容机制利用ArrayList洗牌ArrayList的优缺点 链表链表的实现双向链表的实现 LinkedListLinkedList引入LinkedList的使用LinkedList的构造LinkedList的常用方法介绍Lin…

Power ModeII 插件的下载与使用-----idea

下载 Marketplace里面搜索下载即可 使用 下载后重启软件就可以用了 下面是一些关于Power ModeII &#xff0c;我的个性化设置截图 以及相关设置解释 插件或扩展的设置面板【用于给代码编辑器或集成开发环境&#xff08;IDE&#xff09;添加视觉效果】 主要设置 ENTER POWE…

Maven安装,学习笔记,详细整理maven的一些配置

Maven 1. 初识Maven 2. Maven概述 Maven模型介绍 Maven仓库介绍 Maven安装与配置 3. IDEA集成Maven 4. 依赖管理 01. Maven课程介绍 1.1 课程安排 学习完前端Web开发技术后&#xff0c;我们即将开始学习后端Web开发技术。做为一名Java开发工程师&#xff0c;后端 Web开发技术…

STM32——USART

一、通信 1.1通信是什么&#xff1b; 通信是将一个设备的数据发送到另一个设备中&#xff0c;从而实现硬件的扩展&#xff1b; 1.2通信的目的是什么&#xff1b; 实现硬件的扩展-在STM32中集成了很多功能&#xff0c;例如PWM输出&#xff0c;AD采集&#xff0c;定时器等&am…

数据结构和算法笔记5:堆和优先队列

今天来讲一下堆&#xff0c;在网上看到一个很好的文章&#xff0c;不过它实现堆是用Golang写的&#xff0c;我这里打算用C实现一下&#xff1a; Golang: Heap data structure 1. 基本概念 满二叉树&#xff08;二叉树每层节点都是满的&#xff09;&#xff1a; 完全二叉树&a…

STM32标准库——(5)EXTI外部中断

1.中断系统 中断&#xff1a;在主程序运行过程中&#xff0c;出现了特定的中断触发条件&#xff08;中断源&#xff09;&#xff0c;使得CPU暂停当前正在运行的程序&#xff0c;转而去处理中断程序&#xff0c;处理完成后又返回原来被暂停的位置继续运行 中断优先级&#xff…

【QT+QGIS跨平台编译】之十一:【libzip+Qt跨平台编译】(一套代码、一套框架,跨平台编译)

文章目录 一、libzip介绍二、文件下载三、文件分析四、pro文件五、编译实践一、libzip介绍 libzip是一个开源C库,用于读取,创建和修改zip文件。 libzip可以从数据缓冲区,文件或直接从其他zip归档文件直接复制的压缩数据中添加文件。在不关闭存档的情况下所做的更改可以还原…

uniapp微信小程序-input默认字的样式

需要的是这样的 问题 正常是在input框上面写样式就行&#xff0c;但是uniapp不起作用 解决 直接在input上写placeholder-style"color就解决了 <input class"findInput" type"text" placeholder"关键词查询"placeholder-style"co…

Phoncent博客,探索Rie Kudan的GPT创作之举

近日&#xff0c;大家都在谈论日本作家Rie Kudan&#xff0c;她凭借其小说《东京共鸣塔》&#xff08;"Tokyo-to Dojo-to"&#xff09;荣获了日本极具声望的芥川奖。这本小说引起了广泛的讨论和思考&#xff0c;因为令人惊讶的是&#xff0c;Kudan在其中直接引用了人…

2023美赛A题之Lotka-Volterra【完整思路+代码】

这是2023年的成功&#xff0c;考虑到曾经付费用户的负责&#xff0c;2024年可以发出来了。去年我辅导队伍数量&#xff1a;15&#xff0c;获奖M为主&#xff0c;个别F&#xff0c;H&#xff0c;零S。言归正传&#xff0c;这里我开始分享去年的方案。由于时间久远&#xff0c;我…

【华为 ICT HCIA eNSP 习题汇总】——题目集9

1、缺省情况下&#xff0c;广播网络上 OSPF 协议 Hello 报文发送的周期和无效周期分别为&#xff08;&#xff09;。 A、10s&#xff0c;40s B、40s&#xff0c;10s C、30s&#xff0c;20s D、20s&#xff0c;30s 考点&#xff1a;①路由技术原理 ②OSPF 解析&#xff1a;&…

【ArcGIS微课1000例】0099:土地利用变化分析

本实验讲述在ArcGIS软件中基于两期土地利用数据,做土地利用变化分析。 文章目录 一、实验描述二、实验过程三、注意事项一、实验描述 对城市土地利用情况进行分析时,需要考虑不同时期土地利用图层在空间上的差异性,如农用地转建筑用地的空间变化。而该变化过程表现为各时期…

【LeetCode】排序精选12题

目录 排序&#xff1a; 1. 合并区间&#xff08;中等&#xff09; 2. 数组的相对排序&#xff08;简单&#xff09; 快速排序&#xff1a; 1. 颜色分类&#xff08;中等&#xff09; 2. 排序数组&#xff08;中等&#xff09; 3. 数组中的第K个最大元素&#xff08;中等…

HCIA-HarmonyOS设备开发认证-3.内核基础

目录 前言目标一、进程与线程待续。。。 前言 对于任何一个操作系统而言&#xff0c;内核的运行机制与原理是最为关键的部分。本章内容从多角度了解HarmonyOS的内核运行机制&#xff0c;涵盖进程与线程的概念&#xff0c;内存管理机制&#xff0c;网络特性&#xff0c;文件系统…

Arduino开发实例-DRV8833电机驱动器控制直流电机

DRV8833电机驱动器控制直流电机 文章目录 DRV8833电机驱动器控制直流电机1、DRV8833电机驱动器介绍2、硬件接线图3、代码实现DRV8833 使用 MOSFET,而不是 BJT。 MOSFET 的压降几乎可以忽略不计,这意味着几乎所有来自电源的电压都会传递到电机。 这就是为什么 DRV8833 不仅比基…