摘要:文章从Spring程序的快速使用、Bean标签的使用和其属性的具体使用,每个属性都用代码来解释,运行结果和案例也写的都很明白。

本文分享自华为云社区《怎样使用Spring的配置文件?带大家一起玩转Spring配置文件》,作者:我是一棵卷心菜 。

一、Spring程序快速入门

步骤一:导入 Spring 开发的基本包坐标

创建一个maven工程,导入Spring需要的依赖,为了方便测试,我还导入了一个Junit测试包

<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.3.14</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
</dependency>
</dependencies>

步骤二:编写 Dao 接口和实现类

接下里,需要编写一个Dao接口和其实现类,用来搭建测试的环境,话多不说,直接上代码

public interface UserDao {
void save();
}
public class UserDaoImpl implements UserDao {
@Override
public void save() {
System.out.println("保存成功~~");
}
}

步骤三:创建 Spring 核心配置文件

接口和类写完后,就开始进入正题,在类路径下(resources)创建Spring的核心配置文件,我取名为applicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
</beans>

步骤四:在 Spring 配置文件中配置 UserDaoImpl类

创建好xml配置文件后,我要想调用save()方法,就需要创建UserDaoImpl类的对象,这里采用容器的方式。

    <bean id="userdao" class="com.sht.dao.impl.UserDaoImpl"></bean>

这里的id是自定义的,最好是类名的首字母小写,方便于记忆。class属性值是要创建类对象的所在包路径,我的这个UserDaoImpl类所在的包路径就是com.sht.dao.impl.UserDaoImpl,这一点都不能写错哦!

步骤五:使用 Spring 的 API 获得 Bean 实例

接下来测试一下:

public class UserDaoDemo {
public static void main(String[] args) {
ApplicationContext applicationContext =
new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userdao = (UserDao) applicationContext.getBean("userdao");
userdao.save();
}
}

通过new ClassPathXmlApplicationContext("applicationContext.xml")获取到容器——applicationContext,然后通过id对应的属性值获取到UserDaoImpl类的对象,这样我们就不需要用new的方式来创建对象了!

以上java代码还可以写成另一种方式:

public class UserDaoDemo {
public static void main(String[] args) {
ApplicationContext applicationContext =
new ClassPathXmlApplicationContext("applicationContext.xml");
UserDaoImpl userDao = applicationContext.getBean(UserDaoImpl.class);
userDao.save();
}
}

其中,当参数的数据类型是字符串时,表示根据Bean的id从容器中获得Bean实例,返回是Object,需要强转。

当参数的数据类型是Class类型时,表示根据类型从容器中匹配Bean实例,特别注意的是:当容器中相同类型的Bean有多个时,则此方法会报错。

运行结果:

结果正确,到了这里,spring的配置文件你已经有了基本的了解,接下来让我们看看spring配置文件中其它的细节使用吧!

二、Spring配置文件使用

1、Bean标签基本配置

基本介绍

用于配置对象交由Spring 来创建

默认情况下它调用的是类中的无参构造函数,如果没有无参构造函数则不能创建成功,对于有参构造器,后面会慢慢讲到

基本属性

id:Bean实例在Spring容器中的唯一标识
class:Bean的全限定名称

这两个基本属性在快速入门中已经使用过了,就不最多介绍了

Bean标签范围配置(Scope)

修改xml配置文件,在原来的bean中加入scope="singleton",这就意味着,我们不管创建多少个对象,都是同一个

<bean id="userdao" class="com.sht.dao.impl.UserDaoImpl" scope="singleton"></bean>

写一个测试代码

   public void test1(){
ApplicationContext context =
new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userdao1 = (UserDao) context.getBean("userdao");
System.out.println(userdao1);
UserDao userdao2 = (UserDao) context.getBean("userdao");
System.out.println(userdao2);
System.out.println("是否为同一个对象实例:" + (userdao1 == userdao2));
}

运行结果:

继续修改xml配置文件,把scope="singleton"改为scope="prototype",即表明,我们用其创建对象时,不是同一个对象实例

上面的测试代码不变,运行结果:

Bean标签实例化时机

小伙伴们可能就有疑问了,上面这两种方式是什么时候创建对象实例的呢?不要急,咱们慢慢道来。

为了方便效果的展现,我在UserDaoImpl类的无参构造器中加入一句代码,用来区别创建实例的时机

  public UserDaoImpl(){
System.out.println("UserDaoImpl开始创建");
}

scope为singleton时

运行debug,点击F8

我们可以发现,从加载配置文件的时候就开始创建了对象实例

scope为prototype时

运行debug,按下F8,发现控制台并没有打印“UserDaoImpl开始创建”,说明配置文件并没有加载对象实例

再次按F8下一步

我们可以发现控制台终于打印了“UserDaoImpl开始创建”

总结

当scope的取值为singleton时,当应用加载,创建容器时,对象就被创建了;当scope的取值为prototype时,当使用对象时,才创建新的对象实例。

Bean生命周期配置

在UserDaoImpl类中添加两个方法

  public void init(){
System.out.println("初始化方法");
}
public void destory(){
System.out.println("销毁方法");
}

修改xml配置文件,init-method:指定类中的初始化方法名称,destroy-method:指定类中销毁方法名称

<bean id="userdao" class="com.sht.dao.impl.UserDaoImpl"
scope="singleton"
init-method="init"
destroy-method="destory">
</bean>

编写测试代码

public void test3(){
ApplicationContext context =
new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userdao1 = (UserDao) context.getBean("userdao");
System.out.println(userdao1);
UserDao userdao2 = (UserDao) context.getBean("userdao");
System.out.println(userdao2);
System.out.println("是否为同一个对象实例:" + (userdao1 == userdao2));
//这里用来关闭容器
((ClassPathXmlApplicationContext) context).close();
}

运行结果:

从这个例子,相信大家可以自己推测出创建对象、初始化、方法调用以及销毁的先后顺序了

Bean实例化三种方式

方式一:无参构造方法实例化(重点)

其实我们前面一直使用的就是这一种方法,但是需要注意的是,这种方法会根据默认无参构造方法来创建类对象,如果bean中没有默认无参构造函数,将会创建失败

方式二:工厂静态方法实例化(了解)

创建一个静态工厂类,直接返回UserDaoImpl类的对象实例

public class StaticFactory {
public static UserDao getUserDao(){
return new UserDaoImpl();
}
}

编写xml配置文件的bean,这里的class属性值是静态工厂类对应的包的路径,factory-method对应的属性值是类中的方法名

<bean id="userdao" class="com.sht.factory.StaticFactory" factory-method="getUserDao"></bean>

编写代码测试

public class UserDaoDemo {
public static void main(String[] args) {
ApplicationContext applicationContext =
new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userdao = (UserDao) applicationContext.getBean("userdao");
userdao.save();
}
}

最终的运行结果还会是:“保存成功~~”

方式三:工厂实例方法实例化(了解)

创建一个工厂类,同样返回UserDaoImpl类的对象实例

public class DynamicFactory {
public UserDao getUserDao(){
return new UserDaoImpl();
}
}

编写xml配置文件的bean,这里的class属性值是工厂类对应的包的路径,第一个bean可以得到工厂类的对象,再写一个bean,用来获取UserDaoImpl类的对象实例。factory-bean属性值代表从哪个工厂里面拿,factory-method对应的属性值是类中的方法名

<bean id="factory" class="com.sht.factory.DynamicFactory"></bean>
<bean id="userdao" factory-bean="factory" factory-method="getUserDao"></bean>

最终的运行结果也是:“保存成功~~”

总结

2、Bean的依赖注入

基本介绍

依赖注入(Dependency Injection):它是 Spring 框架核心 IOC 的具体实现。

在编写程序时,通过控制反转,把对象的创建交给了 Spring,但是代码中不可能出现没有依赖的情况。IOC 解耦只是降低他们的依赖关系,但不会消除。例如:业务层仍会调用持久层的方法。

那这种业务层和持久层的依赖关系,在使用 Spring 之后,就让 Spring 来维护了。
简单的说,就是坐等框架把持久层对象传入业务层,而不用我们自己去获取。

Bean的依赖注入方式

方式一:有参构造方法

前面讲的都是无参构造方法,现在讲讲有参构造方法。我们先创建一个UserService接口和其实现类UserServiceImpl

public interface UserService {
void save();
}
public class UserServiceImpl implements UserService {
private UserDao userdao;
public UserServiceImpl(UserDao userdao) {
this.userdao = userdao;
}
@Override
public void save() {
userdao.save();
}
}

编写xml配置文件的bean,class属性值分别对应类下的包路径。<constructor-arg name="userdao" ref="userdao"></constructor-arg>中name属性值是相应set方法名的去掉set后的首字母小写,ref是引入引入数据类型,即表示着对象属性的注入

  <bean id="userdao" class="com.sht.dao.impl.UserDaoImpl"></bean>
<bean id="userService" class="com.sht.service.impl.UserServiceImpl">
<constructor-arg name="userdao" ref="userdao"></constructor-arg>
</bean>

方式二:set方法

接口不变,改变一下UserServiceImpl 类,需要注意的是,setUserdao()的权限是public,不能写成private哦

public class UserServiceImpl implements UserService {
private UserDao userdao;
//权限是public
public void setUserdao(UserDao userdao){
this.userdao = userdao;
}
@Override
public void save() {
userdao.save();
}
}

编写xml配置文件的bean,class属性我就不再说明了。重点讲解的是<property name="userdao" ref="userdao"></property>;name属性值是相应set方法名的去掉set后的首字母小写,ref是引入引入数据类型,即表示着对象属性的注入;这里把UserDaoImpl类的对象注入到UserServiceImpl类中,就可以调用其save()方法了

<bean id="userdao" class="com.sht.dao.impl.UserDaoImpl"></bean>
<bean id="userService" class="com.sht.service.impl.UserServiceImpl">
<property name="userdao" ref="userdao"></property>
</bean>
测试代码:
public void test5(){
ApplicationContext context =
new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = (UserService) context.getBean("userService");
userService.save();
}

运行结果:

此外,set方法中有一个特殊的方法:P命名空间注入。其本质也是set方法注入,但比起上述的set方法注入更加方便,主要体现在配置文件中,如下:

引入xmlns:p="http://www.springframework.org/schema/p"名称空间,其次p:userdao-ref属性值是跟ref对应的属性值的含义是一样的,这种方法可以作为了解

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <bean id="userdao" class="com.sht.dao.impl.UserDaoImpl"></bean>
<bean id="userService" class="com.sht.service.impl.UserServiceImpl" p:userdao-ref="userdao"></bean>
</beans>

3、Bean的依赖注入数据类型

普通数据类型的注入

我用set方法的方式,讲解bean的依赖注入。继续在UserDaoImpl类中添加两个基本数据类型和它们的set方法

  private String name;
private Integer age;
public void setName(String name) {
this.name = name;
}
public void setAge(Integer age) {
this.age = age;
}
public void show() {
System.out.println("名字是" + name + "的年龄是" + age);
}

修改xml配置文件的bean,name的属性值代表着set方法的名,value的属性值表示你想要填入的属性值

<bean id="userDao" class="com.sht.dao.impl.UserDaoImpl">
<property name="name" value="卷心菜"></property>
<property name="age" value="20"></property>
</bean>

写一个测试类:

 public void test7(){
ApplicationContext context =
new ClassPathXmlApplicationContext("applicationContext.xml");
UserDaoImpl contextBean = context.getBean(UserDaoImpl.class);
contextBean.show();
}

运行结果:

集合数据类型的注入

List<String>类型的注入

继续在UserDaoImpl类中改动代码:
private List<String> stringList;
public void setStringList(List<String> stringList) {
this.stringList = stringList;
}
public void showStringList() {
System.out.println(stringList);
}

修改xml配置文件,注入基本数据类型就是两个关键字<list>和<value>

 <bean id="userDao" class="com.sht.dao.impl.UserDaoImpl">
<property name="stringList">
<list>
<value>111</value>
<value>222</value>
<value>333</value>
</list>
</property>
</bean>

运行结果:

List<User>类型的注入

先创建一个User类,有对应的get和set方法,还有toString方法,方便测试使用

public class User {
private Integer id;
private String username;
private String password;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
@Override
public String toString() {
return "User{" +
"id=" + id +
", username='" + username + '\'' +
", password='" + password + '\'' +
'}';
}
}

修改xml配置文件的bean,可以看出,与之前不同的是,<list>中用的是<ref>标签,这个标签表示的是注入引用数据类型

<bean id="user01" class="com.sht.domain.User">
<property name="id" value="1"></property>
<property name="username" value="卷心菜"></property>
<property name="password" value="123456"></property>
</bean> <bean id="user02" class="com.sht.domain.User"></bean> <bean id="userDao" class="com.sht.dao.impl.UserDaoImpl">
<property name="userList">
<list>
<ref bean="user01"></ref>
<ref bean="user02"></ref>
</list>
</property>
</bean>

运行结果:

Map<String,User> 类型的注入

继续在UserDaoImpl类中改动代码:

   private Map<String, User> stringUserMap;
public void setStringUserMap(Map<String, User> stringUserMap) {
this.stringUserMap = stringUserMap;
}
public void showStringUserMap() {
System.out.println(stringUserMap);
}

修改xml配置文件,与之前不同的是,当使用的是map时,标签用的是<map>和标签<entry>,使用方法跟list标签差不多

 <bean id="user01" class="com.sht.domain.User">
<property name="id" value="1"></property>
<property name="username" value="卷心菜"></property>
<property name="password" value="123456"></property>
</bean> <bean id="userDao" class="com.sht.dao.impl.UserDaoImpl">
<property name="stringUserMap">
<map>
<entry key="key1" value-ref="user01"></entry>
</map>
</property>
</bean>

运行结果:

需要注意的是,我们在选择key属性时,会有一个如图所示的key-ref的提示,这代表着我们要用的key是一个引入数据类型,因为我用的key是String,所以选择key

当使用value-ref时,也会出现value的值,其用法跟上面一样

Properties类型的注入

继续在UserDaoImpl类中改动代码:

 private Properties properties;
public void setProperties(Properties properties) {
this.properties = properties;
}
public void showProperties(){
System.out.println(properties);
}

修改xml配置文件,与之前的不同之处就是标签使用的是<props>和标签<prop>,使用的方法也是跟上面的方法类似

 <bean id="userDao" class="com.sht.dao.impl.UserDaoImpl">
<property name="properties">
<props>
<prop key="key01">value1</prop>
<prop key="key02">value2</prop>
</props>
</property>
</bean>

运行结果:

4、引入其他配置文件(分模块开发)

创建一个新的配置文件applicationContext1.xml,实际开发中,Spring的配置内容非常多,这就导致Spring配置很繁杂且体积很大,所以,可以将部分配置拆解到其他配置文件中,而在Spring主配置文件通过import标签进行加载

把applicationContext1.xml配置文件中的配置内容引入到applicationContext.xml的实现方式:

<import resource="applicationContext1.xml"></import>

总结

以上就是Spring配置文件的相关知识点,带大家来看看Spring的重点配置:

点击关注,第一时间了解华为云新鲜技术~

不会使用Spring的配置文件,赶紧把这个甩给他的更多相关文章

  1. Spring的配置文件

    Web.xml将会配置Spring的配置文件位置: <servlet>        <servlet-name>x</servlet-name>        & ...

  2. java Spring使用配置文件读取jdbc.properties

    Spring使用配置文件读取jdbc.properties 在beans.xml中加入两个必须的bean [html]<bean id="propertyConfigurer" ...

  3. 使用JDom解析XML文档模拟Spring的配置文件解析

    在J2EE项目中可能会涉及到一些框架的使用,最近接触到了SSH,拿Spring来说配置文件的使用是相当重要的,Spring的配置文件是一个xml文件,Spring是如何读取到配置文件并进行依赖注入的呢 ...

  4. Spring Boot 配置文件详解

    Spring Boot配置文件详解 Spring Boot提供了两种常用的配置文件,分别是properties文件和yml文件.他们的作用都是修改Spring Boot自动配置的默认值.相对于prop ...

  5. Spring boot 配置文件详解 (properties 和yml )

    从其他框架来看 我们都有自己的配置文件, hibernate有hbm,mybatis 有properties, 同样, Spring boot 也有全局配置文件. Springboot使用一个全局的配 ...

  6. Springboot 系列(二)Spring Boot 配置文件

    注意:本 Spring Boot 系列文章基于 Spring Boot 版本 v2.1.1.RELEASE 进行学习分析,版本不同可能会有细微差别. 前言 不管是通过官方提供的方式获取 Spring ...

  7. java web路径和spring读取配置文件

    此篇博客缘起:部署java web系统到阿里云服务器(ubuntu14.04)的时候,有以下两个问题 找不到自定义的property配置文件 上传图片的时候找不到路径 开发的时候是在windows上的 ...

  8. 史上最全的Spring Boot配置文件详解

    Spring Boot在工作中是用到的越来越广泛了,简单方便,有了它,效率提高不知道多少倍.Spring Boot配置文件对Spring Boot来说就是入门和基础,经常会用到,所以写下做个总结以便日 ...

  9. spring boot 配置文件

    spring boot使用一个全局配置文件:主要是以下两种类型 application.properties  :例:server.port=9998 application.yml(YAML)  : ...

  10. Spring之配置文件bean作用域的详细介绍

    Spring的配置文件applicationContext.xml中bean作用域的详细介绍: 1:对象的创建:单例和多例        scope="singleton",默认值 ...

随机推荐

  1. .net core中你的MD5用对了吗?

    本文的项目环境为 .net 6.0 (.net 5.0 以上都支持) 在 .net 中获取字符串的 MD5 相信是非常容易的事情吧, 但是随便在网上搜一搜发现流传的版本还不少呢,比如: StringB ...

  2. 概率期望 DP 题解合集

    期望这东西学了一次忘了,再学一次过了两天又不会了.我是鱼. 故写此博客以便加深记忆及日后复习. NOIP 前恶补期望(? 希望有用,RP++() 经典问题 1 某事件发生概率为 \(p\),则该事件首 ...

  3. Ansible操作MySQL常用的几个模块

    1. mysql_user 模块 mysql_user模块用来添加,删除用户以及设置用户权限 创建MySQL数据库的用户与口令(非root@localhost用户),直接通过playbooks中的案例 ...

  4. UIKit Inside: frame bounds position anchorPoint center

    iOS 中UIView的属性:frame.bounds.center以及CALayer的属性:position.anchorPoint与视图的位置与大小相关,理解这些属性是进行 iOS 视图编码的基础 ...

  5. 文心一言 VS 讯飞星火 VS chatgpt (129)-- 算法导论11.1 4题

    四.用go语言,我们希望在一个非常大的数组上,通过利用直接寻址的方式来实现一个字典.开始时该数组中可能包含一些无用信息,但要对整个数组进行初始化是不太实际的,因为该数组的规模太大.请给出在大数组上实现 ...

  6. 题解 CF637B

    题目大意: 维护个栈,去重保留最上层 题目分析: 啥也不是,数组模拟 \(\text{stack} + \text{unordered\_map}\) 直接秒掉. 复杂度 \(O(n)\) 代码实现: ...

  7. auto关键词

    前言 这个东西在新版本的 C 语言中还是很常用的,尤其是用到 for 循环中.蒟蒻之前也是对这个了解不多,基本没用过(除非迫不得已,因为我的C++编译器太逊了,不支持此操作,用这个操作还得用在线 ID ...

  8. 贪心算法:7-6 Swan学院社团招新

    Swan学院社团招新,招新宣讲会分散在不同时间段,大一新生小花花想知道自己最多能完整的参加多少个招新宣讲会(参加一个招新宣讲会的时候不能中断或离开). [问题说明]这个问题是对几个相互竞争的招新宣讲会 ...

  9. AtCoder F - Parenthesis Checking

    原题链接:AtCoder F - Parenthesis Checking 一个全由\('('\)和\(')'\)构成的字符串,由以下两个操作: 1 l r交换字符串第\(l\)个和第\(r\)个字符 ...

  10. SpringBoot进阶教程(七十八)邮件服务

    Sun公司提供了JavaMail用来实现邮件发送,但是配置烦琐,Spring中提供了JavaMailSender用来简化邮件配置,Spring Boot则提供了MailSenderAutoConfig ...