news 2026/5/7 7:10:38

Spring学习(六)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
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 代理技术

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/7 7:08:27

1688运营培训/1688运营培训,16年老店铺月询盘暴涨171%

1688运营培训/1688运营培训&#xff0c;16年老店铺月询盘暴涨171% 1688店铺开店16年始终没效果&#xff0c;经过1688运营培训一个月&#xff0c;询盘暴涨171%&#xff0c;他到底是怎么做到的&#xff1f; 今天跟大家分享的这个店铺&#xff0c;主营汽摩配件。以前他的线下客…

作者头像 李华
网站建设 2026/5/7 7:01:32

开源AI 3D生成平台ClaraVerse:从提示词到可交互世界的构建与实战

1. 项目概述与核心价值最近在探索AI驱动的3D内容生成与交互领域时&#xff0c;我深度体验了一个名为ClaraVerse的开源项目。这个项目并非一个简单的工具集合&#xff0c;而是一个旨在构建“AI原生3D世界”的综合性平台。它的核心愿景&#xff0c;是让创造复杂的3D场景和角色变得…

作者头像 李华
网站建设 2026/5/7 6:59:03

实战指南:基于快马平台快速开发全栈tokenpo质押收益农场应用

今天想和大家分享一个最近在InsCode(快马)平台上快速搭建的实战项目——tokenpo质押收益农场应用。这个项目完美模拟了DeFi领域常见的代币质押和流动性挖矿场景&#xff0c;特别适合想学习全栈开发的朋友练手。 项目背景与需求分析 代币质押是区块链应用中非常经典的功能&#…

作者头像 李华
网站建设 2026/5/7 6:56:30

2026 最新 Claude Code 保姆级安装配置教程,国内开发者直接可用

前言 Claude Code 是目前终端端最强的 AI 编程助手之一&#xff0c;可以直接在命令行中和 AI 实时协作编程、读代码、改项目、排查 BUG&#xff0c;极大提升开发效率。 但官方原版配置对国内开发者很不友好&#xff0c;网络、API 鉴权、接口地址都是大难题。今天给大家整理一套…

作者头像 李华
网站建设 2026/5/7 6:55:35

轻量级RPC框架yarmcp:Go语言高性能微服务通信实践

1. 项目概述&#xff1a;一个轻量级、高性能的RPC框架最近在梳理团队内部微服务通信的选型&#xff0c;发现很多项目还在用HTTP API进行服务间调用&#xff0c;虽然简单直接&#xff0c;但性能开销、服务治理和接口契约管理上总感觉差了点意思。市面上成熟的RPC框架不少&#x…

作者头像 李华