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、实体
public class User implements Serializable { private Long id; private String name; } public class Organization implements Serializable { private Long id; private String name; }
1.2、Repository
public abstract class BaseRepository<M extends Serializable> { public void save(M m) { System.out.println("=====repository save:" + m); } } @Repository public class UserRepository extends BaseRepository<User> { } @Repository public class OrganizationRepository extends BaseRepository<Organization> { }
对于Repository,我们一般是这样实现的:首先写一个模板父类,把通用的crud等代码放在BaseRepository;然后子类继承后,只需要添加额外的实现。
1.3、Service
1.3.1、以前Service写法
public abstract class BaseService<M extends Serializable> { private BaseRepository<M> repository; public void setRepository(BaseRepository<M> repository) { this.repository = repository; } public void save(M m) { repository.save(m); } } @Service public class UserService extends BaseService<User> { @Autowired public void setUserRepository(UserRepository userRepository) { setRepository(userRepository); } } @Service public class OrganizationService extends BaseService<Organization> { @Autowired public void setOrganizationRepository(OrganizationRepository organizationRepository) { setRepository(organizationRepository); } }
可以看到,以前必须再写一个setter方法,然后指定注入的具体类型,然后进行注入;
1.3.2、泛型Service的写法
public abstract class BaseService<M extends Serializable> { @Autowired protected BaseRepository<M> repository; public void save(M m) { repository.save(m); } } @Service public class UserService extends BaseService<User> { } @Service public class OrganizationService extends BaseService<Organization> { }
大家可以看到,现在的写法非常简洁。支持泛型式依赖注入。
这个也是我之前非常想要的一个功能,这样对于那些基本的CRUD式代码,可以简化更多的代码。
如果大家用过Spring data jpa的话,以后注入的话也可以使用泛型限定式依赖注入 :
@Autowired private Repository<User> userRepository;
对于泛型依赖注入,最好使用setter注入,这样万一子类想变,比较容易切换。比如https://github.com/zhangkaitao/es,如果有多个实现时,子类可以使用@Qualifier指定使用哪一个。
评论
你是怎么用的?
可以在xml中配置的;但是必须使用注解注入依赖
果然是必须用注解注入,我想问的就是注入部分是不是还能写成xml,昨天想了半天,感觉xml貌似做不了了
可以在xml中配置的;但是必须使用注解注入依赖
这个已经在 Seam 2 及 CDI 中提供了很长时间,Spring4 依然不支持。 Spring 落伍太多了。
现在这种 Generic Type Injection 只是代替 Qualifer 的一种表示方式。
http://my.oschina.net/hantsy/blog/186939
看了下javaee7的CDI规范,确实很强大;
这个已经在 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存储;
这个已经在 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定义 也支持的
这个已经在 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
这个已经在 Seam 2 及 CDI 中提供了很长时间,Spring4 依然不支持。 Spring 落伍太多了。
现在这种 Generic Type Injection 只是代替 Qualifer 的一种表示方式。
http://my.oschina.net/hantsy/blog/186939
且所有Service 继承BaseService<T> ,去掉BaseService的继承则可以正常运行。
这样一来就说明BaseService中的BaseDao<T>不会根据具体的service类型来注入。
public abstract class BaseService<T extends Serializable> { @Autowired private BaseDao<T> baseDao; public void save(T t) { baseDao.save(t); } } @Service public class UserService extends BaseService<User> { @Autowired @Qualifier("UserDao1") private BaseDao<User> dao; @Override public void save(User t) { System.out.println("重写save user方法:"+t.getName()); dao.save(t); } }
public static void main(String[] args) { ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml"); UserService userservice=ctx.getBean(UserService.class); User user=new User(); user.setName("张三"); user.setId(1000); userservice.save(user); }
可以的啊 比如 Repository<User> 即可注入
我定义了两个Dao:UserDao、UserDao1
我在service实现类中指定使用UserDao1 但没起作用,程序启动抛异常 这是什么原因呢?
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
@Service public class UserService extends BaseService<User> { @Autowired @Qualifier("UserDao1") private BaseDao<User> dao; @Override public void save(User t) { System.out.println("重写save user方法:"+t.getName()); dao.save(t); } }
@Repository("UserDao") public class UserDao extends BaseDao<User> { @Override public void save(User t) { System.out.println("userDao save :"+t.getName()); } }
@Repository("UserDao1") public class UserDao1 extends BaseDao<User> { @Override public void save(User t) { System.out.println("userDao1 save>>>>:"+t.getName()); } }
很明显 BaseDao<User> 有两个实现,不过你指定了@Qualifier("UserDao1") 不应该有问题的啊
public abstract class BaseService<T extends Serializable> {
@Autowired
private BaseDao<T> baseDao;
在那篇文章中 对于这种情况推荐使用setter注入
且所有Service 继承BaseService<T> ,去掉BaseService的继承则可以正常运行。
这样一来就说明BaseService中的BaseDao<T>不会根据具体的service类型来注入。
public abstract class BaseService<T extends Serializable> { @Autowired private BaseDao<T> baseDao; public void save(T t) { baseDao.save(t); } } @Service public class UserService extends BaseService<User> { @Autowired @Qualifier("UserDao1") private BaseDao<User> dao; @Override public void save(User t) { System.out.println("重写save user方法:"+t.getName()); dao.save(t); } }
public static void main(String[] args) { ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml"); UserService userservice=ctx.getBean(UserService.class); User user=new User(); user.setName("张三"); user.setId(1000); userservice.save(user); }
可以的啊 比如 Repository<User> 即可注入
我定义了两个Dao:UserDao、UserDao1
我在service实现类中指定使用UserDao1 但没起作用,程序启动抛异常 这是什么原因呢?
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
@Service public class UserService extends BaseService<User> { @Autowired @Qualifier("UserDao1") private BaseDao<User> dao; @Override public void save(User t) { System.out.println("重写save user方法:"+t.getName()); dao.save(t); } }
@Repository("UserDao") public class UserDao extends BaseDao<User> { @Override public void save(User t) { System.out.println("userDao save :"+t.getName()); } }
@Repository("UserDao1") public class UserDao1 extends BaseDao<User> { @Override public void save(User t) { System.out.println("userDao1 save>>>>:"+t.getName()); } }
很明显 BaseDao<User> 有两个实现,不过你指定了@Qualifier("UserDao1") 不应该有问题的啊
tao哥帮忙解决下呗
public static void main(String[] args) { ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml"); UserService userservice=ctx.getBean(UserService.class); User user=new User(); user.setName("张三"); user.setId(1000); userservice.save(user); }
可以的啊 比如 Repository<User> 即可注入
我定义了两个Dao:UserDao、UserDao1
我在service实现类中指定使用UserDao1 但没起作用,程序启动抛异常 这是什么原因呢?
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
@Service public class UserService extends BaseService<User> { @Autowired @Qualifier("UserDao1") private BaseDao<User> dao; @Override public void save(User t) { System.out.println("重写save user方法:"+t.getName()); dao.save(t); } }
@Repository("UserDao") public class UserDao extends BaseDao<User> { @Override public void save(User t) { System.out.println("userDao save :"+t.getName()); } }
@Repository("UserDao1") public class UserDao1 extends BaseDao<User> { @Override public void save(User t) { System.out.println("userDao1 save>>>>:"+t.getName()); } }
很明显 BaseDao<User> 有两个实现,不过你指定了@Qualifier("UserDao1") 不应该有问题的啊
你把代码站内信我 我看看
tao哥帮忙解决下呗
public static void main(String[] args) { ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml"); UserService userservice=ctx.getBean(UserService.class); User user=new User(); user.setName("张三"); user.setId(1000); userservice.save(user); }
可以的啊 比如 Repository<User> 即可注入
我定义了两个Dao:UserDao、UserDao1
我在service实现类中指定使用UserDao1 但没起作用,程序启动抛异常 这是什么原因呢?
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
@Service public class UserService extends BaseService<User> { @Autowired @Qualifier("UserDao1") private BaseDao<User> dao; @Override public void save(User t) { System.out.println("重写save user方法:"+t.getName()); dao.save(t); } }
@Repository("UserDao") public class UserDao extends BaseDao<User> { @Override public void save(User t) { System.out.println("userDao save :"+t.getName()); } }
@Repository("UserDao1") public class UserDao1 extends BaseDao<User> { @Override public void save(User t) { System.out.println("userDao1 save>>>>:"+t.getName()); } }
很明显 BaseDao<User> 有两个实现,不过你指定了@Qualifier("UserDao1") 不应该有问题的啊
可以的啊 比如 Repository<User> 即可注入
我定义了两个Dao:UserDao、UserDao1
我在service实现类中指定使用UserDao1 但没起作用,程序启动抛异常 这是什么原因呢?
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
@Service public class UserService extends BaseService<User> { @Autowired @Qualifier("UserDao1") private BaseDao<User> dao; @Override public void save(User t) { System.out.println("重写save user方法:"+t.getName()); dao.save(t); } }
@Repository("UserDao") public class UserDao extends BaseDao<User> { @Override public void save(User t) { System.out.println("userDao save :"+t.getName()); } }
@Repository("UserDao1") public class UserDao1 extends BaseDao<User> { @Override public void save(User t) { System.out.println("userDao1 save>>>>:"+t.getName()); } }
很明显 BaseDao<User> 有两个实现,不过你指定了@Qualifier("UserDao1") 不应该有问题的啊
可以的啊 比如 Repository<User> 即可注入
我定义了两个Dao:UserDao、UserDao1
我在service实现类中指定使用UserDao1 但没起作用,程序启动抛异常 这是什么原因呢?
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
@Service public class UserService extends BaseService<User> { @Autowired @Qualifier("UserDao1") private BaseDao<User> dao; @Override public void save(User t) { System.out.println("重写save user方法:"+t.getName()); dao.save(t); } }
@Repository("UserDao") public class UserDao extends BaseDao<User> { @Override public void save(User t) { System.out.println("userDao save :"+t.getName()); } }
@Repository("UserDao1") public class UserDao1 extends BaseDao<User> { @Override public void save(User t) { System.out.println("userDao1 save>>>>:"+t.getName()); } }
可以的啊 比如 Repository<User> 即可注入