- @Component :注冊為 Spring 管理的 Bean
- @Repository 持久層
- @Service 業(yè)務層
- @Controller 控制器層
- @Autowired 按照類型注入
- @Qualifier
- 在自動按照類型注入的基礎之上,再按照 Bean 的 id 注入
- 必須和 @Autowired 一起使用奠支;但是給方法參數(shù)注入時,可以獨立使用
- @Qualifier
@Resource:
它的作用相當于 @Autowired潘鲫,只不過 @Autowired 按 byType 自動注入,面@Resource 默認按 byName 自動注入愕贡,當找不到與名稱匹配的bean才會按類型裝配
相同點:
@Resource的作用相當于@Autowired,均可標注在字段或屬性的setter方法上卿泽。
不同點:
- 提供方:@Autowired是由org.springframework.beans.factory.annotation.Autowired提供各谚,換句話說就是由Spring提供紧憾;@Resource是由javax.annotation.Resource提供,即J2EE提供昌渤,需要JDK1.6及以上赴穗。
- 注入方式:@Autowired只按照byType 注入;@Resource默認按byName自動注入膀息,也提供按照byType 注入般眉;
- 屬性:@Autowired按類型裝配依賴對象,默認情況下它要求依賴對象必須存在潜支,如果允許null值甸赃,可以設置它required屬性為false。如果我們想使用按名稱裝配毁腿,可以結合@Qualifier注解一起使用辑奈。
- @Resource注解的使用性更為靈活,可指定名稱已烤,也可以指定類型 ;@Autowired注解進行裝配容易拋出異常妓羊,特別是裝配的bean類型有多個的時候胯究,而解決的辦法是需要在增加@Qualifier進行限定。
@PostConstruct 和 @PreDestroy
JSR-250 為初始化之后/銷毀之前方法的指定定義了兩個注解類躁绸,分別是 @PostConstruct 和 @PreDestroy裕循,這兩個注解只能應用于方法上。標注了 @PostConstruct 注解的方法將在類實例化后調用净刮,而標注了 @PreDestroy 的方法將在類銷毀之前調用剥哑。
@Resource、@PostConstruct 以及@PreDestroy淹父。它們不屬于 Spring 框架株婴,而是在 Java 中定義的。只不過 Spring 框架支持這種 Java注解暑认。
@Configuration 和 @Bean
Spring的官方團隊說 @Component可以替代 @Configuration注解困介;@Bean注解只能寫在方法上,表明使用此方法創(chuàng)建一個對象蘸际,并且放入 Spring 容器座哩。
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component //看這里!A竿根穷!
public @interface Configuration
區(qū)別:
@Bean 注解主要用于方法上姜骡,有點類似于工廠方法,當使用了 @Bean 注解屿良,我們可以連續(xù)使用多種定義bean時用到的注解圈澈,譬如用 @Qualifier 注解定義工廠方法的名稱,用 @Scope 注解定義該 bean 的作用域范圍管引,譬如是 singleton 還是 prototype 等士败。
Spring 中新的 Java 配置支持的核心就是 @Configuration 注解的類。這些類主要包括 @Bean 注解的方法來為 Spring 的 IoC 容器管理的對象定義實例褥伴,配置和初始化邏輯谅将。
使用 @Configuration 來注解類表示類可以被 Spring 的 IoC 容器所使用,作為 bean 定義的資源重慢。
/*
@Configuation等價于<Beans></Beans>
@Bean等價于<Bean></Bean>
*/
@Configuration
public class AppConfig {
@Bean
public MyService myService() {
return new MyServiceImpl();
}
}
這和 Spring 的 XML 文件中的非常類似
<beans>
<bean id="myService" class="com.acme.services.MyServiceImpl"/>
</beans>
舉例對比說明
在@Component類中使用方法或字段時不會使用CGLIB增強(及不使用代理類:調用任何方法饥臂,使用任何變量,拿到的是原始對象似踱,后面會有例子解釋)隅熙。而在@Configuration類中使用方法或字段時則使用CGLIB創(chuàng)造協(xié)作對象(及使用代理:拿到的是代理對象);當調用@Bean注解的方法時它不是普通的Java語義,而是從容器中拿到由Spring生命周期管理核芽、被Spring代理甚至依賴于其他Bean的對象引用囚戚。在@Component中調用@Bean注解的方法和字段則是普通的Java語義,不經過CGLIB處理轧简。
@Configuration
public static class Config {
@Bean
public SimpleBean simpleBean() {
return new SimpleBean();
}
@Bean
public SimpleBeanConsumer simpleBeanConsumer() {
return new SimpleBeanConsumer(simpleBean());
}
}
- 第一個代碼正常工作驰坊,正如預期的那樣,
SimpleBeanConsumer
將會得到一個單例SimpleBean
的鏈接哮独。 - 第二個配置是完全錯誤的拳芙,因為 Spring 會創(chuàng)建一個
SimpleBean
的單例 bean,但是SimpleBeanConsumer
將獲得另一個SimpleBean
實例(也就是相當于直接調用new SimpleBean()
皮璧,這個 bean 是不歸 Spring 管理的)舟扎,既new SimpleBean()
實例是 Spring 上下文控件之外的。
@Component
public static class Config {
@Bean
public SimpleBean simpleBean() {
return new SimpleBean();
}
@Bean
public SimpleBeanConsumer simpleBeanConsumer() {
return new SimpleBeanConsumer(simpleBean());
}
}
原因總結
使用 @Configuration悴务,所有標記為 @Bean的方法將被包裝成一個 CGLIB 包裝器睹限,它的工作方式就好像是這個方法的第一個調用,那么原始方法的主體將被執(zhí)行惨寿,最終的對象將在 Spring 上下文中注冊邦泄。所有進一步的調用只返回從上下文檢索的 bean。
在上面的第二個代碼塊中裂垦,新的SimpleBeanConsumer(simpleBean())
只調用一個純 java 方法顺囊。為了糾正第二個代碼塊,我們可以這樣做
@Component
public static class Config {
@Autowired
SimpleBean simpleBean;
@Bean
public SimpleBean simpleBean() {
return new SimpleBean();
}
@Bean
public SimpleBeanConsumer simpleBeanConsumer() {
return new SimpleBeanConsumer(simpleBean);
}
}
@ComponentScan
用于指定 Spring 在初始化容器時要掃描的包,用注解標識的類會被 Spring 自動掃描并且裝入 Bean 容器中蕉拢,作用和在 Spring 的 xml 配置文件中的:<context:component-scan base-package="com..."/>
是一樣的特碳。
-
有注解(開啟掃描)
- 有路徑:掃描指定路徑
- 沒路徑:默認掃描當前配置類所在包及其子包下組件
沒有注解(不開啟掃描)
使用 includeFilters 來按照規(guī)則只包含某些包的掃描
使用 excludeFilters 來按照規(guī)則排除某些包的掃描
context:annotation-config和context:component-scan
<context-annotation-config/>
該配置的主要作用是“激活”已聲明的 bean诚亚,即“激活” spring 容器內配置的 bean。
如:在 spring 容器內配置了<bean name="demo" class="com.example.Demo"></bean>
午乓,那么則可以使用 @Autowired 注解將 demo 注入相應對象中站宗。但是约啊,該配置對@Component颖对、@Controller、@Service幕侠、@Repository注解的蒸其,但沒有在 spring 容器注冊過的bean無效敏释。
<context:component-scan base-package="" />
該配置包含了<context-annotation-config />
配置的作用,與之不同的是:該配置可以掃描 base-package 指定包下@Component摸袁、@Controller钥顽、@Service、@Repository 注解并將被注解 bean 注冊到 spring 容器內靠汁,使之生效蜂大。
注意:1. 當這兩個配置同時配置時,<context-annotation-config />
將失效蝶怔,以<context:component-scan base-package="" />
為準奶浦。2. @Component、@Controller踢星、@Service财喳、@Repository這些注解本身并不具有聲明注冊bean的功能,在沒有<context:component-scan>
掃描解析之前是沒有任何作用的斩狱。
@PropertySource
注解在類上,指定文件地址注入配置文件
@Value
注解在變量上扎瓶,調用資源(普通文件所踊,網址,配置文件概荷,系統(tǒng)環(huán)境變量等)
@Import
用于導入其他配置類秕岛,在引入其他配置類時,可以不用再寫@Configuration 注解误证。當然继薛,寫上也沒問
@profile
用來標明當前運行環(huán)境的注解
@ModelAttribute
使用地方有三種:
1、標記在方法上愈捅。
標記在方法上遏考,會在每一個@RequestMapping標注的方法前執(zhí)行,如果有返回值蓝谨,則自動將該返回值加入到ModelMap中灌具。
(1) 在有返回的方法上:
當ModelAttribute設置了value青团,方法返回的值會以這個value為key,以參數(shù)接受到的值作為value咖楣,存入到Model中督笆,如下面的方法執(zhí)行之后,最終相當于 model.addAttribute("user_name", name);假如 @ModelAttribute沒有自定義value诱贿,則相當于
model.addAttribute("name", name);
(2) 在沒返回的方法上:
需要手動model.add方法
我們在當前類下建一個請求方法:
在瀏覽器中輸入訪問地址并且加上參數(shù):
http://localhost:8081/api/test/mod?name=我是小菜&age=12
最終輸出如下:
2娃肿、標記在方法的參數(shù)上。
標記在方法的參數(shù)上珠十,會將客戶端傳遞過來的參數(shù)按名稱注入到指定對象中料扰,并且會將這個對象自動加入ModelMap中,便于View層使用.我們在上面的類中加入一個方法如下
在瀏覽器中輸入訪問地址并且加上參數(shù):
http://localhost:8081/api/test/mod2?name=我是小菜&age=12
最終輸出:
從結果就能看出宵睦,用在方法參數(shù)中的@ModelAttribute注解记罚,實際上是一種接受參數(shù)并且自動放入Model對象中,便于使用壳嚎。