Spring核心——JSR250与资源控制

JSR-175与元编程

要说明JSR-250先要解释清楚JSR-175,要解释清楚JSR就的先了解JCP是什么。网上资料不少,就不细说了,简单的说JCP(Java Community Process)是管理Java生态(包括J2SE、J2EE等等)发展的合做组织。JSR(Java Specification Request)就是组织内的成员针对Java的发展提出的一些需求,经过审核以后即会融入到新版本的Java功能中成为Java的一项特性或功能,不一样的发行版本和虚拟机都会遵照这些约定。java

JSR-175的全文标题是 A Metadata Facility for the Java Programming Language (为Java语言提供元数据设施)。它明确提出了在Java平台引入“元编程”(Meta Programming)的思想,要求提供对“元数据”(Meta Data)的支持。这就是咱们如今大量使用的“@”注解(Annotation)功能的最先来源。JSR-175以后的JSR-181(Web服务支持)、JSR-250、JSR-330都是基于“元数据”功能提出的一些更细节的实现。spring

至于“元编程”、“元数据”是什么这里就不详细展开说明了,它的理论很早就提出了,听说最先是在Lisp这一类函数式编程语言上开始使用的。网上有不少相关的资料,简单的说它就是“对源码进行编码”,好比下面这样:编程

class MyClass {
	@Autowired
	private Interface support;
}

经过@Autowired这个注解来对support这个域进行编码就能够很轻松的扩展原先类的功能。框架

JSR-250的Spring实现

JSR-250主要是围绕着“资源”的使用预约义了一些注解(Annotation),这里的“资源”能够理解为一个Class类的实例、一个JavaBean、或者一个Spring中的Bean。编程语言

JSR-250相关的注解所有在 javax.annotationjavax.annotation.security 包中,分红2个部分——资源定义和权限控制。它并无提供具体的实现方式,仅仅是提供了指导性的文档和几个注解,由具体的框架去实现。函数式编程

javax.annotation 中包含一下几个注解:函数

  • @Generated:生成资源的注解,经过该项标记产生的实例是一个资源。相似于Spring中的@Bean注解,用于生成一贯资源。
  • @PostConstruct 创造资源以后的回调处理,Spring已经实现了这个注解,见Bean的定义与控制 一文的介绍。
  • @PreDestroy 销毁资源以前的回调处理,Spring一样实现了这个注解,见Bean的定义与控制
  • @Resource 标记使用资源的位置,Spring一样实现了这个注解的功能(后文会详细介绍)。功能上有些相似于@Autowired、@Inject,可是二者有很多的差异。
  • @Resources 标记使用多项资源的位置,相似于使用@Autowired向一个列表装载数据。

仔细看JSR-250定义的这些注解就会发现,他们都是关于“资源”的构建、销毁、使用的。Spring实现了@PostConstruct、@PreDestroy和@Resource。工具

javax.annotation.security 包中有如下内容:网站

  • @DeclareRoles 声明角色
  • @DenyAll  拒绝全部角色
  • @PermitAll  受权全部惧色
  • @RolesAllowed  角色受权
  • @RunAs 运行模式

security中的内容是在资源建立以后对资源的使用进行管理。和常规的权限控制模型同样——定义角色(@DeclareRoles )、肯定角色对资源的控制权限(@DenyAll、@PermitAll 、@RolesAllowed )。Spring并无实现这里的任何一个注解,在这里就不深刻介绍了。这一块内容在J2EE的构建中有很多的应用。ui

Spring中的@Resource

在没有仔细看Spring的官方文档和JSR-250以前,我一直觉得@Resource这个注解和@Autowired是2个不一样的功能,更早的时候还觉得是管理什么Properties资源的,不少网上的内容也写得比较模糊。虽然@Resource的实现是在 CommonAnnotationBeanPostProcessor 而@Autowired 是在 AutowiredAnnotationBeanPostProcessor,可是实际上二者的功能是重叠的,或者说@Resource的提供的功能是@Autowired的子集。

在Spring中使用@Resource注解时,把Bean理解为一项资源就很好理解了。下面经过一些简单的例子来介绍@Resource的使用。

@Resource的功能是告诉IoC容器标记的位置须要什么样的“资源”,以下:

class Abc {}
class Xyz {}
class Implement {
	@Resource
	private Abc abc;
	
	private Xyz xyz;

    @Resource
    private ApplicationContext context;

	@Resource(name="b_instance")
	public void setInject(Xyz xyz) {
		this.xyz = xyz;
	}
}
<beans>
    <context:annotation-config/>
    <bean id="abc" class="x.y.Abc" />
    <bean id="xyz_instance" name="inject" class="x.y.Xyz" />
    <bean class="x.y.Implement" />
</beans>

运行后,IoC会向标记了@Resource的位置注入Bean——是否是感受和@Autowired如出一辙?可是须要注意的是虽然二者最后都是注入一个Bean,可是@Resource和@Autowired的处理过程是不同的。@Autowired若是没有提供任何参数,那么他优先按照类型注入,若是要对细节进行控制能够配合Primary和Qualifiers功能,详见注解自动装载的介绍。@Resource是按照命名来注入资源的,以上面的代码为例子:

  1. 例如在setter方法上定义了name="xyz_instance"参数,那么会去IoC容器中寻找id、name等于"xyz_instance"的Bean来注入。
  2. 例如在abc这个域(成员变量)上没有定义name参数,那么会使用域的名称(这里是"abc")去IoC中按id、name寻找Bean来注入。
  3. 若是@Resource定义在方法上,而且没有指定name参数,那么他会使用setter的名称(例子中方法名为setInject,名称就是"inject")来寻找并注入数据。
  4. 最后,若是名称匹配不上,容器会根据标记位置的类型来注入数据,例如例如中的ApplicationContext。

因此@Resource的装载资源过程是:1)匹配name参数;2)没有name参数时会根据setter或域的名称来匹配Bean的名称;3)仍是匹配不上就根据标记位置的类型来注入数据。

与@Autowired相比主要有如下几点区别:

  1. 控制粒度没有@Autowired细,某些参数Spring并无实现功能。可是使用他更符合整个Java生态的规范。
  2. 若是是使用类型依赖注入数据,应优先使用@Autowired,效率会好一些。
  3. @Resource经过名称注入与@Autowired相比省去了@Qualifiers等内容。
  4. @Resource只能用在域和Setter方法上。

总的来讲若是是按照类型注入依赖对象,那么最终获得的结果并无任何差别,只是执行过程上有差异。若是按Bean的名称使用,@Resource比@Autowired便捷一些,可是功能少不少。

我的建议若是开发的是一个面向终端用户的应用,好比Web应用、网站什么的,直接用@Autowired就行了。若是制做的是一个给别的开发人员使用的工具,能够考虑@Resourec,他能获得更多框架的支持。

@PostConstruct 与@PreDestroy

@PostConstruct 与@PreDestroy也是JSR-250中定义的注解,Spring都实现了他们的功能,使用方法能够查看 Bean的定义与控制 相关的说明和介绍。

相关文章
相关标签/搜索