Spring4新特性——泛型限定式依赖注入

Spring 4.0已经发布RELEASE版本,不仅支持Java8,而且向下兼容到JavaSE6/JavaEE6,并移出了相关废弃类,新添加如Java8的支 持、Groovy式Bean定义DSL、对核心容器进行增强、对Web框架的增强、Websocket模块的实现、测试的增强等。其中两个我一直想要的增 强就是:支持泛型依赖注入、对cglib类代理不再要求必须有空参构造器了。具体更新请参考:

http://docs.spring.io/spring/docs/4.0.0.RELEASE/spring-framework-reference/htmlsingle/#new-in-4.0

 

1、相关代码:

1.1、实体

Java代码   收藏代码
  1. public class User implements Serializable {  
  2.     private Long id;  
  3.     private String name;  
  4. }  
  5.   
  6. public class Organization implements Serializable {  
  7.     private Long id;  
  8.     private String name;  
  9. }  

 1.2、Repository

Java代码   收藏代码
  1. public abstract class BaseRepository<M extends Serializable> {  
  2.     public void save(M m) {  
  3.         System.out.println("=====repository save:" + m);  
  4.     }  
  5. }  
  6.   
  7. @Repository  
  8. public class UserRepository extends BaseRepository<User> {  
  9. }  
  10.   
  11. @Repository  
  12. public class OrganizationRepository extends BaseRepository<Organization> {  
  13. }  

 对于Repository,我们一般是这样实现的:首先写一个模板父类,把通用的crud等代码放在BaseRepository;然后子类继承后,只需要添加额外的实现。

 

1.3、Service

1.3.1、以前Service写法

Java代码   收藏代码
  1. public abstract class BaseService<M extends Serializable> {  
  2.     private BaseRepository<M> repository;  
  3.     public void setRepository(BaseRepository<M> repository) {  
  4.         this.repository = repository;  
  5.     }  
  6.     public void save(M m) {  
  7.         repository.save(m);  
  8.     }  
  9. }  
  10. @Service  
  11. public class UserService extends BaseService<User> {  
  12.     @Autowired  
  13.     public void setUserRepository(UserRepository userRepository) {  
  14.         setRepository(userRepository);  
  15.     }  
  16. }  
  17.   
  18. @Service  
  19. public class OrganizationService extends BaseService<Organization> {  
  20.     @Autowired  
  21.     public void setOrganizationRepository(OrganizationRepository organizationRepository) {  
  22.         setRepository(organizationRepository);  
  23.     }  
  24. }  

 

可以看到,以前必须再写一个setter方法,然后指定注入的具体类型,然后进行注入;

 

1.3.2、泛型Service的写法

Java代码   收藏代码
  1. public abstract class BaseService<M extends Serializable> {  
  2.     @Autowired  
  3.     protected BaseRepository<M> repository;  
  4.   
  5.     public void save(M m) {  
  6.         repository.save(m);  
  7.     }  
  8. }  
  9.   
  10. @Service  
  11. public class UserService extends BaseService<User> {  
  12. }  
  13.   
  14. @Service  
  15. public class OrganizationService extends BaseService<Organization> {  
  16. }  

 

 大家可以看到,现在的写法非常简洁。支持泛型式依赖注入。

 

这个也是我之前非常想要的一个功能,这样对于那些基本的CRUD式代码,可以简化更多的代码。

 

 

如果大家用过Spring data jpa的话,以后注入的话也可以使用泛型限定式依赖注入 :

Java代码   收藏代码
  1. @Autowired  
  2. private Repository<User> userRepository;  

 

 对于泛型依赖注入,最好使用setter注入,这样万一子类想变,比较容易切换。比如https://github.com/zhangkaitao/es,如果有多个实现时,子类可以使用@Qualifier指定使用哪一个。

 

评论

33 楼 lancijk 2014-08-20   引用
大牛,得顶一顶.. 膜拜学习中...
32 楼 jinnianshilongnian 2014-08-18   引用
awake0474 写道
我用的spring3.2.1 。 好像也支持泛型限定式依赖注入

你是怎么用的?
31 楼 awake0474 2014-08-18   引用
我用的spring3.2.1 。 好像也支持泛型限定式依赖注入
30 楼 userapplewings 2014-03-12   引用
jinnianshilongnian 写道
userapplewings 写道
学习了,我想请问这种反省依赖注入能够写成spring的xml形式呢?在xml中如何实现呢?是将BaseRepository,注入到BaseService中的BaseRepository么?

可以在xml中配置的;但是必须使用注解注入依赖


果然是必须用注解注入,我想问的就是注入部分是不是还能写成xml,昨天想了半天,感觉xml貌似做不了了
29 楼 jinnianshilongnian 2014-03-12   引用
userapplewings 写道
学习了,我想请问这种反省依赖注入能够写成spring的xml形式呢?在xml中如何实现呢?是将BaseRepository,注入到BaseService中的BaseRepository么?

可以在xml中配置的;但是必须使用注解注入依赖
28 楼 userapplewings 2014-03-12   引用
学习了,我想请问这种反省依赖注入能够写成spring的xml形式呢?在xml中如何实现呢?是将BaseRepository,注入到BaseService中的BaseRepository么?
27 楼 jinnianshilongnian 2014-01-02   引用
hantsy 写道
我所期待的是 Spring 能够 produces 和 Inject Generic type bean。。。

这个已经在 Seam 2 及 CDI 中提供了很长时间,Spring4 依然不支持。 Spring 落伍太多了。

现在这种 Generic Type Injection 只是代替 Qualifer 的一种表示方式。

http://my.oschina.net/hantsy/blog/186939


看了下javaee7的CDI规范,确实很强大;
26 楼 jinnianshilongnian 2013-12-31   引用
jinnianshilongnian 写道
jinnianshilongnian 写道
hantsy 写道
我所期待的是 Spring 能够 produces 和 Inject Generic type bean。。。

这个已经在 Seam 2 及 CDI 中提供了很长时间,Spring4 依然不支持。 Spring 落伍太多了。

现在这种 Generic Type Injection 只是代替 Qualifer 的一种表示方式。

http://my.oschina.net/hantsy/blog/186939



因为spring对集合等作了特殊处理 可以参考 http://jinnianshilongnian.iteye.com/blog/1989379

另外我简单的改造了下spring即可支持你说的功能,可以提交下spring看看接受不
https://github.com/zhangkaitao/spring4-showcase/tree/master/spring4-others/src/test/java/com/sishuok/spring4/genericinject

改造的bean工厂如下
https://github.com/zhangkaitao/spring4-showcase/blob/master/spring4-others/src/test/java/org/springframework/beans/factory/support/DefaultListableBeanFactory.java

目前对xml List定义 也支持的

注:XML方式的话,注入时必须必须指定名字(因为xml中配置的集合没有泛型信息,除非只有一个集合 Bean)   除非bean definition使用ResolvableType 而不是class存储;
25 楼 jinnianshilongnian 2013-12-31   引用
jinnianshilongnian 写道
hantsy 写道
我所期待的是 Spring 能够 produces 和 Inject Generic type bean。。。

这个已经在 Seam 2 及 CDI 中提供了很长时间,Spring4 依然不支持。 Spring 落伍太多了。

现在这种 Generic Type Injection 只是代替 Qualifer 的一种表示方式。

http://my.oschina.net/hantsy/blog/186939



因为spring对集合等作了特殊处理 可以参考 http://jinnianshilongnian.iteye.com/blog/1989379

另外我简单的改造了下spring即可支持你说的功能,可以提交下spring看看接受不
https://github.com/zhangkaitao/spring4-showcase/tree/master/spring4-others/src/test/java/com/sishuok/spring4/genericinject

改造的bean工厂如下
https://github.com/zhangkaitao/spring4-showcase/blob/master/spring4-others/src/test/java/org/springframework/beans/factory/support/DefaultListableBeanFactory.java

目前对xml List定义 也支持的
24 楼 jinnianshilongnian 2013-12-31   引用
hantsy 写道
我所期待的是 Spring 能够 produces 和 Inject Generic type bean。。。

这个已经在 Seam 2 及 CDI 中提供了很长时间,Spring4 依然不支持。 Spring 落伍太多了。

现在这种 Generic Type Injection 只是代替 Qualifer 的一种表示方式。

http://my.oschina.net/hantsy/blog/186939



因为spring对集合等作了特殊处理 可以参考 http://jinnianshilongnian.iteye.com/blog/1989379

另外我简单的改造了下spring即可支持你说的功能,可以提交下spring看看接受不
https://github.com/zhangkaitao/spring4-showcase/tree/master/spring4-others/src/test/java/com/sishuok/spring4/genericinject
23 楼 hantsy 2013-12-30   引用
我所期待的是 Spring 能够 produces 和 Inject Generic type bean。。。

这个已经在 Seam 2 及 CDI 中提供了很长时间,Spring4 依然不支持。 Spring 落伍太多了。

现在这种 Generic Type Injection 只是代替 Qualifer 的一种表示方式。

http://my.oschina.net/hantsy/blog/186939

22 楼 jinnianshilongnian 2013-12-19   引用
manong_java 写道
我貌似找到原因了有两个同类型的Dao:UserDao和UserDao1
且所有Service 继承BaseService<T> ,去掉BaseService的继承则可以正常运行。
这样一来就说明BaseService中的BaseDao<T>不会根据具体的service类型来注入。

Java代码   收藏代码
  1. public abstract class BaseService<T extends Serializable> {  
  2.   
  3.     @Autowired  
  4.     private BaseDao<T> baseDao;  
  5.       
  6.     public void save(T t)  
  7.     {  
  8.         baseDao.save(t);  
  9.     }  
  10. }  
  11.   
  12. @Service  
  13. public class UserService extends BaseService<User> {  
  14.   
  15.     @Autowired  
  16.     @Qualifier("UserDao1")  
  17.     private BaseDao<User> dao;  
  18.       
  19.     @Override  
  20.     public void save(User t) {  
  21.         System.out.println("重写save user方法:"+t.getName());  
  22.         dao.save(t);  
  23.     }  
  24. }  


Java代码   收藏代码
  1. public static void main(String[] args) {  
  2.     ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");  
  3.     UserService userservice=ctx.getBean(UserService.class);  
  4.     User user=new User();  
  5.     user.setName("张三");  
  6.     user.setId(1000);  
  7.     userservice.save(user);  
  8. }  


jinnianshilongnian 写道
manong_java 写道
jinnianshilongnian 写道
manong_java 写道
他这个泛型支持 能根据user对象 自动注入UserRepository 对象吗?????
可以的啊 比如 Repository<User> 即可注入


我定义了两个Dao:UserDao、UserDao1
我在service实现类中指定使用UserDao1 但没起作用,程序启动抛异常 这是什么原因呢?

Java代码   收藏代码
  1. Caused by: org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type [com.hulibo.spring.dao.impl.BaseDao] is defined: expected single matching bean but found 2: UserDao,UserDao1  


Java代码   收藏代码
  1. @Service  
  2. public class UserService extends BaseService<User> {  
  3.   
  4.     @Autowired  
  5.     @Qualifier("UserDao1")  
  6.     private BaseDao<User> dao;  
  7.       
  8.     @Override  
  9.     public void save(User t) {  
  10.         System.out.println("重写save user方法:"+t.getName());  
  11.         dao.save(t);  
  12.     }  
  13.   
  14. }  

Java代码   收藏代码
  1. @Repository("UserDao")  
  2. public class UserDao extends BaseDao<User> {  
  3.   
  4.     @Override  
  5.     public void save(User t) {  
  6.         System.out.println("userDao save :"+t.getName());  
  7.     }  
  8.   
  9.       
  10. }  


Java代码   收藏代码
  1. @Repository("UserDao1")  
  2. public class UserDao1 extends BaseDao<User> {  
  3.   
  4.     @Override  
  5.     public void save(User t) {  
  6.         System.out.println("userDao1 save>>>>:"+t.getName());  
  7.     }  
  8. }  


很明显   BaseDao<User>  有两个实现,不过你指定了@Qualifier("UserDao1") 不应该有问题的啊



public abstract class BaseService<T extends Serializable> { 
 
    @Autowired 
    private BaseDao<T> baseDao; 

在那篇文章中 对于这种情况推荐使用setter注入
21 楼 manong_java 2013-12-19   引用
我貌似找到原因了有两个同类型的Dao:UserDao和UserDao1
且所有Service 继承BaseService<T> ,去掉BaseService的继承则可以正常运行。
这样一来就说明BaseService中的BaseDao<T>不会根据具体的service类型来注入。

Java代码   收藏代码
  1. public abstract class BaseService<T extends Serializable> {  
  2.   
  3.     @Autowired  
  4.     private BaseDao<T> baseDao;  
  5.       
  6.     public void save(T t)  
  7.     {  
  8.         baseDao.save(t);  
  9.     }  
  10. }  
  11.   
  12. @Service  
  13. public class UserService extends BaseService<User> {  
  14.   
  15.     @Autowired  
  16.     @Qualifier("UserDao1")  
  17.     private BaseDao<User> dao;  
  18.       
  19.     @Override  
  20.     public void save(User t) {  
  21.         System.out.println("重写save user方法:"+t.getName());  
  22.         dao.save(t);  
  23.     }  
  24. }  


Java代码   收藏代码
  1. public static void main(String[] args) {  
  2.     ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");  
  3.     UserService userservice=ctx.getBean(UserService.class);  
  4.     User user=new User();  
  5.     user.setName("张三");  
  6.     user.setId(1000);  
  7.     userservice.save(user);  
  8. }  


jinnianshilongnian 写道
manong_java 写道
jinnianshilongnian 写道
manong_java 写道
他这个泛型支持 能根据user对象 自动注入UserRepository 对象吗?????
可以的啊 比如 Repository<User> 即可注入


我定义了两个Dao:UserDao、UserDao1
我在service实现类中指定使用UserDao1 但没起作用,程序启动抛异常 这是什么原因呢?

Java代码   收藏代码
  1. Caused by: org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type [com.hulibo.spring.dao.impl.BaseDao] is defined: expected single matching bean but found 2: UserDao,UserDao1  


Java代码   收藏代码
  1. @Service  
  2. public class UserService extends BaseService<User> {  
  3.   
  4.     @Autowired  
  5.     @Qualifier("UserDao1")  
  6.     private BaseDao<User> dao;  
  7.       
  8.     @Override  
  9.     public void save(User t) {  
  10.         System.out.println("重写save user方法:"+t.getName());  
  11.         dao.save(t);  
  12.     }  
  13.   
  14. }  

Java代码   收藏代码
  1. @Repository("UserDao")  
  2. public class UserDao extends BaseDao<User> {  
  3.   
  4.     @Override  
  5.     public void save(User t) {  
  6.         System.out.println("userDao save :"+t.getName());  
  7.     }  
  8.   
  9.       
  10. }  


Java代码   收藏代码
  1. @Repository("UserDao1")  
  2. public class UserDao1 extends BaseDao<User> {  
  3.   
  4.     @Override  
  5.     public void save(User t) {  
  6.         System.out.println("userDao1 save>>>>:"+t.getName());  
  7.     }  
  8. }  


很明显   BaseDao<User>  有两个实现,不过你指定了@Qualifier("UserDao1") 不应该有问题的啊
20 楼 jinnianshilongnian 2013-12-19   引用
manong_java 写道
是的我故意写两个Repository<user>来测试根据类型自动注入,但事实就是出现这个异常
tao哥帮忙解决下呗
Java代码   收藏代码
  1. public static void main(String[] args) {  
  2.     ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");  
  3.     UserService userservice=ctx.getBean(UserService.class);  
  4.     User user=new User();  
  5.     user.setName("张三");  
  6.     user.setId(1000);  
  7.     userservice.save(user);  
  8. }  


jinnianshilongnian 写道
manong_java 写道
jinnianshilongnian 写道
manong_java 写道
他这个泛型支持 能根据user对象 自动注入UserRepository 对象吗?????

可以的啊 比如 Repository<User> 即可注入


我定义了两个Dao:UserDao、UserDao1
我在service实现类中指定使用UserDao1 但没起作用,程序启动抛异常 这是什么原因呢?

Java代码   收藏代码
  1. Caused by: org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type [com.hulibo.spring.dao.impl.BaseDao] is defined: expected single matching bean but found 2: UserDao,UserDao1  


Java代码   收藏代码
  1. @Service  
  2. public class UserService extends BaseService<User> {  
  3.   
  4.     @Autowired  
  5.     @Qualifier("UserDao1")  
  6.     private BaseDao<User> dao;  
  7.       
  8.     @Override  
  9.     public void save(User t) {  
  10.         System.out.println("重写save user方法:"+t.getName());  
  11.         dao.save(t);  
  12.     }  
  13.   
  14. }  

Java代码   收藏代码
  1. @Repository("UserDao")  
  2. public class UserDao extends BaseDao<User> {  
  3.   
  4.     @Override  
  5.     public void save(User t) {  
  6.         System.out.println("userDao save :"+t.getName());  
  7.     }  
  8.   
  9.       
  10. }  


Java代码   收藏代码
  1. @Repository("UserDao1")  
  2. public class UserDao1 extends BaseDao<User> {  
  3.   
  4.     @Override  
  5.     public void save(User t) {  
  6.         System.out.println("userDao1 save>>>>:"+t.getName());  
  7.     }  
  8. }  


很明显   BaseDao<User>  有两个实现,不过你指定了@Qualifier("UserDao1") 不应该有问题的啊

你把代码站内信我 我看看
19 楼 manong_java 2013-12-19   引用
是的我故意写两个Repository<user>来测试根据类型自动注入,但事实就是出现这个异常
tao哥帮忙解决下呗
Java代码   收藏代码
  1. public static void main(String[] args) {  
  2.     ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");  
  3.     UserService userservice=ctx.getBean(UserService.class);  
  4.     User user=new User();  
  5.     user.setName("张三");  
  6.     user.setId(1000);  
  7.     userservice.save(user);  
  8. }  


jinnianshilongnian 写道
manong_java 写道
jinnianshilongnian 写道
manong_java 写道
他这个泛型支持 能根据user对象 自动注入UserRepository 对象吗?????

可以的啊 比如 Repository<User> 即可注入


我定义了两个Dao:UserDao、UserDao1
我在service实现类中指定使用UserDao1 但没起作用,程序启动抛异常 这是什么原因呢?

Java代码   收藏代码
  1. Caused by: org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type [com.hulibo.spring.dao.impl.BaseDao] is defined: expected single matching bean but found 2: UserDao,UserDao1  


Java代码   收藏代码
  1. @Service  
  2. public class UserService extends BaseService<User> {  
  3.   
  4.     @Autowired  
  5.     @Qualifier("UserDao1")  
  6.     private BaseDao<User> dao;  
  7.       
  8.     @Override  
  9.     public void save(User t) {  
  10.         System.out.println("重写save user方法:"+t.getName());  
  11.         dao.save(t);  
  12.     }  
  13.   
  14. }  

Java代码   收藏代码
  1. @Repository("UserDao")  
  2. public class UserDao extends BaseDao<User> {  
  3.   
  4.     @Override  
  5.     public void save(User t) {  
  6.         System.out.println("userDao save :"+t.getName());  
  7.     }  
  8.   
  9.       
  10. }  


Java代码   收藏代码
  1. @Repository("UserDao1")  
  2. public class UserDao1 extends BaseDao<User> {  
  3.   
  4.     @Override  
  5.     public void save(User t) {  
  6.         System.out.println("userDao1 save>>>>:"+t.getName());  
  7.     }  
  8. }  


很明显   BaseDao<User>  有两个实现,不过你指定了@Qualifier("UserDao1") 不应该有问题的啊
18 楼 jinnianshilongnian 2013-12-19   引用
manong_java 写道
jinnianshilongnian 写道
manong_java 写道
他这个泛型支持 能根据user对象 自动注入UserRepository 对象吗?????

可以的啊 比如 Repository<User> 即可注入


我定义了两个Dao:UserDao、UserDao1
我在service实现类中指定使用UserDao1 但没起作用,程序启动抛异常 这是什么原因呢?

Java代码   收藏代码
  1. Caused by: org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type [com.hulibo.spring.dao.impl.BaseDao] is defined: expected single matching bean but found 2: UserDao,UserDao1  


Java代码   收藏代码
  1. @Service  
  2. public class UserService extends BaseService<User> {  
  3.   
  4.     @Autowired  
  5.     @Qualifier("UserDao1")  
  6.     private BaseDao<User> dao;  
  7.       
  8.     @Override  
  9.     public void save(User t) {  
  10.         System.out.println("重写save user方法:"+t.getName());  
  11.         dao.save(t);  
  12.     }  
  13.   
  14. }  

Java代码   收藏代码
  1. @Repository("UserDao")  
  2. public class UserDao extends BaseDao<User> {  
  3.   
  4.     @Override  
  5.     public void save(User t) {  
  6.         System.out.println("userDao save :"+t.getName());  
  7.     }  
  8.   
  9.       
  10. }  


Java代码   收藏代码
  1. @Repository("UserDao1")  
  2. public class UserDao1 extends BaseDao<User> {  
  3.   
  4.     @Override  
  5.     public void save(User t) {  
  6.         System.out.println("userDao1 save>>>>:"+t.getName());  
  7.     }  
  8. }  


很明显   BaseDao<User>  有两个实现,不过你指定了@Qualifier("UserDao1") 不应该有问题的啊
17 楼 manong_java 2013-12-19   引用
jinnianshilongnian 写道
manong_java 写道
他这个泛型支持 能根据user对象 自动注入UserRepository 对象吗?????

可以的啊 比如 Repository<User> 即可注入


我定义了两个Dao:UserDao、UserDao1
我在service实现类中指定使用UserDao1 但没起作用,程序启动抛异常 这是什么原因呢?

Java代码   收藏代码
  1. Caused by: org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type [com.hulibo.spring.dao.impl.BaseDao] is defined: expected single matching bean but found 2: UserDao,UserDao1  


Java代码   收藏代码
  1. @Service  
  2. public class UserService extends BaseService<User> {  
  3.   
  4.     @Autowired  
  5.     @Qualifier("UserDao1")  
  6.     private BaseDao<User> dao;  
  7.       
  8.     @Override  
  9.     public void save(User t) {  
  10.         System.out.println("重写save user方法:"+t.getName());  
  11.         dao.save(t);  
  12.     }  
  13.   
  14. }  

Java代码   收藏代码
  1. @Repository("UserDao")  
  2. public class UserDao extends BaseDao<User> {  
  3.   
  4.     @Override  
  5.     public void save(User t) {  
  6.         System.out.println("userDao save :"+t.getName());  
  7.     }  
  8.   
  9.       
  10. }  


Java代码   收藏代码
  1. @Repository("UserDao1")  
  2. public class UserDao1 extends BaseDao<User> {  
  3.   
  4.     @Override  
  5.     public void save(User t) {  
  6.         System.out.println("userDao1 save>>>>:"+t.getName());  
  7.     }  
  8. }  
16 楼 jinnianshilongnian 2013-12-19   引用
manong_java 写道
他这个泛型支持 能根据user对象 自动注入UserRepository 对象吗?????

可以的啊 比如 Repository<User> 即可注入
15 楼 manong_java 2013-12-19   引用
他这个泛型支持 能根据user对象 自动注入UserRepository 对象吗?????
14 楼 keeley 2013-12-14   引用
厉害!......

你可能感兴趣的:(Spring4新特性——泛型限定式依赖注入)