1.Spring容器

Spring容器最基本的接口就是BeanFactory, 负责配置,创建和管理bean。我们通常不直接使用BeanFactory接口,而是使用其子接口ApplicationContext.

接口ApplicationContext常用实现类是FileSystemXmlApplicationContext和ClassPathXmlApplicationContext等。后者最常用。

ApplicationContext的实例就是一个容器,Spring容器的意义在于创建和初始化类对象,管理类的依赖关系。

2.ApplicationContext的事件机制

AppicationContext的事件机制是观察者模式的实现,按以下方式可以实现。

  • 首先我们需要自定义一个事件类,此事件类是需要继承ApplicationEvent类
  • 然后我们定义一个监听类,监听类作为一个Spring容器中的bean,同时需要实现ApplicationListner接口
  • 然后我们就可以使用ApplicationContext的实例发布事件,相应监听类的实例(bean)负责监听具体的事件

下面是一个简单的例子,

事件类,必须继承ApplicationEvent,是否为Spring容器的bean无所谓,

 package spi;

 import org.springframework.context.ApplicationEvent;

 public class EmailEvent extends ApplicationEvent {
private String address;
public String getAddress() {
return address;
}
public void setAddress(String address) {
this.address = address;
}
public String getText() {
return text;
}
public void setText(String text) {
this.text = text;
}
private String text;
public EmailEvent(Object source) {
super(source);
}
public EmailEvent(Object source, String address, String text) {
super(source);
this.address = address;
this.text = text;
}
}

监听类,必须作为Spring容器的bean,同时需要实现ApplicationListener接口,重写onApplicationEvent方法

 package spi;

 import org.springframework.context.ApplicationEvent;
import org.springframework.context.ApplicationListener; public class EmailNotifier implements ApplicationListener { @Override
public void onApplicationEvent(ApplicationEvent evt) {
if (evt instanceof EmailEvent) {
EmailEvent emailEvent = (EmailEvent)evt;
System.out.println("邮件地址:"+emailEvent.getAddress());
System.out.println("邮件内容:"+emailEvent.getText());
} else {
System.out.println("其他事件:"+evt);
}
} }

将监听类配置进Spring容器配置文件中,并没有什么特殊之处,

<bean class="spi.EmailNotifier" />

下面写一个测试类,

     public static void test2() {
ApplicationContext ctx = new ClassPathXmlApplicationContext("beans.xml");
EmailEvent ele = new EmailEvent("test", "test@test.com", "this is a test");
ctx.publishEvent(ele);
}

执行测试类,发现我们不仅监听到了想要监听的事件EmailEvent,同时还有一个系统事件ContextRefreshedEvent也被监听到了,输出如下,

 其他事件:org.springframework.context.event.ContextRefreshedEvent[source=org.springframework.context.support.ClassPathXmlApplicationContext@1873eb2: startup date [Fri Feb 03 01:19:21 CST 2017]; root of context hierarchy]
邮件地址:test@test.com
邮件内容:this is a test

3.ApplicationContext的国际化支持

ApplicationContext接口继承了MessageSource接口,因此具有国际化功能。MessageSource接口提供了getMessage(...)方法用来进行字符串转换。

Spring要实现国际化,需要将MessageSource的实例配置成Spring容器中的bean,在bean的属性(即依赖注入)中配置国际化文件,

     <!-- ApplicationContext的实例将会查找是否有messageSource的实例并初始化它 -->
<bean id="messageSource" class="org.springframework.context.support.ResourceBundleMessageSource">
<!-- 这里即依赖注入 -->
<property name="basenames">
<list>
<!-- 在这里添加国际化配置文件 -->
<value>message</value>
</list>
</property>
</bean>

为此我们需要创建两份国际化配置文件,第一份为英语,文件名 message_en_US.properties,内容如下,

 str1=welcome,{0}
str2=now is : {0}

第二份为中文,文件名为message.properties,内容如下,

 str1=欢迎,{0}
str2=现在时间是:{0}

由于这个文件包含了非西欧字符,因此我们用java自带的native2ascii进行转换,命令为

 cd C:\Program Files (x86)\Java\jdk1..0_79\bin\
native2ascii C:\PROJECT\JavaBasic\PROJECT_JavaBasic\src\message.properties C:\PROJECT\JavaBasic\PROJECT_JavaBasic\src\message_zh_CN.properties

转换后得到message_zh_CN.properties文件即可。

接着写一个测试类,

     public static void test3() {
ApplicationContext ctx = new ClassPathXmlApplicationContext("beans.xml");
String str1 = ctx.getMessage("str1", new String[]{"孙悟空"},
Locale.getDefault(Locale.Category.FORMAT));
String str2 = ctx.getMessage("str2", new Object[]{new Date()},
Locale.getDefault(Locale.Category.FORMAT));
System.out.println(str1);
System.out.println(str2); String str3 = ctx.getMessage("str1", new String[]{"孙悟空"},
Locale.US);
String str4 = ctx.getMessage("str2", new Object[]{new Date()},
Locale.US);
System.out.println(str3);
System.out.println(str4);
}

我们分别在中文环境和英文环境测试了两个字符串str1和str2,得到结果如下,

 欢迎,孙悟空
现在时间是:17-2-3 上午1:31
welcome,孙悟空
now is : 2/3/17 1:31 AM

上面的文字“孙悟空”因为是直接写在代码里面,不属于properties配置文件的,因此在中英文结果都保留原来的文字。

4.让Bean获取Spring容器

如果需要让Bean主动获取它所在的Spring容器的引用,可以让该Bean实现BeanFactoryAware接口,并实现setBeanFactory(BeanFactory beanFactory)方法。

下面是一个例子,

 package spi;

 import java.util.Locale;

 import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware; public class GetContextViaBean implements ApplicationContextAware {
private ApplicationContext ctx;
@Override
public void setApplicationContext(ApplicationContext ctx)
throws BeansException {
this.ctx = ctx;
}
public void SayHi(String name) {
System.out.println(ctx.getMessage("str1", new String[]{name}, Locale.US));
} }

将这个类作为一个普通bean配置进Spring中,

<bean id="getContextViaBean" class="spi.GetContextViaBean" />

下面写一个测试类,来实现前面的国际化的功能,

     public static void test4() {
ApplicationContext ctx = new ClassPathXmlApplicationContext("beans.xml");
GetContextViaBean bean = ctx.getBean("getContextViaBean", GetContextViaBean.class);
bean.SayHi("孙悟空");
}

执行结果为,

 welcome,孙悟空

与前面国际化方式相比,其实本质上没有区别,只不过前面的国际化是在测试类中,直接使用ApplicationContext的实例调用MessageSource的getMessage()方法,

而这里是在Bean中,主动去获取所在容器(ApplicationContext)的引用,并在beanzhong通过ApplicationContext的引用调用了MessageSource的getMessage()方法。

Spring中的容器的更多相关文章

  1. Spring 中 IoC 容器简介

    IoC 是一种通过描述来生成或者获取对象的技术,可以说 Spring 是一种基于 IoC 容器编程的框架 在一个系统中可以生成各种对象,并且这些对象都需要进行管理.为了描述这些对象关系,我们需要一个容 ...

  2. spring中IOC容器注册和获取bean的实例

    spring中常用的功能主要的是ioc和aop,此处主要说明下,实例注册和使用的方法,此为学习后的笔记记录总结 1.使用xml文件配置 在idea中创建maven工程,然后创建实例Person,然后在 ...

  3. 深入了解Spring中的容器

    1.创建Bean的3种方式 1.1使用构造器创建bean实例 这是最常见的方式,如果不采用构造注入,bean类需要有默认构造函数.如果采用构造注入,则需要配置xml文件的<constructor ...

  4. spring中获取容器中的Bean为什么前转成接口而不是实现类

    简单介绍一下上下文,userService是服务层接口有一个save方法,userServiceImpl是该接口的实现类重写了save方法. applicationContext.xml如图: 后台代 ...

  5. 半夜思考之查漏补缺, Spring 中的容器后处理器

    之前学 Spring 的时候 , 还没听过容器后处理器 , 但是一旦写出来 , 就会觉得似曾相识 . 容器配置器通常用于对 Spring 容器进行处理 , 并且总是在容器实例化任何其他 Bean 之前 ...

  6. Spring中Ioc容器的注入方式

    1 通过setter方法注入 bean类: package com.test; public class UserServiceImplement implements IUserService { ...

  7. 挖坟之Spring.NET IOC容器初始化

    因查找ht项目中一个久未解决spring内部异常,翻了一段时间源码.以此文总结springIOC,容器初始化过程. 语言背景是C#.网上有一些基于java的spring源码分析文档,大而乱,乱而不全, ...

  8. Spring之IOC容器

    在前面博客中介绍什么是依赖注入时有提到:依赖注入是组件之间依赖关系由容器在运行期决定,即由容器动态的将某个依赖关系注入到组件之中.那什么是容器?既然Spring框架实现了IOC,那Spring中的容器 ...

  9. Spring中常见的设计模式——单例模式

    一.单例模式的应用场景 单例模式(singleton Pattern)是指确保一个类在任何情况下都绝对只有一个实例,并提供一个全局访问点.J2EE中的ServletContext,ServletCon ...

随机推荐

  1. HTML5和CSS3中的交互新特性

    当文章的标题是一副用photoshop制作的图片,那么势必在搜索引擎中无法搜索到.并且因为图片的体积不算小.可能在网速慢的的时候不得不耐心的等待图片的刷新. 所以.我们来谈谈有没有一种新的方法能够避免 ...

  2. c22---枚举

    // // main.c // 枚举基本概念 #include <stdio.h> int main(int argc, const char * argv[]) { // int sex ...

  3. P4396 [AHOI2013]作业 分块+莫队

    这个题正解是莫队+树状数组,但是我个人非常不喜欢树状数组这种东西,所以决定用分块来水这个题.直接在莫队维护信息的时候,维护单点同时维护块内信息就行了. 莫队就是这几行核心代码: void add(in ...

  4. package-org.springframework.ui-interface:Model.class

    ylbtech-package-org.springframework.ui-interface:Model.class 1.返回顶部 1. /* * Copyright 2002-2012 the ...

  5. 第20章 Redis配置

    20.1 Redis基础配置文件 20.2 Redis备份(持久化) save 900 1 save 300 10 save 60 10000 # By default Redis will stop ...

  6. php 获取随机字符串(原创)

    //获取随机数字字母字符串 function get_rand_str($len=8){ $randArr=array_merge(range(0,9),range('a','z'),range('A ...

  7. BZOJ 1485 卡特兰数 数学

    思路: 通过打表观察 这是个卡特兰数 但是它mod的数不是质数 怎么办呢 把所有数分解质因数好了 线性筛出mindiv  顺着mindiv分解质因数 复杂度$O(nlogn)$ //By Sirius ...

  8. 创建异步对象XHR的兼容写法、get、post上传数据的方式

    兼容ie7以下,创建异步对象的函数 function creatXHR(){ if(typeof XMLHttpRequest != "undefined"){ return ne ...

  9. 【Arduino】基于arduino的激光坦克

    历经了总共40来个小时,终于将这个激光坦克做好了. 这是本人的第一件像样的arduino作品. 用arduino为主控制器,配有32路舵机驱动板(在这有些大材小用了),以及一个wr703n的路由器当做 ...

  10. XML 之 命名空间详解

    最近学xml 遇到了点小问题qaq 找了n多的博客大佬,反复看了半小时终于明白了,可能我太蠢了... 基础的知识就不赘述,直接放“栗子”,切重点: <?xml version="1.0 ...