亲宝软件园·资讯

展开

Spring IOC Bean注解属性注入

把苹果咬哭的测试笔记 人气:0

基于注解方式进行属性注入

涉及到 4 个注解

一、@Autowired

现在有这样的一个结构,我要在 service 里注入 dao 对象。

1. 添加对象注解

给 service 和 dao 添加注解,用来创建对象。

为了区分明显,service 层用 @Service,dao 层用 @Repository。

// content of UserService
@Service
public class UserService {
    public void add() {
        System.out.println("service add() ... ...");
    }
}
@Repository
public class UserDaoImpl implements UserDao{
    @Override
    public void add() {
        System.out.println("UserDao add()... ...");
    }
}

2. 在service中注入dao对象

@Service
public class UserService {
    // 定义 dao 属性
    @Autowired
    private UserDao userDao;
    public void add() {
        System.out.println("service add() ... ...");
        userDao.add();
    }
}

先定义 dao 属性,然后在上面添加注解 @Autowired 即可,而且不需要我们手动添加 set 方法了。

为了方便看到注入成功效果,里面还调用了 userDao.add() 方法。

3. 创建测试函数测试效果

public class TestService {
    @Test
    public void testService() {
        ApplicationContext context
                = new ClassPathXmlApplicationContext("bean1.xml");
        UserService userService = context.getBean("userService", UserService.class);
        System.out.println(userService);
        userService.add();
    }
}

这里调用了 userService.add() 方法,而 userService.add() 内部又调用了 userDao.add(),所以应该可以看到 2 个输出:

com.pingguo.spring5.service.UserService@23f7d05d
service add() ... ...
UserDao add()... ...
Process finished with exit code 0

二、@Qualifier

根据属性名称进行注入,跟 @Autowired 一起使用。

为什么要一起使用?

比如现在,我新增一个 dao 的实现类 UserDaoImpl2

@Repository
public class UserDaoImpl2 implements UserDao {
    @Override
    public void add() {
        System.out.println("UserDao UserDaoImpl2 add()... ...");
    }
}

所以现在 UserDaoImpl 和 UserDaoImpl2 这2个类都是同一种类型,那么 @Autowired 注入的时候到底是用哪个类呢?于是报错了。

所以,要加上 @Qualifier 来指定其中一个的具体名称。比如,我这里使用 UserDaoImpl2 。

再次执行测试函数:

com.pingguo.spring5.service.UserService@75f32542
service add() ... ...
UserDao UserDaoImpl2 add()... ...
Process finished with exit code 0

可以看到 UserDaoImpl2 里的内容被输出,注入正确。

三、@Resource

既可以根据类型注入,也可以根据名称注入。

1. 替代 @Autowired

可以直接替代 @Autowired 使用。

现在我把 UserDaoImpl2 这个实现类注释掉,只留下一个实现类。

service 里使用 @Resource 注解:

@Service
public class UserService {
    // 定义 dao 属性
    @Resource
    private UserDao userDao;
    public void add() {
        System.out.println("service add() ... ...");
        userDao.add();
    }
}

运行测试:

com.pingguo.spring5.service.UserService@23f7d05d
service add() ... ...
UserDao add()... ...
Process finished with exit code 0

结果正确。

2. 替换 @Qualifier

替换 @Qualifier 使用的话,里面的值用 name 传递,比如 @Resource(name = "userDaoImpl2")

现在去掉 UserDaoImpl2 的注释,让 dao 有 2 个实现类。

service 中 使用 @Resource 注解:

@Service
public class UserService {
    // 定义 dao 属性
    @Resource(name = "userDaoImpl2")
    private UserDao userDao;
    public void add() {
        System.out.println("service add() ... ...");
        userDao.add();
    }
}

运行测试:

com.pingguo.spring5.service.UserService@197d671
service add() ... ...
UserDao UserDaoImpl2 add()... ...
Process finished with exit code 0

结果正确。

这里要提一下的是,Resource 是来自拓展包 javax。另外 2个 则是来自官方框架包,优先使用。

四、@Value

上面使用的都是注入对象类型的属性,使用 @Value 可以注入普通类型,比如 String。

在 service 里增加一个 String 类型的属性 userName,并且在下面的方法里做个打印输出。

@Service
public class UserService {
    // 定义 dao 属性
    @Autowired
    @Qualifier(value = "userDaoImpl2")
    private UserDao userDao;
    @Value(value = "spring 从0开始")
    private String userName;
    public void add() {
        System.out.println("service add() ... ...");
        System.out.println(userName);
        userDao.add();
    }
}

运行测试:

com.pingguo.spring5.service.UserService@184cf7cf
service add() ... ...
spring 从0开始
UserDao UserDaoImpl2 add()... ...
Process finished with exit code 0

注入成功。

加载全部内容

相关教程
猜你喜欢
用户评论