Spring注解、控制反转

1、@Autowired

        @Autowired顾名思义,就是自动装配,其作用是为了消除代码Java代码里面的getter/setter与bean属性中的property。当然,getter看个人需求,如果私有属性需要对外提供的话,应当予以保留。

        @Autowired默认按类型匹配的方式,在容器查找匹配的Bean,当有且仅有一个匹配的Bean时,Spring将其注入@Autowired标注的变量中。

2、Qualifier(指定注入Bean的名称)

        如果容器中有一个以上匹配的Bean,则可以通过@Qualifier注解限定Bean的名称,

3、Resource

        @Resource注解与@Autowired注解作用非常相似

        这是详细一些的用法,说一下@Resource的装配顺序:

        (1)、@Resource后面没有任何内容,默认通过name属性去匹配bean,找不到再按type去匹配

        (2)、指定了name或者type则根据指定的类型去匹配bean

        (3)、指定了name和type则根据指定的name和type去匹配bean,任何一个不匹配都将报错

然后,区分一下@Autowired和@Resource两个注解的区别:

        (1)、@Autowired默认按照byType方式进行bean匹配,@Resource默认按照byName方式进行bean匹配

        (2)、@Autowired是Spring的注解,@Resource是J2EE的注解,这个看一下导入注解的时候这两个注解的包名就一清二楚了

        Spring属于第三方的,J2EE是Java自己的东西,因此,建议使用@Resource注解,以减少代码和Spring之间的耦合。

4、Service

        上面这个例子,还可以继续简化,因为spring的配置文件里面还有15行~17行三个bean,下一步的简化是把这三个bean也给去掉,使得spring配置文件里面只有一个自动扫描的标签,增强Java代码的内聚性并进一步减少配置文件。

        要继续简化,可以使用@Service。

        使用注解来构造IoC容器

        用注解来向Spring容器注册Bean。需要在applicationContext.xml中注册<context:component-scan base-package=”pagkage1[,pagkage2,…,pagkageN]”/>。

        如:在base-package指明一个包

        <context:component-scan base-package="cn.gacl.java"/>

        表明cn.gacl.java包及其子包中,如果某个类的头上带有特定的注解【@Component/@Repository/@Service/@Controller】,就会将这个对象作为Bean注册进Spring容器。也可以在中指定多个包,如:

        <context:component-scan base-package="cn.gacl.dao.impl,cn.gacl.service.impl,cn.gacl.action"/>

        多个包逗号隔开。


1、@Component

        @Component是所有受Spring 管理组件的通用形式,@Component注解可以放在类的头上,        @Component不推荐使用。

2、@Controller

        @Controller对应表现层的Bean,也就是Action,

        使用@Controller注解标识UserAction之后,就表示要把UserAction交给Spring容器管理,Spring容器中会存在一个名字为"userAction"action,这个名字是根据UserAction类名来取的。注意:如果@Controller不指定其value【@Controller】,则默认的bean名字为这个类的类名首字母小写如果指定value【@Controller(value="UserAction")】或者【@Controller("UserAction")】,则使用value作为bean的名字

        这里的UserAction还使用了@Scope注解,@Scope("prototype")表示将Action的范围声明为原型,可以利用容器的scope="prototype"来保证每一个请求有一个单独的Action来处理,避免strutsAction的线程安全问题。spring 默认scope 是单例模式(scope="singleton"),这样只会创建一个Action对象,每次访问都是同一Action对象,数据不安全,struts2 是要求每次次访问都对应不同的Actionscope="prototype" 可以保证当有请求的时候都创建一个Action对象。

3、@Service

@Service对应的是业务层Bean

     @Service("userService")注解是告诉Spring,当Spring要创建UserServiceImpl的的实例时,bean的名字必须叫做"userService",这样当Action需要使用UserServiceImpl的的实例时,就可以由Spring创建好的"userService",然后注入给Action:在Action只需要声明一个名字叫"userService"的变量来接收由Spring注入的"userService"即可,具体代码如下:

// 注入userService@Resource(name = "userService")privateUserService userService;

控制反转、依赖注入

        注意:在Action声明的"userService"变量的类型必须是"UserServiceImpl"或者是其父类"UserService",否则由于类型不一致而无法注入,由于Action中的声明的"userService"变量使用了@Resource注解去标注,并且指明了其name = "userService",这就等于告诉Spring,说我Action要实例化一个"userService",你Spring快点帮我实例化好,然后给我,当Spring看到userService变量上的@Resource的注解时,根据其指明的name属性可以知道,Action中需要用到一个UserServiceImpl的实例,此时Spring就会把自己创建好的名字叫做"userService"的UserServiceImpl的实例注入给Action中的"userService"变量,帮助Action完成userService的实例化,这样在Action中就不用通过"UserService userService = new UserServiceImpl();"这种最原始的方式去实例化userService了。如果没有Spring,那么当Action需要使用UserServiceImpl时,必须通过"UserService userService = new UserServiceImpl();"主动去创建实例对象,但使用了Spring之后,Action要使用UserServiceImpl时,就不用主动去创建UserServiceImpl的实例了,创建UserServiceImpl实例已经交给Spring来做了,Spring把创建好的UserServiceImpl实例给Action,Action拿到就可以直接用了。Action由原来的主动创建UserServiceImpl实例后就可以马上使用,变成了被动等待由Spring创建好UserServiceImpl实例之后再注入给Action,Action才能够使用。这说明Action对"UserServiceImpl"类的“控制权”已经被“反转”了,原来主动权在自己手上,自己要使用"UserServiceImpl"类的实例,自己主动去new一个出来马上就可以使用了,但现在自己不能主动去new "UserServiceImpl"类的实例,new "UserServiceImpl"类的实例的权力已经被Spring拿走了,只有Spring才能够new "UserServiceImpl"类的实例,而Action只能等Spring创建好"UserServiceImpl"类的实例后,再“恳求”Spring把创建好的"UserServiceImpl"类的实例给他,这样他才能够使用"UserServiceImpl",这就是Spring核心思想“控制反转”,也叫“依赖注入”,“依赖注入”也很好理解,Action需要使用UserServiceImpl干活,那么就是对UserServiceImpl产生了依赖,Spring把Acion需要依赖的UserServiceImpl注入(也就是“给”)给Action,这就是所谓的“依赖注入”。对Action而言,Action依赖什么东西,就请求Spring注入给他,对Spring而言,Action需要什么,Spring就主动注入给他。

4、@ Repository

        @Repository对应数据访问层Bean  ,例如:

        @Repository(value="userDao")publicclassUserDaoImplextendsBaseDaoImpl {………}

        @Repository(value="userDao")注解是告诉Spring,让Spring创建一个名字叫"userDao"的UserDaoImpl实例。

        当Service需要使用Spring创建的名字叫"userDao"的UserDaoImpl实例时,就可以使用@Resource(name = "userDao")注解告诉Spring,Spring把创建好的userDao注入给Service即可。

        // 注入userDao,从数据库中根据用户Id取出指定用户时需要用到@Resource(name = "userDao")privateBaseDao userDao;


Spring常用注解汇总 

        本文汇总了Spring的常用注解,以方便大家查询和使用,具体如下:

        使用注解之前要开启自动扫描功能,其中base-package为需要扫描的包(含子包)。

        <context:component-scan base-package="cn.test"/>


        @Configuration把一个类作为一个IoC容器,它的某个方法头上如果注册了@Bean,就会作为这个Spring容器中的Bean。

        @Scope注解 作用域

        @Lazy(true) 表示延迟初始化

        @Service用于标注业务层组件、 

        @Controller用于标注控制层组件(如struts中的action)

        @Repository用于标注数据访问组件,即DAO组件。

        @Component泛指组件,当组件不好归类的时候,我们可以使用这个注解进行标注。

        @Scope用于指定scope作用域的(用在类上)

        @PostConstruct用于指定初始化方法(用在方法上)

        @PreDestory用于指定销毁方法(用在方法上)

        @DependsOn:定义Bean初始化及销毁时的顺序

        @Primary:自动装配时当出现多个Bean候选者时,被注解为@Primary的Bean将作为首选者,否则将抛出异常

        @Autowired 默认按类型装配,如果我们想使用按名称装配,可以结合@Qualifier注解一起使用。如下:

        @Autowired @Qualifier("personDaoBean") 存在多个实例配合使用

        @Resource默认按名称装配,当找不到与名称匹配的bean才会按类型装配。

        @PostConstruct 初始化注解

        @PreDestroy 摧毁注解 默认 单例  启动就加载

        @Async异步方法调用

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容