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

依赖注入(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容器的工作流程:

  1. 找到所有需要赋值的对象(Bean)
  2. 找出所有需要被赋的值(依赖)
  3. 按照规则把值赋给对象(装配)

四、通过"赋值"理解三种注入方式

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和自动装配理解为一种特殊的赋值方式,但这种赋值:

  1. 把赋值权交给了容器
  2. 使赋值更灵活可控
  3. 让对象更专注于业务逻辑
  4. 使系统更易于测试和维护

就像做菜:

  • 传统方式:自己种菜、自己切菜、自己做菜
  • DI方式:有人送菜上门(还能按需换菜),你只管烹饪

理解了这个核心,就能更自然地掌握Spring的DI和自动装配机制。

相关文章:

  • PBKDF2全面指南(SpringBoot实现版)
  • AI agents系列之智能体框架介绍
  • Docker华为云创建私人镜像仓库
  • K-均值聚类机器学习算法的优缺点
  • C++第三方库【JSON】nlohman/json
  • CefSharp浏览器(AntdUI.Tabs)标签页关闭时资源释放ChromiumWebBrowser示例源码
  • 【文献笔记】LLM-based and retrieval-augmented control code generation
  • SmolVLM新模型技术解读笔记
  • 联邦学习与协作学习:数据隐私与模型协同进化的未来之路
  • 在SpringBoot中访问 static 与 templates 目录下的内容
  • 在 MySQL 单表存储 500 万数据的场景下,如何设计读取
  • 冲刺高分!挑战7天一篇nhanes机器学习SCI!DAY1-7
  • 1023 Have Fun with Numbers
  • Python基础语法——常量变量
  • 【Linux】进程的程序替换、自定义shell命令行解释器
  • 批量将多个文件按扩展名分类到不同文件夹
  • 如何实现动态请求地址(baseURL)
  • 数据库案例1--视图和索引
  • lvs + keepalived + dns 高可用
  • 嵌入式开发
  • 建投读书会·东西汇流|西风东渐中的上海营造
  • 第六季了,姐姐们还能掀起怎样的风浪
  • 恒安集团创始人许连捷逝世:白手起家缔造百亿纸品巨头,个人曾捐赠超10亿
  • 希音、Temu告知美国消费者4月25日起涨价:关税变化导致运营成本上升
  • 工作坊|早期左翼文学的多重张力与历史回响
  • 企业跨境支付的最大挑战及解决方案