‘@Autowired’ 和‘@Inject’的报错信息完全相同,他们都是通过 ‘AutowiredAnnotationBeanPostProcessor’ 类实现的依赖注入,二者具有可互换性。 ‘@Resource’通过 ‘CommonAnnotationBeanPostProcessor’ 类实现依赖注入,即便如此他们在依赖注入时的表现还是极为相近的,以下是他们在实现依赖注入时执行顺序的概括:
@Autowired and @Inject
- Matches by Type
- Restricts by Qualifiers
- Matches by Name
@Resource
- Matches by Name
- Matches by Type
- Restricts by Qualifiers (ignored if match is found by name)
‘@Resource’在依据name注入的时候速度性能表现的比 ‘@Autowired’ 和‘@Inject’优越,但这是微不足道的,不足以作为优先选择 ‘@Resource’的原因。我倾向于使用 ‘@Resource’是因为它配置起来更简洁。
@Resource(name="person")
@Autowired@Qualifier("person")
@Inject@Qualifier("person")
你也许会说使用字段 默认 名称作为注入时候的bean name,其他两种方式就会一样简洁:
@Resourceprivate Party person;
@Autowiredprivate Party person;
@Injectprivate Party person;
确实如此。但是当你需要重构代码的时候又如何呢?使用’@Resource‘方式只需简单修改name属性即可,而无需触及注入Bean的名称(注入Bean的时候同意使用接口名称)。所以我建议使用注解方式实现注入的时候遵循以下语法风格:
1.在你的组件中明确限定bean名称而不是使用默认值 [@Component("beanName")]。
2.同时使用’@Resource‘和它的’name'属性 [@Resource(name="beanName")]。
3.避免使用‘@Qualifier’注解,除非你要创建一系列类似beans的集合。例如,你也许需要建立一个set集合来存放一系列“规则”定义。这个时候可以选择‘@Qualifier'注解方式。这种方式使得将大量遵循相同规则的类放入集合中变得容易。
4.使用如下配置限定需要尽心组件扫描的包: [context:component-scan base-package="com.sourceallies.person"]。这样做可以减小spring扫描很多无效的包的情况。
遵循以上原则能增强你的,注解风格的,spring配置的可读性和稳定性。
反射机制
反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问、检测和修改它本身状态或行为的一种能力。
对于Java来说,就是在运行状态中,查询这个类、对象的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法、修改它的属性。
而所谓的依赖注入就是指在运行时(不是编译时)改变对象的属性或者调用对象的构造方法。
其实依赖注入是很容易实现的,你自己可以尝试一下,Spring的伟大之处并不在于技术有多难,而是在于他把这种技术发展成一种开发模式和框架。
具体java反射相关,参考 http://blog.csdn.net/liujiahan629629/article/details/18013523
模拟spring解析autowired,参考 http://blog.csdn.net/pingnanlee/article/details/45917639
注解原理 http://blog.csdn.net/rommel1/article/details/17096141
基于反射和注解机制,简单模拟Spring解析Autowired注解的过程。
1、自定义Autowired注解,代码如下
- package com.basic.reflect;
- import java.lang.annotation.ElementType;
- import java.lang.annotation.Retention;
- import java.lang.annotation.RetentionPolicy;
- import java.lang.annotation.Target;
- @Retention(RetentionPolicy.RUNTIME)
- @Target({ ElementType.FIELD, ElementType.METHOD })
- public @interface Autowired {
- //注解的name属性
- public String name() default "";
- }
2、定义相关的业务和Dao类,以及获取Bean的容器(在容器中基于反射实现自动注入)
- public class PersonDao {
- public int add(Object o) {
- System.out.println("dao autowird ok ");
- return 0;
- }
- }
- public class ServiceImpl {
- @Autowired
- private PersonDao personDao;
- public int addPerson(Object obj) {
- return personDao.add(obj);
- }
- }
- public class BeanContainer {
- public static Object getBean(String name) {
- try {
- Class<?> clazz = Class.forName("com.basic.reflect.ServiceImpl");
- Object bean = clazz.newInstance();
- Field[] fileds = clazz.getDeclaredFields();
- for (Field f : fileds) {
- if (f.isAnnotationPresent(Autowired.class)) {
- // 基于类型注入
- Class<?> c = f.getType();
- Object value = c.newInstance();
- //允许访问private字段
- f.setAccessible(true);
- //把引用对象注入属性
- f.set(bean, value);
- }
- }
- return bean;
- } catch (Exception e) {
- e.printStackTrace();
- }
- return null;
- }
- }
3、测试类
- public class Test {
- public static void main(String[] args) {
- ServiceImpl impl = (ServiceImpl) BeanContainer.getBean("service");
- String name = "test";
- impl.addPerson(name);
- }
- }
总结,上面的代码简单模拟了Spring解析Autowired的过程,写的比较简单,但是精髓已经写到了,主要就是利用反射机制生成实例并且解析注解对其属性进行赋值。其实Spring框架的核心Ioc和Aop的实现,还是利用了java最基本的东西,比如Aop就是利用了动态代理,Ioc就是利用了反射机制