网站首页 > java教程 正文
#Spring核心 #IOC容器 #AOP编程 #Java框架设计
一、IOC(控制反转):对象管理的革命
1.1 传统开发 vs IOC模式对比
对象管理方式 | 传统开发 | Spring IOC |
对象创建 | 开发者手动new创建 | 容器自动实例化并注入 |
依赖管理 | 硬编码依赖关系 | 通过配置/注解声明依赖 |
耦合度 | 高耦合,难以替换组件 | 低耦合,易扩展和维护 |
通俗理解:
将对象的“控制权”从程序员手中交给Spring容器,就像从“自己做饭”变成“点外卖”——只需声明需求,容器负责配送。
1.2 IOC核心实现:Bean容器与依赖注入
1.2.1 Bean的生命周期
实例化 → 属性填充 → 初始化 → 使用 → 销毁
1.2.2 依赖注入方式
方式 | 代码示例 | 特点 |
构造器注入 | new UserService(userRepo) | 强依赖,不可变 |
Setter注入 | userService.setUserRepo(userRepo) | 灵活,支持可选依赖 |
注解注入 | @Autowired private UserRepo userRepo; | 简洁,主流选择 |
1.2.3 配置方式对比
// XML配置(传统)
<bean id="userService" class="com.example.UserService">
<property name="userRepo" ref="userRepo"/>
</bean>
// Java Config(现代)
@Configuration
public class AppConfig {
@Bean
public UserService userService(UserRepo userRepo) {
return new UserService(userRepo);
}
}
// 注解驱动(最简)
@Service
public class UserService {
@Autowired
private UserRepo userRepo;
}
二、AOP(面向切面编程):横切关注点的优雅解决方案
2.1 AOP核心概念图解
┌──────────────┐
│ 切面(Aspect) │
└──────┬───────┘
│
┌───────────┼───────────┐
▼ ▼ ▼
┌─────────┐ ┌─────────┐ ┌─────────┐
│ 日志记录 │ │ 事务管理 │ │ 权限校验 │
└─────────┘ └─────────┘ └─────────┘
2.2 关键术语解析
术语 | 说明 | 代码示例 |
切面(Aspect) | 封装横切逻辑的模块 | @Aspect public class LogAspect { ... } |
连接点(JoinPoint) | 可插入切面的方法执行点 | public void userService.saveUser() |
通知(Advice) | 切面在连接点的执行逻辑 | @Before("execution(* save*(..))") |
切点(Pointcut) | 匹配连接点的表达式 | @Pointcut("within(com.service.*)") |
2.3 AOP代理原理与实现
2.3.1 两种代理方式对比
代理类型 | JDK动态代理 | CGLIB代理 |
实现原理 | 基于接口 | 基于类继承 |
性能 | 调用快,创建慢 | 创建快,调用稍慢 |
限制 | 目标类必须实现接口 | 可代理无接口类(final类除外) |
2.3.2 示例:日志切面实现
@Aspect
@Component
public class LogAspect {
// 定义切点:拦截所有Service层方法
@Pointcut("execution(* com.example.service.*.*(..))")
public void serviceLayer() {}
// 前置通知:记录方法入参
@Before("serviceLayer()")
public void logMethodParams(JoinPoint jp) {
String methodName = jp.getSignature().getName();
Object[] args = jp.getArgs();
System.out.println("方法 " + methodName + " 参数: " + Arrays.toString(args));
}
// 环绕通知:计算方法执行时间
@Around("serviceLayer()")
public Object logExecutionTime(ProceedingJoinPoint pjp) throws Throwable {
long start = System.currentTimeMillis();
Object result = pjp.proceed();
long duration = System.currentTimeMillis() - start;
System.out.println(pjp.getSignature() + " 执行耗时: " + duration + "ms");
return result;
}
}
三、IOC与AOP的协同作用
3.1 协作流程示例
1. IOC容器创建Bean实例
2. AOP代理机制介入
↓
3. 生成代理对象(JDK/CGLIB)
↓
4. 将代理对象注入到其他Bean
3.2 典型应用场景
技术 | 应用场景 | 实现方式 |
IOC | 依赖管理、模块解耦 | @Autowired, @Component |
AOP | 日志、事务、安全、性能监控 | @Aspect, @Transactional |
四、常见问题解答
Q1:IOC和DI有什么区别?
- IOC(控制反转):设计理念,将对象创建权交给容器
- DI(依赖注入):实现IOC的具体技术手段
Q2:AOP会降低程序性能吗?
- 代理创建阶段:略有开销(尤其是CGLIB)
- 运行阶段:几乎无影响,现代JVM优化能力强
Q3:如何选择JDK代理和CGLIB?
- 强制使用CGLIB:
- spring.aop.proxy-target-class=true
五、最佳实践指南
5.1 IOC配置原则
- 优先使用注解驱动(@ComponentScan + @Autowired)
- 明确Bean作用域(默认单例,必要时使用@Scope("prototype"))
- 避免循环依赖(可通过@Lazy延迟加载解决)
5.2 AOP使用技巧
- 切点表达式优化:
- // 精确匹配Service层
@Pointcut("within(@org.springframework.stereotype.Service *)") - 优先使用@Around实现复杂逻辑
- 通过@Order控制多个切面的执行顺序
六、动手实验
任务1:实现依赖注入
// 要求:通过构造器注入实现UserService与UserRepository的解耦
public class UserService {
private final UserRepository userRepo;
public UserService(UserRepository userRepo) {
this.userRepo = userRepo;
}
}
任务2:创建事务切面
// 要求:自定义注解@MyTransactional,实现方法级事务管理
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface MyTransactional {}
@Aspect
@Component
public class TransactionAspect {
@Around("@annotation(MyTransactional)")
public Object manageTransaction(ProceedingJoinPoint pjp) {
// 实现事务开启/提交/回滚逻辑
}
}
资源推荐:
- Spring官方文档 - Core Technologies
- 《Spring实战(第6版)》:深度解析IOC/AOP实现原理
互动讨论:你在项目中是如何组合使用IOC和AOP的?遇到过哪些挑战?欢迎留言分享!
- 上一篇: 应聘者:什么,第一轮面试就用这么简单的算法题?
- 下一篇:已经是最后一篇了
猜你喜欢
- 2025-05-02 应聘者:什么,第一轮面试就用这么简单的算法题?
- 2025-05-02 Go slice切片使用教程,一次通关!
- 2025-05-02 几经反转 谷歌胜诉后甲骨文能否接招?丨C位
- 2025-05-02 java组件HuTool相关工具类的使用二
- 2025-05-02 深入解析 Java 链表:从构建到逆序打印的技术之旅
- 2025-05-02 2023年Java基础面试题目收集整理归纳(持续更新)
- 2025-05-02 我的世界:末地烛的“奇妙”玩法?关于末地棒,你不知道的11件事
- 2025-05-02 程序员去大公司面试,Java岗大厂面试官常问的那些问题,进阶学习
- 2025-05-02 90%人踩过的坑!Java运算符优先级与类型转换的终极避坑指南!
- 2025-05-02 如何判断回文数?不要再将整数转为字符串来解决这个问题了
你 发表评论:
欢迎- 最近发表
- 标签列表
-
- java反编译工具 (77)
- java反射 (57)
- java接口 (61)
- java随机数 (63)
- java7下载 (59)
- java数据结构 (61)
- java 三目运算符 (65)
- java对象转map (63)
- Java继承 (69)
- java字符串替换 (60)
- 快速排序java (59)
- java并发编程 (58)
- java api文档 (60)
- centos安装java (57)
- java调用webservice接口 (61)
- java深拷贝 (61)
- 工厂模式java (59)
- java代理模式 (59)
- java.lang (57)
- java连接mysql数据库 (67)
- java重载 (68)
- java 循环语句 (66)
- java反序列化 (58)
- java时间函数 (60)
- java是值传递还是引用传递 (62)
本文暂时没有评论,来添加一个吧(●'◡'●)