写一个HelloSpring程序:
导包,在父项目中已经完成
新建实体类
package xyz.luck1y.pojo;
public class Hello {
private String str;
public String getStr() {
return str;
}
public void setStr(String str) {
this.str = str;
}
@Override
public String toString() {
return "Hello{" +
"str='" + str + '\'' +
'}';
}
}
xml配置
官方示例:https://www.docs4dev.com/docs/zh/spring-framework/5.1.3.RELEASE/reference/core.html#beans
<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">
<bean id="hello" class="xyz.luck1y.pojo.Hello">
<property name="str" value="Spring"/>
bean>
beans>
测试:
提供给
ApplicationContext
构造函数的位置路径是资源字符串,这些资源字符串使容器可以从各种外部资源(例如本地文件系统,JavaCLASSPATH
等)加载配置元数据。ApplicationContext context = new ClassPathXmlApplicationContext("services.xml", "daos.xml");
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import xyz.luck1y.pojo.Hello;
public class MyTest {
public static void main(String[] args) {
// 获取Spring的上下文对象
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
// 我们的对象现在都在Spring中管理了,我们要使用,直接去里面取出来就可以了
Hello hello = (Hello) context.getBean("hello");
System.out.println(hello);
}
}
这里一开始会给我报错:
Exception in thread "main" org.springframework.beans.factory.NoSuchBeanDefinitionException: No bean named 'hello' available
然后我去网上冲浪了一下,发现我学到这还没有使用注解,但是
注解如果没有指定bean的名字,默认为小写开头的类名。例如类名是ProvincialServiceImpl,则spring返回provincialServiceImpl的bean名。
我的类名这里是大写的,并且在beans.xml文件里我的配置是这样的:
<bean id="Hello" class="xyz.luck1y.pojo.Hello">
<property name="str" value="Spring"/>
bean>
<bean id="hello" class="xyz.luck1y.pojo.Hello">
<property name="str" value="Spring"/>
bean>
再次测试发现没有问题
结果:
思考:
Hello对象是谁创建的?
hello对象是由Spring创建的
Hello对象的属性是怎么设置的?
hello对象的属性是由Spring容器设置的
这个过程就叫做控制反转:
new ClassPathXmlApplicationContext
来浏览一下底层源码到了现在,我们彻底不用再程序中去改动了,要实现不同的操作,只需要在xml配置文件中进行修,所谓的IOC,就是:对象由Spring来创建,管理,装配!
如果实体类在XML中配置过了,那么实体类就会有标志:
下面,我们改造一下前面推导IOC的程序:
配置xml:
<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">
<bean id="mysqlImpl" class="xyz.luck1y.dao.UserDaoMySQLImpl"/>
<bean id="userServiceImpl" class="xyz.luck1y.service.UserServiceImpl">
<property name="userDao" ref="mysqlImpl"/>
bean>
beans>
测试类
import org.springframework.context.support.ClassPathXmlApplicationContext;
import xyz.luck1y.service.UserServiceImpl;
public class MyTest {
public static void main(String[] args) {
// 获取ApplicationContext;拿到Spring的容器
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
// 容器在手,天下我有,需要什么就直接get什么
UserServiceImpl userServiceImpl = (UserServiceImpl) context.getBean("userServiceImpl");
userServiceImpl.getUser();
}
}
新建实体类,并且在无参构造里输出一句话,表示调用了无参构造
package xyz.luck1y.pojo;
public class User {
private String name;
public User() {
System.out.println("User的无参构造");
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public void show(){
System.out.println("name=" + name);
}
}
配置beans.xml
<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">
<bean id="user" class="xyz.luck1y.pojo.User">
<property name="name" value="刘子"/>
bean>
beans>
测试:
import org.springframework.context.support.ClassPathXmlApplicationContext;
import xyz.luck1y.pojo.User;
public class MyTest {
public static void main(String[] args) {
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
User user = (User) context.getBean("user");
user.show();
}
}
结果
这说明,getBean方法调用了无参构造,也就是说在getBean之前,对象已经创建好了。
其实对象的实例化在获取上下文的时候就创建好了
接下来我们看看去掉无参构造,即添加有参构造的情况。
添加有参构造方法
package xyz.luck1y.pojo;
public class User {
private String name;
// 我们不写无参构造也会默认存在的,要写有参构造来顶掉无参构造
public User(String name) {
this.name = name;
}
// public User() {
// System.out.println("User的无参构造");
// }
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public void show(){
System.out.println("name=" + name);
}
}
测试:
运行前发现beans.xml已经爆红
运行:初始化出错
那我们又该如何使用有参构造方法呢?
第一种方式:使用下标赋值
<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">
<bean id="user" class="xyz.luck1y.pojo.User">
<constructor-arg index="0" value="刘子"/>
bean>
beans>
第二种方式:通过参数类型匹配
不建议使用:因为如果出现两个同类型的参数会出现混淆,会按照参数出现的先后顺序来赋值。
<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">
<bean id="user" class="xyz.luck1y.pojo.User">
<constructor-arg type="java.lang.String" value="刘子"/>
bean>
beans>
第三种方式:通过参数名匹配
<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">
<bean id="user" class="xyz.luck1y.pojo.User">
<constructor-arg name="name" value="刘子"/>
bean>
beans>
此外在配置文件加载的时候,Spring容器中管理的对象,即所有的bean都被实例化了,且每个bean在内存中只有一份实例,比如我们getBean同一个id,然后进行比较,发现是true
Spring标签比MyBatis的标签要少很多。
别名:
<alias name="user" alias="userTest"/>
name对应的是bean的id,alias是给这个id起的别名,别名区分大小写
在测试类中getBean方法通过getBean(“别名”)
还是可以获取到对象的
<bean id="user" class="xyz.luck1y.pojo.User" name="user2 user3,user4;user5">
<constructor-arg name="name" value="刘子"/>
bean>
id:bean的唯一标识符,也就是相当于对象名
class:bean对象所对应的全限定名(类名+包名)
name:也是别名,name可以同时取多个别名,可以通过逗号/空格/分号来分割
这里有一个高频问点:Bean的生命周期,可以自行搜索查找
Bean 的生命周期概括起来就是 4 个阶段:
- 实例化(Instantiation)
- 属性赋值(Populate)
- 初始化(Initialization)
- 销毁(Destruction)
一般用于团队开发,可以将多个配置文件,导入合并为一个。
假如有几个人写的不同的配置文件
在测试时,扫描汇总后的xml、总的xml配置文件applicationContext.xml
import org.springframework.context.support.ClassPathXmlApplicationContext;
import xyz.luck1y.pojo.User;
public class MyTest {
public static void main(String[] args) {
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
User user = (User) context.getBean("user2");
user.show();
}
}
这时就可以在总的xml配置文件中使用import标签:
<import resource="beans.xml"/>
<import resource="zhangsan.xml"/>
<import resource="lisi.xml"/>
名字重复会报错
导入之后,其他xml中的bean,总的xml中也就可以拿到了。
import 使用的时候:
- 与主配置的id重名,使用主配置的id
- 多个import中配置的id重名(主配置中没有),调用最后import中配置的id,即后面的覆盖前面的
- 调用别名,哪个配置有这个别名,使用哪个配置
相同id,相同内容 —> 进行合并;
相同id,不同内容 —> 后来居上,进行覆盖。
以上配置算是初级入门,深入的部分还不止这些~