当前位置: 首页 > news >正文

Spring学习(六)

一、AOP概念的引入(JDK动态代理)

为什么需要JDK动态代理?

在传统的业务开发中,我们经常会遇到这样的场景:需要在多个业务方法中添加相同逻辑,比如事务管理、日志记录、权限校验等。

使用JDK动态代理技术,将这些横切关注点(事务、日志等)从业务逻辑中抽取出来。

1.创建maven的项目,引入开发的坐标

<dependencies> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>5.0.2.RELEASE</version> </dependency> <dependency> <groupId>commons-logging</groupId> <artifactId>commons-logging</artifactId> <version>1.2</version> </dependency> <dependency> <groupId>log4j</groupId> <artifactId>log4j</artifactId> <version>1.2.12</version> </dependency> <!-- 有单元测试的环境,Spring5版本,Junit4.12版本 --> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.12</version> <scope>test</scope> </dependency> <!-- 连接池 --> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>1.1.10</version> </dependency> <!-- mysql驱动包 --> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>5.1.6</version> </dependency> <!-- Spring整合Junit测试的jar包 --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>5.0.2.RELEASE</version> <scope>test</scope> </dependency> </dependencies>

2.QcbyUtils类,进行事务管理

package com.qcby.Utils; import com.alibaba.druid.pool.DruidDataSource; import javax.sql.DataSource; import java.sql.Connection; import java.sql.SQLException; /** * 事务的工具类 */ public class QcbyUtils { private static DruidDataSource ds = null; // 使用ThreadLocal存储当前线程中的Connection对象 private static ThreadLocal<Connection> threadLocal = new ThreadLocal<Connection>(); // 在静态代码块中创建数据库连接池 static { try { // 通过代码创建C3P0数据库连接池 ds = new DruidDataSource(); ds.setDriverClassName("com.mysql.cj.jdbc.Driver"); ds.setUrl("jdbc:mysql:///spring_db"); ds.setUsername("root"); ds.setPassword("Xuhaoyu666!"); } catch (Exception e) { throw new ExceptionInInitializerError(e); } } /** * @Method: getConnection * @Description: 从数据源中获取数据库连接 * @Anthor: * @return Connection * @throws SQLException */ public static Connection getConnection() throws SQLException { // 从当前线程中获取Connection Connection conn = threadLocal.get(); if (conn == null) { // 从数据源中获取数据库连接 conn = getDataSource().getConnection(); // 将conn绑定到当前线程 threadLocal.set(conn); } return conn; } /** * @Method: startTransaction * @Description: 开启事务 * @Anthor: * */ public static void startTransaction() { try { Connection conn = threadLocal.get(); if (conn == null) { conn = getConnection(); // 把 conn绑定到当前线程上 threadLocal.set(conn); } // 开启事务 conn.setAutoCommit(false); } catch (Exception e) { throw new RuntimeException(e); } } /** * @Method: rollback * @Description:回滚事务 * @Anthor: */ public static void rollback() { try { // 从当前线程中获取Connection Connection conn = threadLocal.get(); if (conn != null) { // 回滚事务 conn.rollback(); } } catch (Exception e) { throw new RuntimeException(e); } } /** * @Method: commit * @Description:提交事务 * @Anthor: */ public static void commit() { try { // 从当前线程中获取Connection Connection conn = threadLocal.get(); if (conn != null) { // 提交事务 conn.commit(); } } catch (Exception e) { throw new RuntimeException(e); } } /** * @Method: close * @Description:关闭数据库连接(注意,并不是真的关闭,而是把连接还给数据库连接池) * @Anthor: * */ public static void close() { try { // 从当前线程中获取Connection Connection conn = threadLocal.get(); if (conn != null) { conn.close(); // 解除当前线程上绑定conn threadLocal.remove(); } } catch (Exception e) { throw new RuntimeException(e); } } /** * @Method: getDataSource * @Description: 获取数据源 * @Anthor: * @return DataSource */ public static DataSource getDataSource() { // 从数据源中获取数据库连接 return ds; } }

3.AccountService的接口和实现类

package com.qcby.service; import com.qcby.domain.Account; import java.sql.SQLException; public interface AccountService { public void save(Account account1, Account account2) throws SQLException; }
package com.qcby.service; import com.qcby.dao.AccountDao; import com.qcby.domain.Account; import java.sql.SQLException; public class AccountServiceImpl implements AccountService { private AccountDao accountDao; public void setAccountDao(AccountDao accountDao) { this.accountDao = accountDao; } @Override public void save(Account account1, Account account2) throws SQLException{ // try{ // 保存1账号 accountDao.save(account1); // 模拟异常 // int a = 1 / 0; // 保存2账号 accountDao.save(account2); // } catch (ArithmeticException e) { // e.printStackTrace(); // throw new SQLException("事务执行失败,发生除零错误", e); // } } }

4.AccountDao的接口和实现类

package com.qcby.dao; import com.qcby.domain.Account; public interface AccountDao { public void save(Account account); }
package com.qcby.dao; import com.qcby.Utils.QcbyUtils; import com.qcby.domain.Account; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.SQLException; public class AccountDaoImpl implements AccountDao { public void save(Account account) { try{ Connection conn = QcbyUtils.getConnection(); String sql = "insert into account values(null,?,?)"; PreparedStatement stmt = conn.prepareStatement((sql)); stmt.setString(1,account.getName()); stmt.setDouble(2,account.getMoney()); stmt.executeUpdate(); stmt.close(); } catch (SQLException e){ e.printStackTrace(); } } }

5.生成代理对象

package com.qcby.JDKProxy; import com.qcby.Utils.QcbyUtils; import com.qcby.service.AccountService; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; public class JdkProxy { public static Object getProxy(final AccountService accountService){ Object proxy = Proxy.newProxyInstance(JdkProxy.class.getClassLoader(), accountService.getClass().getInterfaces(), new InvocationHandler() { @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { Object result = null; try{ QcbyUtils.startTransaction(); result = method.invoke(accountService,args); QcbyUtils.commit(); } catch (Exception e){ e.printStackTrace(); QcbyUtils.rollback(); } return result; } }); return proxy; } }

6.测试方法

import com.qcby.JDKProxy.CglibProxy; import com.qcby.JDKProxy.JdkProxy; import com.qcby.dao.AccountDao; import com.qcby.dao.AccountDaoImpl; import com.qcby.domain.Account; import com.qcby.service.AccountService; import com.qcby.service.AccountServiceImpl; import org.junit.Test; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; import java.sql.SQLException; /** * 达内教育--腾讯课程认证机构 * 史招阳 */ public class DemoTest { @Test public void run1() throws SQLException { ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml"); // 获取service对象 AccountService accountService = (AccountService) ac.getBean("accountService"); Account account1 = new Account(null,"熊大",10000.00); Account account2 = new Account(null,"美羊羊",11000.00); // accountService.save(account1,account2); Object proxyobj = JdkProxy.getProxy(accountService); AccountService proxy = (AccountService) proxyobj; proxy.save(account1,account2); } }

二、AOP相关的概念

1.AOP的概述

什么是AOP的技术?

在软件业,AOP为Aspect Oriented Programming的缩写,意为:面向切面编程
AOP是一种编程范式,隶属于软工范畴,指导开发者如何组织程序结构
AOP最早由AOP联盟的组织提出的,制定了一套规范.Spring将AOP思想引入到框架中,必须遵守AOP联盟的规范
通过预编译方式或者运行期动态代理实现程序功能的统一维护的一种技术
AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型
利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率
AOP:面向切面编程(思想,解决OOP遇到的一些问题)
AOP采取横向抽取机制,取代了传统纵向继承体系重复性代码(性能监视、事务管理、安全检查、缓存)

为什么要学习AOP?
可以在不修改源代码的前提下,对程序进行增强!!

2. AOP的优势

运行期间,不修改源代码的情况下对已有的方法进行增强

AOP优势:
1. 减少重复代码
2. 提高开发效率
3. 维护方便

3. AOP的底层原理

JDK 的动态代理技术

1. 为接口创建代理类的字节码文件
2. 使用 ClassLoader 将字节码文件加载到 JVM
3. 创建代理类实例对象,执行对象的目标方法

CGLIB 代理技术

http://www.jsqmd.com/news/768626/

相关文章:

  • 基于Alexa与Node.js的智能DNS查询技能开发实战
  • 西南林业大学考研辅导班机构推荐:排行榜单与哪家好评测 - michalwang
  • 别再死磕手册了!Xilinx 7系列FPGA配置模式选型指南(SPI/BPI/SelectMAP/JTAG)
  • AI 算法盒子国内外主流厂商全景盘点(2026)
  • 写论文软件哪个好?2026 实测:虎贲等考 AI 凭真文献 + 全流程 + 强合规,成毕业论文首选
  • 河南师范大学考研辅导班机构推荐:排行榜单与哪家好评测 - michalwang
  • Gitee统一SCA解决方案:重新定义开源治理新范式
  • 系统右键菜单集成Cursor编辑器:一键直达提升开发效率
  • 从“解决”到“消解”:电车难题作为AI元人文的第一次工程实验
  • C++模板技术(泛型编程)
  • 基于Next.js与多模型支持的私有化AI聊天应用部署与定制指南
  • 大模型训练优化框架Socratic-Zero解析与应用
  • GPTs提示词设计指南:从原理到实践,打造专属AI助手
  • 1688运营培训/1688运营培训,16年老店铺月询盘暴涨171%
  • 基于LoRA的对话模型微调实战:从开源模型到专属AI助手
  • 熵减开发悖论突破方案:软件测试的破局之道
  • 长沙理工大学考研辅导班机构推荐:排行榜单与哪家好评测 - michalwang
  • 2026 热门网页游戏推荐,耐玩不氪金的网页游戏大盘点
  • AI赋能:让快马平台生成能理解内容与风格的智能Pinterest下载器
  • 用STC15单片机+DS1302做个简易电子钟?附完整工程代码和数码管显示避坑指南
  • 深度拆解Scrapy Selector:XPath实战手册,从入门到高吞吐量抓取架构
  • Kubernetes Operator开发脚手架:从CRD定义到生产就绪的完整实践
  • 抛丸区高大空间供暖选垂直送风型适配吗?
  • 软考高级网络规划设计师教程(第3版)
  • SwiftUI与WebSocket构建iOS原生IM应用:从原理到实战
  • 长江大学考研辅导班机构推荐:排行榜单与哪家好评测 - michalwang
  • 短剧拉片网站2026推荐,满足多样分析需求
  • 高安全等级建筑中紧固件如何保证可靠性_2026上海紧固件专业展
  • AI 写论文哪个软件最好?2026 实测:虎贲等考 AI 凭全流程合规 + 真文献实证,稳坐毕业论文神器榜首
  • 基于RAG的长文本智能处理系统:从原理到工程实践