依赖注入(DI)与自动装配:本质就是“赋值“吗?
一、最简化的理解:确实可以看作"赋值"
用最直白的语言来说,依赖注入(DI)和自动装配的核心确实可以理解为"给对象的属性赋值"。但这种赋值不是普通的赋值,而是一种有特定规则和优点的赋值方式。
1.1 普通赋值 vs DI赋值
普通赋值:
// 普通赋值方式
public class OrderService {private OrderRepository repository;public OrderService() {this.repository = new OrderRepositoryImpl(); // 自己new对象赋值}
}
DI赋值:
// DI赋值方式
public class OrderService {private OrderRepository repository;// 由外部传入(注入)值public OrderService(OrderRepository repository) {this.repository = repository; // 别人传给我,我不自己new}
}
1.2 自动装配就是"自动赋值"
自动装配更进一步,连手动传值都不需要了,容器会自动找到合适的值赋给属性:
public class OrderService {@Autowired // 自动找到OrderRepository类型的bean赋值到这里private OrderRepository repository;
}
二、为什么这种"赋值"方式特别?
虽然本质是赋值,但这种赋值方式有三大特殊之处:
2.1 赋值权反转了
- 传统方式:对象自己负责给自己属性赋值
- DI方式:别人(容器)负责给你的属性赋值
就像:
- 你自己去超市买菜(传统方式)
- 外卖小哥把菜送到你家(DI方式)
2.2 赋值的内容更灵活
可以赋不同的实现类而不改代码:
// 测试时赋Mock值
OrderService service = new OrderService(new MockOrderRepository());// 生产时赋真实值
OrderService service = new OrderService(new JdbcOrderRepository());
2.3 赋值的时机和方式可控
Spring可以控制:
- 什么时候赋值(立即或延迟)
- 赋同一个值给所有人(单例)或每人新值(原型)
- 赋值前后执行什么操作(生命周期回调)
三、从"赋值"角度理解DI的核心概念
3.1 Bean就是待赋值的对象
@Bean // 告诉Spring:这个对象需要你管理赋值
public DataSource dataSource() {return new HikariDataSource();
}
3.2 依赖就是需要被赋的值
public class UserService {// 这两个属性需要被"赋值"private UserRepository userRepository;private PasswordEncoder passwordEncoder;
}
3.3 容器就是负责赋值的"大管家"
Spring容器的工作流程:
- 找到所有需要赋值的对象(Bean)
- 找出所有需要被赋的值(依赖)
- 按照规则把值赋给对象(装配)
四、通过"赋值"理解三种注入方式
4.1 构造器注入 - 创建时就赋值
public class ProductService {private ProductRepository repo;// 创建ProductService时必须赋值public ProductService(ProductRepository repo) {this.repo = repo;}
}
4.2 Setter注入 - 创建后单独赋值
public class PaymentService {private PaymentGateway gateway;// 专门留个赋值入口public void setGateway(PaymentGateway gateway) {this.gateway = gateway;}
}
4.3 字段注入 - 直接给属性赋值
public class ReportService {@Autowired // 直接给这个字段赋值private DataProvider provider;
}
五、"赋值"视角看自动装配的规则
5.1 byType - 按类型匹配赋值
@Autowired
private UserRepository repository;
// 找UserRepository类型的bean来赋值
5.2 byName - 按属性名匹配赋值
@Autowired
private UserRepository userRepository;
// 找名为"userRepository"的bean来赋值
5.3 @Qualifier - 指定赋哪个值
@Autowired
@Qualifier("jdbcUserRepo") // 明确指定赋哪个值
private UserRepository repository;
六、为什么这种"赋值"方式更好?
6.1 赋值更灵活
想换实现?只需换赋的值,不用改类代码:
// 配置A:赋MySQL实现
@Bean
public UserRepository userRepository() {return new JdbcUserRepository();
}// 配置B:赋MongoDB实现
@Bean
public UserRepository userRepository() {return new MongoUserRepository();
}
6.2 测试更方便
测试时可以赋模拟值:
@Test
void testService() {// 赋一个Mock值用于测试UserService service = new UserService(new MockUserRepository());
}
6.3 依赖关系更清晰
所有赋值需求一目了然:
public class OrderService {// 明确声明我需要这些值才能工作private final OrderRepository repo;private final PaymentService payment;private final NotificationService notify;public OrderService(OrderRepository repo, PaymentService payment,NotificationService notify) {//...赋值}
}
七、这种"赋值"的局限性
7.1 太依赖运行时
普通赋值编译时就能发现问题,DI赋值可能到运行时才报错:
@Autowired
private NonExistentBean bean; // 编译不报错,运行才失败
7.2 赋值过程不直观
不知道值从哪里来:
@Service
public class MyService {@Autowired // 魔法般的出现了值private MysteryComponent component;
}
7.3 过度赋值风险
可能赋了不需要的值:
public class SimpleService {@Autowired // 其实只需要A,但BC也被自动赋进来了private DependencyA a;@Autowiredprivate DependencyB b;@Autowiredprivate DependencyC c;
}
八、最佳"赋值"实践建议
8.1 优先用构造器赋值
public class GoodService {private final Dependency dep;// 明确所有必需的赋值public GoodService(Dependency dep) {this.dep = dep;}
}
8.2 必要处用Setter赋值
public class FlexibleService {private OptionalDependency optDep;// 可选赋值用setterpublic void setOptDep(OptionalDependency optDep) {this.optDep = optDep;}
}
8.3 谨慎用字段自动赋值
public class CarefulService {@Autowired // 只在确实必要时用private HardToTestComponent component;
}
总结:超越简单赋值的智慧
确实可以把DI和自动装配理解为一种特殊的赋值方式,但这种赋值:
- 把赋值权交给了容器
- 使赋值更灵活可控
- 让对象更专注于业务逻辑
- 使系统更易于测试和维护
就像做菜:
- 传统方式:自己种菜、自己切菜、自己做菜
- DI方式:有人送菜上门(还能按需换菜),你只管烹饪
理解了这个核心,就能更自然地掌握Spring的DI和自动装配机制。