Spring核心——纯Java运行与@Bean

3.0新增容器启动方法

在3.0之前的Spring核心框架中,我们启动一个Spring容器必须使用一个XML文件。而到了3.X之后的版本Spring为创建容器新增了一个入口类——AnnotationConfigApplicationContext

AnnotationConfigApplicationContext和过去的ClassPathXmlApplicationContext、FileSystemXmlApplicationContext等方法不同的是他不用再指定任何XML配置文件,而是可以通过指定类向容器添加Bean。我们通过几个简单的例子来说明他的使用。

直接添加Bean

我们可以通过AnnotationConfigApplicationContext直接向容器添加指定的类作为Bean,先定义我们的class:

packagechkui.springcore.example.javabase.simple.pureBean;classLolBean{publicStringtoString(){return"I AM LOL!";}}classWowBean{publicStringtoString(){return"I AM WOW!";}}

然后向容器添加这些Bean:

packagechkui.springcore.example.javabase.simple;publicclassWithoutAnnotation{publicstaticvoidmain(String[] args){ApplicationContext ctx =newAnnotationConfigApplicationContext(WowBean.class, LolBean.class);System.out.println(ctx.getBean(WowBean.class));System.out.println(ctx.getBean(LolBean.class));}}

这样就启动了一个Spring的容器,并且容器中包含了WowBean和LolBean这两个类的单例。

替代<beans>标签

@Configuration在之前介绍Spring核心容器的文章中出现过一两次,配合各种注解的使用@Configuration可以替代<beans>配置中的所有功能。基本上AnnotationConfigApplicationContext和@Configuration组合使用就可以实现Spring容器纯Java启动。请看下面的例子。

我们在前面例子的基础上增加几个类:

packagechkui.springcore.example.javabase.simple.bean;publicclassDotaBean{publicStringtoString(){return"I AM Dota!";}}@ComponentpublicclassPseBean{@OverridepublicStringtoString(){return"I AM PSE!";}}

注意DotaBean上是没有@Component注解的。然后添加@Configuration配置:

packagechkui.springcore.example.javabase.simple.bean;@Configuration@ComponentScan("chkui.springcore.example.javabase.simple.bean")publicclassConfig{@BeanpublicDotaBeandotaBean(){returnnewDotaBean();}}

最后运行他们:

packagechkui.springcore.example.javabase.simple;publicclassWithScan{publicstaticvoidmain(String[] args){ApplicationContext ctx =newAnnotationConfigApplicationContext(Config.class, WowBean.class, LolBean.class);System.out.println(ctx.getBean(Config.class));System.out.println(ctx.getBean(PseBean.class));System.out.println(ctx.getBean(WowBean.class));System.out.println(ctx.getBean(LolBean.class));System.out.println(ctx.getBean(DotaBean.class));}}

@Component已经在Stereotype组件与Bean扫描这篇文章介绍过,@ComponentScan的作用等价于标签,属性参数都是一一对应的,只不过前者是驼峰命名规则(camelCase)——@ComponentScan(basePackages="..."),后者是短横线命名规则(kebab-case)——。实际上使用Annotation来替换XML配置中的内容,大部分都使用这种转换方式。

@Configuration和@Bean标签会在后续的内容中详细介绍。@Bean主要用于方法标记,表明这个方法返回一个要添加到容器中的Bean。

AnnotationConfigApplicationContext的其他使用方法

除了以上常规的使用方法,AnnotationConfigApplicationContext还有其他方式向容器添加Bean。

可以使用AnnotationConfigApplicationContext::register方法来添加配置和Bean:

publicstaticvoidmain(String[] args){    AnnotationConfigApplicationContext ctx =newAnnotationConfigApplicationContext();//动态添加配置文件ctx.register(Config1.class, Config2.class);//动态添加Beanctx.register(Bean1.class);//刷新ctx.refresh();}

注意最后的refresh方法,这个方法来源于ConfigurableApplicationContext接口,然后是在AbstractApplicationContext中实现的。他的过程相当于销毁之前已经创建的资源,然后再重新创建了一个新的容器。这里的代码会执行以下几步:

new AnnotationConfigApplicationContext():创建一个新的容器,容器中没有自定义的Bean。

AnnotationConfigApplicationContext::register:向容器添加BeanDefinition,但是这些BeanDefinition并没有转化为容器中的Bean。

ConfigurableApplicationContext::refresh():纳入新添加的BeanDefinition重建容器。

还可以直接使用AnnotationConfigApplicationContext::scan方法扫描指定的路径:

publicstaticvoidmain(String[] args){    AnnotationConfigApplicationContext ctx =newAnnotationConfigApplicationContext();    ctx.scan("com.acme");    ctx.refresh();}

 执行原理和上面介绍的一样。

需要注意的是:如果你的工程中需要使用AnnotationConfigApplicationContext::register、AnnotationConfigApplicationContext::scan等方法创建容器和其中Bean的依赖关系,所有的Bean都只能在register或scan中添加。如果你既在AnnotationConfigApplicationContext的构造方法中添加了Bean,又使用AnnotationConfigApplicationContext::refresh()方法会抛出一个重复执行refresh的异常。AnnotationConfigApplicationContext::refresh()方法全局也只能被调用一次。

@Bean注解

@Bean注解等价于配置文件中的<bean>标签,对应的参数也是将短横线命名切换为驼峰命名——<bean init-method="..."> => @Bean(initMethod="...")。@Bean注解只能使用在方法上,方法必须是在@Configuration标记的类或者其他Bean中,两者存在的差异会在后续的文章中介绍。下面通过一个例子来说明Bean的使用。

定义两个要添加到容器中的Bean:

packagechkui.springcore.example.javabase.beanAnnotation.bean;classFinalFantasy{@OverridepublicStringtoString(){return"Final Fantasy 1~15";}publicvoidinit(){System.out.println("Final Fantasy init!");}publicvoiddestroy(){System.out.println("Final Fantasy destroy!");}}classDragonQuest{publicStringtoString(){return"Dragon Quest 1~11";}@PostConstructpublicvoidinit(){System.out.println("Dragon Quest init!");}@PreDestroypublicvoiddestroy(){System.out.println("Dragon Quest destroy!");}}

定义一个功能接口及其实现类:

packagechkui.springcore.example.javabase.beanAnnotation.bean;interfaceSupport{voidsetFinalFantasy(FinalFantasy ff);FinalFantasygetFinalFantasy();}classSupportImplimplementsSupport{privateFinalFantasy ff;publicvoidsetFinalFantasy(FinalFantasy ff){this.ff = ff;}publicFinalFantasygetFinalFantasy(){returnff;}}

然后顶一个@Configuration类:

packagechkui.springcore.example.javabase.beanAnnotation.bean;publicclassBeanAnnotationConfig{@BeanpublicSupportsupport(FinalFantasy ff){Support support =newSupportImpl();support.setFinalFantasy(ff);returnsupport;}@Bean(initMethod="init", destroyMethod="destroy")@Description("Final Fantasy")publicFinalFantasyfinalFantasy(){returnnewFinalFantasy();}@Bean(name= {"dragon-quest","DragonQuest"})publicDragonQuestdragonQuest(){returnnewDragonQuest();}}

最后运行他们:

publicclassBeanAnnotApp{publicstaticvoidmain(String[] args){ApplicationContext ctx =newAnnotationConfigApplicationContext(BeanAnnotationConfig.class);Support support = ctx.getBean(Support.class);System.out.println(support.getFinalFantasy());System.out.println(ctx.getBean(DragonQuest.class));}}

在配置类BeanAnnotationConfig中,我们配置了3个Bean。这里的写在方法上的@Bean注解和写在配置文件中的<bean>注解一个效果:

@Bean中的initMethoddestroyMethod对应<bean>标签中的init-methoddestroy-method属性。

@Bean中的name参数只有一个值时相当于id,有多个的时候相当于设置了多个别名

Support support(FinalFantasy ff):我们可以直接在方法中暴露参数来引入其他Bean,这就类似于配置中ref的功能。

如果不指定initMethoddestroyMethod,使用JSR-330的生命周期注解(@PostConstruct、@PreDestroy)同样有效

欢迎工作一到五年的Java工程师朋友们加入Java技术交流群:659270626

群内提供免费的Java架构学习资料(里面有高可用、高并发、高性能及分布式、Jvm性能调优、Spring源码,MyBatis,Netty,Redis,Kafka,Mysql,Zookeeper,Tomcat,Docker,Dubbo,Nginx等多个知识点的架构资料)合理利用自己每一分每一秒的时间来学习提升自己,不要再用"没有时间“来掩饰自己思想上的懒惰!趁年轻,使劲拼,给未来的自己一个交代!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,036评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,046评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,411评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,622评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,661评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,521评论 1 304
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,288评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,200评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,644评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,837评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,953评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,673评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,281评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,889评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,011评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,119评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,901评论 2 355

推荐阅读更多精彩内容