您的位置:首页 > 其它

(6) 用@Resource注解完成属性装配 【依赖注入----手动装配】 以及 编码解析其原理【经典】

2012-06-11 19:57 369 查看
手工装配依赖对象,在这种方式中又有两种编程方式
1. 在xml配置文件中,通过在bean节点下配置,如

<bean id="orderService" class="cn.itcast.service.OrderServiceBean">

<constructor-arg index=“0” type=“java.lang.String” value=“xxx”/>//构造器注入

<property name=“name” value=“zhao/>//属性setter方法注入

</bean>

2. 在java代码中使用@Autowired或@Resource注解方式进行装配。但我们需要在xml配置文件中配置以下信息:

<beans xmlns="http://www.springframework.org/schema/beans"

xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xmlns:context="http://www.springframework.org/schema/context"

xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-2.5.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd">
<context:annotation-config/>

< /beans>

这个配置隐式注册了多个对注释进行解析处理的处理器:AutowiredAnnotationBeanPostProcessor,CommonAnnotationBeanPostProcessor,PersistenceAnnotationBeanPostProcessor,RequiredAnnotationBeanPostProcessor

注: @Resource注解在spring安装目录的lib\j2ee\common-annotations.jar

在java代码中使用@Autowired或@Resource注解方式进行装配,这两个注解的区别是:@Autowired 默认按类型装配,@Resource默认按名称装配,当找不到与名称匹配的bean才会按类型装配。

(1)

@Autowired

private PersonDao personDao;//用于字段上

@Autowired

public void setOrderDao(OrderDao orderDao) {//用于属性的setter方法上

this.orderDao = orderDao;

}

@Autowired注解是按类型装配依赖对象,默认情况下它要求依赖对象必须存在,如果允许null值,可以设置它required属性为false。如果我们想使用按名称装配,可以结合@Qualifier注解一起使用。如下:

@Autowired @Qualifier("personDaoBean")

private PersonDao personDao;

(2)

@Resource注解和@Autowired一样,也可以标注在字段或属性的setter方法上,但它默认按名称装配。

名称可以通过@Resource的name属性指定,如果没有指定name属性,当注解标注在字段上,即默认取字段的名称作为bean名称寻找依赖对象,当注解标注在属性的setter方法上,即默认取属性名作为bean名称寻找依赖对象。

@Resource(name=“personDaoBean”)

private PersonDao personDao;//用于字段上


注意:如果没有指定name属性,并且按照默认的名称仍然找不到依赖对象时, @Resource注解会回退到按类型装配。但一旦指定了name属性,就只能按名称装配了。

参照代码验证:实践出真知

[html]
view plaincopyprint?

<?xml
version="1.0"
encoding="UTF-8"?>

<beans
xmlns="http://www.springframework.org/schema/beans"

xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xmlns:context="http://www.springframework.org/schema/context"

xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-2.5.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd">

<context:annotation-config/>

<bean
id="personDao"
class="cn.itm.dao.impl.PersonDaoBean"></bean>

<bean
id="personService"
class="cn.itm.service.impl.PersonServiceBean">

<!--
<constructor-arg
index="0"
type="cn.itm.dao.PersonDao"/>

<constructor-arg
index="1"
value="大家好。"></constructor-arg>

-->

</bean>

</beans>

[java]
view plaincopyprint?

package cn.itm.service.impl;

import javax.annotation.Resource;

import cn.itm.dao.PersonDao;

import cn.itm.service.PersonService;

public class PersonServiceBean
implements PersonService{

// 首先获取,personDao的名称,然后 在SPring容器里面,寻找与此名称所匹配的bean,如果找到 就把 这个bean 注入到这个字段上来。

@Resource
private PersonDao personDao;
private String name;

public PersonServiceBean(){}

public PersonServiceBean(PersonDao personDao, String name) {

this.personDao = personDao;

this.name = name;
}

public void save(){

personDao.add();
}

}

package cn.itm.service.impl;

import javax.annotation.Resource;

import cn.itm.dao.PersonDao;
import cn.itm.service.PersonService;

public class PersonServiceBean implements PersonService{

// 首先获取,personDao的名称,然后 在SPring容器里面,寻找与此名称所匹配的bean,如果找到 就把 这个bean 注入到这个字段上来。
@Resource private PersonDao personDao;
private String name;

public PersonServiceBean(){}

public PersonServiceBean(PersonDao personDao, String name) {
this.personDao = personDao;
this.name = name;
}

public void save(){
personDao.add();
}

}


[java]
view plaincopyprint?

package cn.itm.dao.impl;

import cn.itm.dao.PersonDao;

public class PersonDaoBean
implements PersonDao {

public void add(){

System.out.println("执行PersonDaoBean的add方法。。。");

}
}

[java]
view plaincopyprint?

package cn.itm.dao;

public interface PersonDao {

public abstract
void add();

}

package cn.itm.dao;

public interface PersonDao {

public abstract void add();

}


[java]
view plaincopyprint?

package cn.itm.service;

public interface PersonService {

public void save();

}

[java]
view plaincopyprint?

package junit.test;

import org.junit.BeforeClass;

import org.junit.Test;

import org.springframework.context.ApplicationContext;

import org.springframework.context.support.ClassPathXmlApplicationContext;

import cn.itm.service.PersonService;

public class SpringTest {

@BeforeClass

public static
void setUpBeforeClass() throws Exception {

}

// 专门用来实例化  Spring 容器的。

@Test public
void instanceSpring(){

ApplicationContext ctx = new ClassPathXmlApplicationContext("beans.xml");

PersonService personService = (PersonService) ctx.getBean("personService");

personService.save();
}
}

package junit.test;

import org.junit.BeforeClass;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

import cn.itm.service.PersonService;

public class SpringTest {

@BeforeClass
public static void setUpBeforeClass() throws Exception {
}

// 专门用来实例化  Spring 容器的。
@Test public void instanceSpring(){

ApplicationContext ctx = new ClassPathXmlApplicationContext("beans.xml");
PersonService personService = (PersonService) ctx.getBean("personService");
personService.save();
}
}


可以正常运行,通过。

编码解析其原理:

1,主要实现的方法:

[java]
view plaincopyprint?

/**
* 实现注解的方式 注入:
*
*/
private void annotationInject() {

// 循环所有的 bean对象:
for(String beanName : sigletons.keySet()){

// 获取bean对象:
Object bean = sigletons.get(beanName);

// 判断 bean对象是否存在:

if(bean !=
null){
// 得到 bean的属性描述:

try {

PropertyDescriptor[] ps = Introspector.getBeanInfo(bean.getClass()).getPropertyDescriptors();

for(PropertyDescriptor propertyDesc
/*这里是 bean 里面的属性*/ : ps){
// 获取setter方法:

Method setter = propertyDesc.getWriteMethod();

if(setter !=
null && setter.isAnnotationPresent(ItmResource.class)){
// 还要 看看 setter方法是否存在 注解:
// 取得注解:

ItmResource resource = setter.getAnnotation(ItmResource.class);

Object value = null;

if(resource.name() !=
null && !"".equals(resource.name())){

// 判断是否在 集合里面:从集合里取出来 注入进去。

value = sigletons.get(resource.name());

}else{

// 如果 没有指定 name属性 怎么办 ?

value = sigletons.get(propertyDesc.getName());

if(value ==
null){
for(String key : sigletons.keySet()){
// 历遍所有的bean

// 是否存在 类型匹配的bean。
if(propertyDesc.getPropertyType().isAssignableFrom(sigletons.get(key).getClass())){

value = sigletons.get(key);

break;

}
}
}
}
setter.setAccessible(true);

setter.invoke(bean, value);
// 注入进去。
}
}
// 对字段进行处理:

Field[] fields = bean.getClass().getDeclaredFields();

for(Field field : fields ){

if(field.isAnnotationPresent(ItmResource.class)){

// 取得注解:

ItmResource resource = field.getAnnotation(ItmResource.class);

Object value = null;

if(resource.name() !=
null && !"".equals(resource.name())){

// 判断是否在 集合里面:从集合里取出来 注入进去。

value = sigletons.get(resource.name());

}else{

// 如果 没有指定 name属性 怎么办 ?

value = sigletons.get(field.getName());
// 有 name直接取得对象。
if(value ==
null){
for(String key : sigletons.keySet()){
// 历遍所有的bean

// 是否存在 类型匹配的bean。 【再根据 字段的类型去寻找】
if(field.getType().isAssignableFrom(sigletons.get(key).getClass())){

value = sigletons.get(key);

break;

}
}
}
}

field.setAccessible(true);

field.set(bean, value);

}
}
} catch (Exception e) {

e.printStackTrace();
}
}
}

}

[java]
view plaincopyprint?

package junit.test;

import java.beans.Introspector;

import java.beans.PropertyDescriptor;

import java.lang.reflect.Field;

import java.lang.reflect.Method;

import java.net.URL;
import java.util.ArrayList;

import java.util.HashMap;
import java.util.List;

import java.util.Map;

import org.apache.commons.beanutils.ConvertUtils;

import org.dom4j.Document;

import org.dom4j.Element;
import org.dom4j.XPath;

import org.dom4j.io.SAXReader;

public class ItmClassPathXMLApplicationContext {

private List<BeanDefinition> beanDefines =
new ArrayList<BeanDefinition>();

// 存放实例
private Map<String,Object> sigletons =
new HashMap<String,Object>();

public ItmClassPathXMLApplicationContext(String fileName){

this.readXML(fileName);

this.instanceBeans();

this.annotationInject();

this.injectObject();

}

/**
* 实现注解的方式  注入:
*
*/
private void annotationInject() {

// 循环所有的 bean对象:
for(String beanName  : sigletons.keySet()){

// 获取bean对象:
Object bean = sigletons.get(beanName);

// 判断 bean对象是否存在:

if(bean !=
null){
// 得到 bean的属性描述:

try {

PropertyDescriptor[] ps = Introspector.getBeanInfo(bean.getClass()).getPropertyDescriptors();

for(PropertyDescriptor propertyDesc
/*这里是 bean 里面的属性*/ : ps){
// 获取setter方法:

Method setter = propertyDesc.getWriteMethod();

if(setter !=
null  && setter.isAnnotationPresent(ItmResource.class)){
// 还要   看看   setter方法是否存在 注解:
// 取得注解:

ItmResource resource = setter.getAnnotation(ItmResource.class);

Object value  = null;

if(resource.name() !=
null && !"".equals(resource.name())){

// 判断是否在  集合里面:从集合里取出来  注入进去。

value = sigletons.get(resource.name());

}else{

// 如果 没有指定 name属性  怎么办 ?

value = sigletons.get(propertyDesc.getName());

if(value ==
null){
for(String key  : sigletons.keySet()){
// 历遍所有的bean

// 是否存在  类型匹配的bean。
if(propertyDesc.getPropertyType().isAssignableFrom(sigletons.get(key).getClass())){

value = sigletons.get(key);

break;

}
}
}
}
setter.setAccessible(true);

setter.invoke(bean, value);
// 注入进去。
}
}
// 对字段进行处理:

Field[] fields  = bean.getClass().getDeclaredFields();

for(Field field : fields ){

if(field.isAnnotationPresent(ItmResource.class)){

// 取得注解:

ItmResource resource = field.getAnnotation(ItmResource.class);

Object value  = null;

if(resource.name() !=
null && !"".equals(resource.name())){

// 判断是否在  集合里面:从集合里取出来  注入进去。

value = sigletons.get(resource.name());

}else{

// 如果 没有指定 name属性  怎么办 ?

value = sigletons.get(field.getName());
// 有 name直接取得对象。
if(value ==
null){
for(String key  : sigletons.keySet()){
// 历遍所有的bean

// 是否存在  类型匹配的bean。 【再根据 字段的类型去寻找】
if(field.getType().isAssignableFrom(sigletons.get(key).getClass())){

value = sigletons.get(key);

break;

}
}
}
}

field.setAccessible(true);

field.set(bean, value);

}
}
} catch (Exception e) {

e.printStackTrace();
}
}
}

}

private void injectObject() {

for(BeanDefinition beanDefinition : beanDefines){

// 得到  bean 。。
Object bean = sigletons.get(beanDefinition.getId());

if(bean != null){

try {

// 得到 bean的属性描述:

PropertyDescriptor[] ps = Introspector.getBeanInfo(bean.getClass()).getPropertyDescriptors();

// 循环  bean里面的 所有的属性:

for( PropertyDefinition propertyDefinition: beanDefinition.getPropertys()){

for(PropertyDescriptor propertyDesc
/*这里是 bean 里面的属性*/ : ps){
if(propertyDefinition.getName().equals(propertyDesc.getName())){

// 如果相等 说明是存在 于 这个bean的。。。

Method setter = propertyDesc.getWriteMethod();
// 获取属性的 setter方法。
// 最好做一下判断:

if(setter !=
null){

Object value =
null;
if(propertyDefinition.getRef() !=
null && !"".equals(propertyDefinition.getRef().trim())){

// 注入依赖对象:
value = sigletons.get(propertyDefinition.getRef());

}else{

// 注入基本类型。。。把字符串的值  传换成  属性的值。

value = ConvertUtils.convert(propertyDefinition.getValue(), propertyDesc.getPropertyType());

}
setter.setAccessible(true);
// 允许访问 私有的方法。。
setter.invoke(bean, value);// 把引用对象注入到属性。

}
break;

}
}
}
} catch (Exception e) {

e.printStackTrace();
}
}
}
}

/**
* 通过反射技术,完成  bean 的实例化:
*/
private void instanceBeans() {

for(BeanDefinition beanDefinition : beanDefines){

try {
if(beanDefinition.getClassName() !=
null && !"".equals(beanDefinition.getClassName().trim())){

sigletons.put(beanDefinition.getId(), Class.forName(beanDefinition.getClassName()).newInstance());

}
} catch (Exception e) {

e.printStackTrace();
}
}

}

/**
* 读取  XML 的配置文件:
* @param fileName

*/
@SuppressWarnings("unchecked")

private void readXML(String fileName) {

// 创建读取器:

SAXReader saxReader = new SAXReader();

Document document = null;

try{
URL xmlPath = this.getClass().getClassLoader().getResource(fileName);

document = saxReader.read(xmlPath);
// 读取文件的内容。。。

Map<String,String> nsMap =
new HashMap<String,String>();
nsMap.put("ns", "http://www.springframework.org/schema/beans"); // 加入命名空间

// 创建beans/bean 查询路径。

XPath xsub = document.createXPath("//ns:beans/ns:bean");

// 设置命名空间。

xsub.setNamespaceURIs(nsMap);

// 获取文档下 所有bean节点:

List<Element> beans = xsub.selectNodes(document);

for(Element element : beans){

String id = element.attributeValue("id");
// 获取id属性值。
String clazz = element.attributeValue("class");
//  获取  class 属性值。
BeanDefinition beanDefine =
new BeanDefinition(id, clazz);

// 查询的相对路径:

XPath propertysub = element.createXPath("ns:property");

propertysub.setNamespaceURIs(nsMap);// 设置命名空间。

List<Element> propertys = propertysub.selectNodes(element);

for(Element property : propertys){

String propertyName = property.attributeValue("name");

String propertyRef = property.attributeValue("ref");

String propertyValue = property.attributeValue("value");

System.out.println(propertyName +
"==" + propertyRef);
PropertyDefinition propertyDefinition =
new PropertyDefinition(propertyName, propertyRef, propertyValue);

// 放到  bean里面去:

beanDefine.getPropertys().add(propertyDefinition);

}
beanDefines.add(beanDefine);
}

}catch(Exception e){

e.printStackTrace();
}
}

/**
* 获取  bean实例
* @param beanName
* @return
*/
public Object getBean(String beanName){

return this.sigletons.get(beanName);

}

}

package junit.test;

import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.net.URL;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.apache.commons.beanutils.ConvertUtils;
import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.XPath;
import org.dom4j.io.SAXReader;

public class ItmClassPathXMLApplicationContext {

private List<BeanDefinition> beanDefines = new ArrayList<BeanDefinition>();
// 存放实例
private Map<String,Object> sigletons = new HashMap<String,Object>();

public ItmClassPathXMLApplicationContext(String fileName){
this.readXML(fileName);
this.instanceBeans();
this.annotationInject();
this.injectObject();
}

/**
* 实现注解的方式  注入:
*
*/
private void annotationInject() {
// 循环所有的 bean对象:
for(String beanName  : sigletons.keySet()){
// 获取bean对象:
Object bean = sigletons.get(beanName);
// 判断 bean对象是否存在:
if(bean != null){
// 得到 bean的属性描述:
try {
PropertyDescriptor[] ps = Introspector.getBeanInfo(bean.getClass()).getPropertyDescriptors();
for(PropertyDescriptor propertyDesc /*这里是 bean 里面的属性*/ : ps){
// 获取setter方法:
Method setter = propertyDesc.getWriteMethod();
if(setter != null  && setter.isAnnotationPresent(ItmResource.class)){  // 还要   看看   setter方法是否存在 注解:
// 取得注解:
ItmResource resource = setter.getAnnotation(ItmResource.class);

Object value  = null;
if(resource.name() != null && !"".equals(resource.name())){
// 判断是否在  集合里面:从集合里取出来  注入进去。
value = sigletons.get(resource.name());
}else{
// 如果 没有指定 name属性  怎么办 ?
value = sigletons.get(propertyDesc.getName());
if(value == null){
for(String key  : sigletons.keySet()){ // 历遍所有的bean
// 是否存在  类型匹配的bean。
if(propertyDesc.getPropertyType().isAssignableFrom(sigletons.get(key).getClass())){
value = sigletons.get(key);
break;
}
}
}
}
setter.setAccessible(true);
setter.invoke(bean, value);  // 注入进去。
}
}
// 对字段进行处理:
Field[] fields  = bean.getClass().getDeclaredFields();
for(Field field : fields ){
if(field.isAnnotationPresent(ItmResource.class)){
// 取得注解:
ItmResource resource = field.getAnnotation(ItmResource.class);

Object value  = null;
if(resource.name() != null && !"".equals(resource.name())){
// 判断是否在  集合里面:从集合里取出来  注入进去。
value = sigletons.get(resource.name());
}else{
// 如果 没有指定 name属性  怎么办 ?
value = sigletons.get(field.getName());  // 有 name直接取得对象。
if(value == null){
for(String key  : sigletons.keySet()){ // 历遍所有的bean
// 是否存在  类型匹配的bean。 【再根据 字段的类型去寻找】
if(field.getType().isAssignableFrom(sigletons.get(key).getClass())){
value = sigletons.get(key);
break;
}
}
}
}

field.setAccessible(true);
field.set(bean, value);
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
}

}

private void injectObject() {
for(BeanDefinition beanDefinition : beanDefines){
// 得到  bean 。。
Object bean = sigletons.get(beanDefinition.getId());
if(bean != null){
try {
// 得到 bean的属性描述:
PropertyDescriptor[] ps = Introspector.getBeanInfo(bean.getClass()).getPropertyDescriptors();
// 循环  bean里面的 所有的属性:
for( PropertyDefinition propertyDefinition: beanDefinition.getPropertys()){
for(PropertyDescriptor propertyDesc /*这里是 bean 里面的属性*/ : ps){
if(propertyDefinition.getName().equals(propertyDesc.getName())){
// 如果相等 说明是存在 于 这个bean的。。。
Method setter = propertyDesc.getWriteMethod();   // 获取属性的 setter方法。
// 最好做一下判断:
if(setter != null){

Object value = null;
if(propertyDefinition.getRef() != null && !"".equals(propertyDefinition.getRef().trim())){
// 注入依赖对象:
value = sigletons.get(propertyDefinition.getRef());
}else{
// 注入基本类型。。。把字符串的值  传换成  属性的值。
value = ConvertUtils.convert(propertyDefinition.getValue(), propertyDesc.getPropertyType());
}
setter.setAccessible(true); // 允许访问 私有的方法。。
setter.invoke(bean, value);// 把引用对象注入到属性。
}
break;
}
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
}

/**
* 通过反射技术,完成  bean 的实例化:
*/
private void instanceBeans() {
for(BeanDefinition beanDefinition : beanDefines){
try {
if(beanDefinition.getClassName() != null && !"".equals(beanDefinition.getClassName().trim())){
sigletons.put(beanDefinition.getId(), Class.forName(beanDefinition.getClassName()).newInstance());
}
} catch (Exception e) {
e.printStackTrace();
}
}

}

/**
* 读取  XML 的配置文件:
* @param fileName
*/
@SuppressWarnings("unchecked")
private void readXML(String fileName) {
// 创建读取器:
SAXReader saxReader = new SAXReader();
Document document = null;

try{
URL xmlPath = this.getClass().getClassLoader().getResource(fileName);
document = saxReader.read(xmlPath); // 读取文件的内容。。。

Map<String,String> nsMap = new HashMap<String,String>();
nsMap.put("ns", "http://www.springframework.org/schema/beans"); // 加入命名空间

// 创建beans/bean 查询路径。
XPath xsub = document.createXPath("//ns:beans/ns:bean");

// 设置命名空间。
xsub.setNamespaceURIs(nsMap);

// 获取文档下 所有bean节点:
List<Element> beans = xsub.selectNodes(document);
for(Element element : beans){
String id = element.attributeValue("id");             // 获取id属性值。
String clazz = element.attributeValue("class");		  //  获取  class 属性值。
BeanDefinition beanDefine = new BeanDefinition(id, clazz);

// 查询的相对路径:
XPath propertysub = element.createXPath("ns:property");
propertysub.setNamespaceURIs(nsMap);// 设置命名空间。

List<Element> propertys = propertysub.selectNodes(element);
for(Element property : propertys){
String propertyName = property.attributeValue("name");
String propertyRef = property.attributeValue("ref");

String propertyValue = property.attributeValue("value");

System.out.println(propertyName + "==" + propertyRef);
PropertyDefinition propertyDefinition = new PropertyDefinition(propertyName, propertyRef, propertyValue);

// 放到  bean里面去:
beanDefine.getPropertys().add(propertyDefinition);

}
beanDefines.add(beanDefine);
}

}catch(Exception e){
e.printStackTrace();
}
}

/**
* 获取  bean实例
* @param beanName
* @return
*/
public Object getBean(String beanName){
return this.sigletons.get(beanName);
}

}


3,

[java]
view plaincopyprint?

/**
* 1 解析属性。
* 2 看看 在哪里 有注解。
* 3 看看 有没有 配置 name属性。如果没有,再怎么办?
*
* @author Administrator
*
*/

@Retention(RetentionPolicy.RUNTIME)
// 运行期。
@Target( {ElementType.FIELD ,ElementType.METHOD})
// 指定在 字段 与 方法上。
public @interface ItmResource {

public String name()
default "";

}

[java]
view plaincopyprint?

ItmClassPathXMLApplicationContext ctx = new ItmClassPathXMLApplicationContext("beans.xml");

PersonService personService = (PersonService) ctx.getBean("personService");

personService.save();

ItmClassPathXMLApplicationContext ctx = new ItmClassPathXMLApplicationContext("beans.xml");
PersonService personService = (PersonService) ctx.getBean("personService");
personService.save();
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: