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. HDU 5187

    超简单的公式题(2^n-2).不过,要过可不容易,因为会爆64位,所以,可以使用快速乘法. #include <iostream> #include <cstdio> #inc ...

  2. cocos2dx 3.1从零学习(二)——菜单、场景切换、场景传值

    回想一下上一篇的内容,我们已经学会了创建一个新的场景scene,加入sprite和label到层中.掌握了定时事件schedule. 我们能够顺利的写出打飞机的主场景框架. 上一篇的内容我练习了七个新 ...

  3. JAVA进阶-泛型

    >泛型:泛型指代了參数的类型化类型,一般被用在接口.类.方法中 >作用:用来确定參数的范围,在书写代码的时候提前检查代码的错误性 >泛型的声明,下面给出类声明,依此类推: class ...

  4. luogu3111 [USACO14DEC]牛慢跑Cow Jog_Sliver

    题目大意 有N (1 <= N <= 100,000)头奶牛在一个单人的超长跑道上慢跑,每头牛的起点位置都不同.由于是单人跑道,所有他们之间不能相互超越.当一头速度快的奶牛追上另外一头奶牛 ...

  5. hdoj--3635--Dragon Balls(并查集记录深度)

    Dragon Balls Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Tot ...

  6. 【概率证明】—— sum and product rules of probability

    1. sum and product rules of probability ⎧⎩⎨p(x)=∫p(x,y)dyp(x,y)=p(x|y)p(y) sum rule of probability 的 ...

  7. python 下串口数据的读取,解析,和保存-

    #!/usr/bin/python # -*-coding: utf-8 -*- import serial import threading import binascii from datetim ...

  8. Redis五种数据类型及应用场景

    MySql+Memcached架构的问题 实际MySQL是适合进行海量数据存储的,通过Memcached将热点数据加载到cache,加速访问,很多公司都曾经使用过这样的架构,但随着业务数据量的不断增加 ...

  9. Python金融量化

    Python股票数据分析 最近在学习基于python的股票数据分析,其中主要用到了tushare和seaborn.tushare是一款财经类数据接口包,国内的股票数据还是比较全的 官网地址:http: ...

  10. (C++)错误提示 c2352 :非静态成员函数的非法调用

    静态成员函数相当于全局函数,只是有一个类名字空间的限制.而类成员函数是成员内部的函数,同一个类的对象实例可以有很多,每一个实例都有自已不同的成员变量值,成员函数一般都是对成员自已的成员变量值在操作.所 ...