Spring IOC容器
<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>com.llxazy</groupId> <artifactId>spring_demo</artifactId> <version>1.0-SNAPSHOT</version> <dependencies> <!-- https://mvnrepository.com/artifact/org.springframework/spring-context --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>5.2.3.RELEASE</version> </dependency> </dependencies> </project>
2、编写实体类
Person.java
package com.llxazy.bean; public class Person { private int id; private String name; private int age; private String gender; //此处省略构造方法、get、set方法、toString方法 }
3、注册对象
ioc.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"> <!--注册一个对象,spring会自动创建这个对象--> <!-- 一个bean标签就表示一个对象 id:这个对象的唯一标识 class:注册对象的完全限定名 --> <bean id="person" class="com.llxazy.bean.Person"> <!--使用property标签给对象的属性赋值 name:表示属性的名称 value:表示属性的值 --> <property name="id" value="1001"></property> <property name="name" value="李四"></property> <property name="age" value="28"></property> <property name="gender" value="男"></property> </bean> </beans>
4、测试
SpringTest.java
import com.llxazy.bean.Person; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class SpringTest { public static void main(String[] args) { ApplicationContext context = new ClassPathXmlApplicationContext("ioc.xml"); Person person = (Person) context.getBean("person"); System.out.println(person); } }
总结:
1、ApplicationContext就是IOC容器的接口,可以通过此对象获取容器中创建的对象;
2、对象在Spring容器中默认是在创建完成的时候就已经创建完成,不是需要用的时候才创建,此种情况满足的是单例模式;
3、对象在IOC容器中存储的时候默认都是单例的,如果需要多例需要修改属性;
4、创建对象给属性赋值的时候是通过setter方法实现的;
5、对象的属性是由setter/getter方法决定的,而不是定义的成员属性。
一、spring对象的获取及属性赋值方式:
1、
<!--给person类添加构造方法--> <bean id="person2" class="com.llxazy.bean.Person"> <constructor-arg name="id" value="1"></constructor-arg> <constructor-arg name="name" value="lisi"></constructor-arg> <constructor-arg name="age" value="20"></constructor-arg> <constructor-arg name="gender" value="女"></constructor-arg> </bean> <!--在使用构造器赋值的时候可以省略name属性,但是此时就要求必须严格按照构造器参数的顺序来填写了--> <bean id="person3" class="com.llxazy.bean.Person"> <constructor-arg value="1"></constructor-arg> <constructor-arg value="lisi"></constructor-arg> <constructor-arg value="20"></constructor-arg> <constructor-arg value="女"></constructor-arg> </bean> <!--如果想不按照顺序来添加参数值,那么可以搭配index属性来使用--> <bean id="person4" class="com.llxazy.bean.Person"> <constructor-arg value="lisi" index="1"></constructor-arg> <constructor-arg value="1" index="0"></constructor-arg> <constructor-arg value="女" index="3"></constructor-arg> <constructor-arg value="20" index="2"></constructor-arg> </bean> <!--当有多个参数个数相同,不同类型的构造器的时候,可以通过type来强制类型--> 将person的age类型设置为Integer类型 public Person(int id, String name, Integer age) { this.id = id; this.name = name; this.age = age; System.out.println("Age"); } public Person(int id, String name, String gender) { this.id = id; this.name = name; this.gender = gender; System.out.println("gender"); } <bean id="person5" class="com.llxazy.bean.Person"> <constructor-arg value="1"></constructor-arg> <constructor-arg value="lisi"></constructor-arg> <constructor-arg value="20" type="java.lang.Integer"></constructor-arg> </bean> <!--如果不修改为integer类型,那么需要type跟index组合使用--> <bean id="person5" class="com.llxazy.bean.Person"> <constructor-arg value="1"></constructor-arg> <constructor-arg value="lisi"></constructor-arg> <constructor-arg value="20" type="int" index="2"></constructor-arg> </bean>
4、通过命名空间为bean赋值,简化配置文件中属性声明的写法
导入命名空间
<?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="person6" class="com.mashibing.bean.Person" p:id="3" p:name="wangwu" p:age="22" p:gender="男"></bean>
5、为复杂类型进行赋值操作
给复杂类型赋值,如集合、数组、其他对象等。
实体类:
public class Person { private int id; private String name="李四"; private int age; private String gender; private Address address; private String[] hobbies; private List<Book> books; private Set<Integer> sets; private Map<String,Object> maps; private Properties properties; //此处省略构造方法、get、set方法、toString方法 } public class Book { private String name; private String author; private double price; //此处省略构造方法、get、set方法、toString方法 } public class Address { private String province; private String city; private String town; //此处省略构造方法、get、set方法、toString方法 }
ioc.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" xmlns:p="http://www.springframework.org/schema/p" xmlns:util="http://www.springframework.org/schema/util" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/util https://www.springframework.org/schema/util/spring-util.xsd" > <!--给复杂类型的赋值都在property标签内进行--> <bean id="person" class="com.llxazy.bean.Person"> <property name="name"> <!--赋空值--> <null></null> </property> <!--通过ref引用其他对象,引用外部bean--> <property name="address" ref="address"></property> <!--引用内部bean--> <!-- <property name="address"> <bean class="com.llxazy.bean.Address"> <property name="province" value="北京"></property> <property name="city" value="北京"></property> <property name="town" value="西城区"></property> </bean> </property>--> <!--为list赋值--> <property name="books"> <list> <!--内部bean--> <bean id="book1" class="com.llxazy.bean.Book"> <property name="name" value="水浒传"></property> <property name="author" value="施耐庵"></property> <property name="price" value="100"></property> </bean> <!--外部bean--> <ref bean="book2"></ref> </list> </property> <!--给map赋值--> <property name="maps" ref="myMap"></property> <!--给property赋值--> <property name="properties"> <props> <prop key="aaa">aaa</prop> <prop key="bbb">222</prop> </props> </property> <!--给数组赋值--> <property name="hobbies"> <array> <value>book</value> <value>movie</value> <value>game</value> </array> </property> <!--给set赋值--> <property name="sets"> <set> <value>111</value> <value>222</value> <value>222</value> </set> </property> </bean> <bean id="address" class="com.llxazy.bean.Address"> <property name="province" value="河北"></property> <property name="city" value="邯郸"></property> <property name="town" value="武安"></property> </bean> <bean id="book2" class="com.llxazy.bean.Book"> <property name="name" value="西游记"></property> <property name="author" value="吴承恩"></property> <property name="price" value="200"></property> </bean> <!--级联属性--> <bean id="person2" class="com.llxazy.bean.Person"> <property name="address" ref="address"></property> <property name="address.province" value="北京"></property> </bean> <!--util名称空间创建集合类型的bean--> <util:map id="myMap"> <entry key="key1" value="value1"></entry> <entry key="key2" value-ref="book2"></entry> <entry key="key03"> <bean class="com.llxazy.bean.Book"> <property name="name" value="西游记" ></property> <property name="author" value="吴承恩" ></property> <property name="price" value="100" ></property> </bean> </entry> </util:map> </beans>
6、继承关系bean的配置
ioc.xml
<bean id="person" class="com.llxazy.bean.Person"> <property name="id" value="1"></property> <property name="name" value="zhangsan"></property> <property name="age" value="21"></property> <property name="gender" value="男"></property> </bean> <!--parent:指定bean的配置信息继承于哪个bean--> <bean id="person2" class="com.llxazy.bean.Person" parent="person"> <property name="name" value="lisi"></property>
</bean>
如果想实现Java文件的抽象类,不需要将当前bean实例化的话,可以使用abstract属性
<bean id="person" class="com.llxazy.bean.Person" abstract="true"> <property name="id" value="1"></property> <property name="name" value="zhangsan"></property> <property name="age" value="21"></property> <property name="gender" value="男"></property> </bean> <!--parent:指定bean的配置信息继承于哪个bean--> <bean id="person2" class="com.llxazy.bean.Person" parent="person"> <property name="name" value="lisi"></property> </bean>
7、bean对象创建的依赖关系
bean对象在创建的时候是按照bean在配置文件的顺序决定的,也可以使用depend-on标签来决定顺序
<bean id="book" class="com.llxazy.bean.Book" depends-on="person,address"></bean> <bean id="address" class="com.llxazy.bean.Address"></bean> <bean id="person" class="com.llxazy.bean.Person"></bean>
<!-- bean的作用域:singleton、prototype、request、session 通过scope属性可以指定当前bean的作用域 默认情况下是单例的 prototype:多实例的 容器启动的时候不会创建多实例bean,只有在获取对象的时候才会创建该对象 每次创建都是一个新的对象 singleton:默认的单例对象 在容器启动完成之前就已经创建好对象 获取的所有对象都是同一个 在Spring4.x版本中还包含另外两个作用域 request:每次发送请求都会有一个新的对象 session:每一次会话都会有一个新的对象 注意: 如果是singleton作用域的话,每次在创建IOC容器之前此对象已经创建完成 如果是prototype作用域的话,每次是在需要用到此对象的时候才会创建 --> <bean id="person4" class="com.llxazy.bean.Person" scope="prototype"></bean>
9、利用工厂模式创建bean对象
在利用工厂模式创建bean实例的时候有两种方式,分别是静态工厂和实例工厂。
静态工厂:工厂本身不需要创建对象,但是可以通过静态方法调用,对象=工厂类.静态工厂方法名();
实例工厂:工厂本身需要创建对象,工厂类 工厂对象=new 工厂类;工厂对象.get对象名();
静态工厂:
public class PersonStaticFactory { public static Person getPerson(String name){ Person person = new Person(); person.setId(1); person.setName(name); return person; } } <!-- 静态工厂的使用: class:指定静态工厂类 factory-method:指定哪个方法是工厂方法 --> <bean id="person5" class="com.llxazy.factory.PersonStaticFactory" factory-method="getPerson"> <!--constructor-arg:可以为方法指定参数--> <constructor-arg value="lisi"></constructor-arg> </bean>
动态工厂:
public class PersonInstanceFactory { public Person getPerson(String name){ Person person = new Person(); person.setId(1); person.setName(name); return person; } } <!--实例工厂使用--> <!--创建实例工厂类--> <bean id="personInstanceFactory" class="com.llxazy.factory.PersonInstanceFactory"></bean> <!-- factory-bean:指定使用哪个工厂实例 factory-method:指定使用哪个工厂实例的方法 --> <bean id="person6" class="com.llxazy.bean.Person" factory-bean="personInstanceFactory" factory-method="getPerson"> <constructor-arg value="wangwu"></constructor-arg> </bean>
/** * 实现了FactoryBean接口的类是Spring中可以识别的工厂类,spring会自动调用工厂方法创建实例 * 此方式是Spring创建bean方式的一种补充,用户可以按照需求创建对象,创建的对象交由spring IOC容器 * 进行管理,无论是否是单例,都是在用到的时候才会创建该对象,不用该对象则不会创建 */ public class MyFactoryBean implements FactoryBean<Person> { /** * 工厂方法,返回需要创建的对象 * @return * @throws Exception */ @Override public Person getObject() throws Exception { Person person = new Person(); person.setName("maliu"); return person; } /** * 返回创建对象的类型,spring会自动调用该方法返回对象的类型 * @return */ @Override public Class<?> getObjectType() { return Person.class; } /** * 创建的对象是否是单例对象 * @return */ @Override public boolean isSingleton() { return false; } } ioc.xml <bean id="myfactorybean" class="com.llxazy.factory.MyFactoryBean"></bean>
11、bean对象的初始化和销毁方法
<!--spring容器在创建对象时可以指定具体的初始化和销毁方法 init-method:在对象创建完成之后会调用初始化方法 destory-method:在容器关闭的时候会调用销毁方法 bean生命周期表示bean的创建到销毁 如果bean是单例,容器在启动的时候会创建好,关闭的时候会销毁创建的bean 如果bean是多例,获取的时候创建对象,销毁的时候不会有任何的调用 --> <bean id="address" class="com.llxazy.bean.Address" init-method="init" destroy-method="destory"></bean> //测试类 public class MyTest { public static void main(String[] args) { ApplicationContext context = new ClassPathXmlApplicationContext("ioc2.xml"); Address address = context.getBean("address", Address.class); System.out.println(address); //applicationContext没有close方法,需要使用具体的子类 ((ClassPathXmlApplicationContext)context).close(); } }
12、配置bean对象初始化方法的前后处理方法
spring中包含一个BeanPostProcessor的接口,可以在bean的初始化方法的前后调用该方法,如果配置了初始化方法的前置和后置处理器,无论是否包含初始化方法,都会进行调用
public class MyBeanPostProcessor implements BeanPostProcessor { /** * 在初始化方法调用之前执行 * @param bean 初始化的bean对象 * @param beanName xml配置文件中的bean的id属性 * @return * @throws BeansException */ @Override public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { System.out.println("postProcessBeforeInitialization:"+beanName+"调用初始化前置方法"); return bean; } /** * 在初始化方法调用之后执行 * @param bean * @param beanName * @return * @throws BeansException */ @Override public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException { System.out.println("postProcessAfterInitialization:"+beanName+"调用初始化后缀方法"); return bean; } } //ioc.xml <bean id="myBeanPostProcessor" class="com.llxazy.bean.MyBeanPostProcessor"></bean>
二、
<!-- https://mvnrepository.com/artifact/com.alibaba/druid --> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>1.1.21</version> </dependency> <!-- https://mvnrepository.com/artifact/mysql/mysql-connector-java --> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>5.1.47</version> </dependency>
2、编写配置文件
<?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"> <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"> <property name="username" value="root"></property> <property name="password" value="root"></property> <property name="url" value="jdbc:mysql://localhost:3306/demo"></property> <property name="driverClassName" value="com.mysql.jdbc.Driver"></property> </bean> </beans>
三、spring引用外部配置文件
1、在resource中添加dbconfig.properties
username=root
password=root
url=jdbc:mysql://localhost:3306/demo
driverClassName=com.mysql.jdbc.Driver
2、ioc.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" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"> <!--加载外部配置文件 在加载外部依赖文件的时候需要context命名空间 --> <context:property-placeholder location="classpath:dbconfig.properties"/> <!--在配置文件编写属性的时候需要注意: spring容器在进行启动时,会读取当前系统的某些环境变量的配置 当前系统的用户名是用username表示的,所以最好的方式是添加前缀来区分--> <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"> <property name="username" value="${username}"></property> <property name="password" value="${password}"></property> <property name="url" value="${url}"></property> <property name="driverClassName" value="${driverClassName}"></property> </bean> </beans>
四、spring基于xml文件的自动装配
当一个对象中需要引用另外一个对象的时候,在之前的配置中我们都是通过property标签来进行手动配置的,其实在spring中还提供了一个非常强大的功能就是自动装配(autowire),可以按照我们指定的规则进行配置,配置的方式有以下几种:
default/no:不自动装配
byName:按照名字进行装配,以属性名作为id去容器中查找组件,进行赋值,如果找不到则装配null
byType:按照类型进行装配,以属性的类型作为查找依据去容器中找到这个组件,如果有多个类型相同的bean对象,那么会报异常,如果找不到则装配null
constructor:按照构造器进行装配,先按照有参构造器参数的类型进行装配,没有就直接装配null;如果按照类型找到了多个,那么就使用参数名作为id继续匹配,找到就装配,找不到就装配null
<?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"> <bean id="address" class="com.llxazy.bean.Address"> <property name="province" value="陕西"></property> <property name="city" value="西安"></property> <property name="town" value="蓝田"></property> </bean> <bean id="person" class="com.llxazy.bean.Person" autowire="byName"></bean> <bean id="person2" class="com.llxazy.bean.Person" autowire="byType"></bean> <bean id="person3" class="com.llxazy.bean.Person" autowire="constructor"></bean> </beans>
五、SpEL的使用
SpEL:Spring Expression Language,spring的表达式语言,支持运行时查询操作对象
使用#{...}作为语法规则,所有的大括号中的字符都认为是SpEL.
<bean id="person4" class="com.llxazy.bean.Person"> <!--支持任何运算符--> <property name="age" value="#{12*2}"></property> <!--可以引用其他bean的某个属性值--> <property name="name" value="#{address.province}"></property> <!--引用其他bean--> <property name="address" value="#{address}"></property> <!--调用静态方法--> <property name="hobbies" value="#{T(java.util.UUID).randomUUID().toString().substring(0,4)}"></property> <!--调用非静态方法--> <property name="gender" value="#{address.getCity()}"></property> </bean>