依赖注入(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;
    @Autowired
    private DependencyB b;
    @Autowired
    private 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;
    
    // 可选赋值用setter
    public void setOptDep(OptionalDependency optDep) {
        this.optDep = optDep;
    }
}

8.3 谨慎用字段自动赋值

public class CarefulService {
    @Autowired  // 只在确实必要时用
    private HardToTestComponent component;
}

总结:超越简单赋值的智慧

确实可以把DI和自动装配理解为一种特殊的赋值方式,但这种赋值:

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

就像做菜:

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

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

你可能感兴趣的:(java,开发语言,spring,DI,Ioc)