JavaEE基础01-spring
Posted 旋转的格子
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了JavaEE基础01-spring相关的知识,希望对你有一定的参考价值。
本章内容的学习与研究都基于spring4
可以到spring官网去下载最新的spring扎包。
现在作一个简单测试类,我使用的是idea,创建module时,勾选spring,写一个applicationContext.xml文件。
.xml文件里加入
<bean id="helloWorld" class="com.guigu.spring.beans.HelloWorld"> <property name="name" value="spring"></property>
</bean>
把.xml文件和spring建立联系。
package com.guigu.spring.beans; /** * Created by Zhuxiang on 2020/5/9. */ public class HelloWorld { public String name; public HelloWorld() { System.out.println("HelloWorld is construct!"); } public void setName(String name) { this.name = name; } @Override public String toString() { return "HelloWorld{" + "name=\'" + name + \'\\\'\' + \'}\'; } }
package com.guigu.spring.beans; import org.springframework.context.support.ClassPathXmlApplicationContext; /** * Created by Zhuxiang on 2020/5/9. */ public class Main { public static void main(String[] args) { // HelloWorld helloWorld = new HelloWorld(); // helloWorld.setName("123"); ClassPathXmlApplicationContext c = new ClassPathXmlApplicationContext("applicationContext.xml"); HelloWorld helloWorld = (HelloWorld) c.getBean("helloWorld"); System.out.println(helloWorld); } }
运行之后成功。
IOC 前生 --- 分离接口与实现
IOC 前生 --- 采用工厂设计模式
IOC --- 采用反转控制
上面的例子就是用的ClassPathXmlApplicationContext类。
ApplicationContext 的主要实现类:
ClassPathXmlApplicationContext:从 类路径下加载配置文件
FileSystemXmlApplicationContext: 从文件系统中加载配置文件
ConfigurableApplicationContext 扩展于 ApplicationContext,新增加两个主要方法:refresh() 和 close(), 让 ApplicationContext 具有启动、刷新和关闭上下文的能力
ApplicationContext 在初始化上下文时就实例化所有单例的 Bean。 WebApplicationContext 是专门为 WEB 应用而准备的,它允许从相对于 WEB 根目录的路径中完成初始化工作
依赖注入的方式
Spring 支持 3 种依赖注入的方式
属性注入
构造器注入
工厂方法注入(很少使用,不推荐)
属性注入
属性注入即通过 setter 方法注入Bean 的属性值或依赖的对象(上面.xml中的<property name="name" value="spring">"name"就是指bean里的setter方法的名字)
属性注入使用 <property> 元素, 使用 name 属性指定 Bean 的属性名称,value 属性或 <value> 子节点指定属性值
属性注入是实际应用中最常用的注入方式
构造器注入
通过构造方法注入Bean 的属性值或依赖的对象,它保证了 Bean 实例在实例化后就可以使用。
构造器注入在 <constructor-arg> 元素里声明属性, <constructor-arg> 中没有 name 属性
按索引匹配入参:
按类型匹配入参:
注入属性值的细节
字面值:可用字符串表示的值,可以通过 <value> 元素标签或 value 属性进行注入。例子:<property name="name" value="spring">中的"spring"
基本数据类型及其封装类、String 等类型都可以采取字面值注入的方式。例子:<constructor-arg value="奥迪" type="java.lang.String"/>
若字面值中包含特殊字符,可以使用 <![CDATA[]]> 把字面值包裹起来。例子:把这样一个值显示出来<spring^^>,<property name="name" ><value><![CDATA[<spring^^>]]></value></property>
引用其它 Bean
组成应用程序的 Bean 经常需要相互协作以完成应用程序的功能. 要使 Bean 能够相互访问, 就必须在 Bean 配置文件中指定对 Bean 的引用
在 Bean 的配置文件中, 可以通过 <ref> 元素或 ref 属性为 Bean 的属性或构造器参数指定对 Bean 的引用.
也可以在属性或构造器里包含 Bean 的声明, 这样的 Bean 称为内部 Bean
<!-- 配置 bean --> <bean id="dao5" class="com.guigu.spring.ref.Dao"> <property name="dataSource" value="data"></property> </bean> <bean id="service" class="com.guigu.spring.ref.Service"> <!-- 通过 ref 属性值指定当前属性指向哪一个 bean! --> <property name="dao" ref="dao5"></property> </bean> <!--以下为内部引用--> <!-- <!–内部bean–>--> <!-- <bean id="service" class="com.guigu.spring.ref.Service">--> <!-- <property name="dao">--> <!-- <bean class="com.guigu.spring.ref.Dao">--> <!-- <property name="dataSource" value="内部beandata"></property>--> <!-- </bean>--> <!-- </property>--> <!-- </bean>-->
注入参数详解:null 值和级联属性
可以使用专用的 <null/> 元素标签为 Bean 的字符串或其它对象类型的属性注入 null 值(不写例子了,用的也少)
和 Struts、Hiberante 等框架一样,Spring 支持级联属性的配置。
<bean id="service2" class="com.guigu.spring.ref.Service"> <!-- <property name="dao" ref="dao5"></property>--><!--用属性注入和下面用构造器同效果--> <constructor-arg ref="dao5"></constructor-arg><!--必须先关联非空的bean,属性需要先初始化,才可以这样赋值。不然会报错。同时记得在本bean(Service)里写空参构造器--> <property name="dao.dataSource" value="service为级联属性dao.dataSource赋值"></property> </bean>
public class Dao {
private String dataSource;
public void setDataSource(String dataSource) {
this.dataSource = dataSource;
}
public void save(){
System.out.println("save by " + dataSource);
}
public Dao() {
System.out.println("Dao\'s Constructor...");
}
}
public class Service {
private Dao dao;
public Service(Dao dao) {
this.dao = dao;
}
public Service() {
}
public void setDao(Dao dao) {
this.dao = dao;
}
public Dao getDao() {
return dao;
}
public void save(){
System.out.println("Service\'s save");
dao.save();
}
}
public class Main {
public static void main(String[] args) {
// HelloWorld helloWorld = new HelloWorld();
// helloWorld.setName("123");
ClassPathXmlApplicationContext c = new ClassPathXmlApplicationContext("applicationContext.xml");
HelloWorld helloWorld = (HelloWorld) c.getBean("helloWorld");
System.out.println(helloWorld);
Service service = (Service) c.getBean("service");
service.save();
Service service2 = (Service) c.getBean("service2");
service2.save();
}
}
集合属性
在 Spring中可以通过一组内置的 xml 标签(例如: <list>, <set> 或 <map>) 来配置集合属性.
配置 java.util.List 类型的属性, 需要指定 <list> 标签, 在标签里包含一些元素. 这些标签可以通过 <value> 指定简单的常量值, 通过 <ref> 指定对其他 Bean 的引用. 通过<bean> 指定内置 Bean 定义. 通过 <null/> 指定空元素. 甚至可以内嵌其他集合.
数组的定义和 List 一样, 都使用 <list>
配置 java.util.Set 需要使用 <set> 标签, 定义元素的方法与 List 一样.
<!--集合--> <bean id="car1" class="com.guigu.spring.beans.Car"> <property name="name" value="car1"></property> <property name="price" value="1000"></property> </bean> <bean id="person1" class="com.guigu.spring.beans.Person"> <property name="name" value="p1"></property> <property name="cars"> <list> <!--引入bean--> <ref bean="car1"/> <!--内部bean--> <bean class="com.guigu.spring.beans.Car"> <constructor-arg value="car2" type="java.lang.String"></constructor-arg> <constructor-arg value="2000" type="java.lang.Integer"></constructor-arg> </bean> </list> </property> </bean>
补充:不写class了,应该可以直接看出来。person类属性有String name和list<car> cars。car类有name和price。main方法不多说了。
Java.util.Map 通过 <map> 标签定义, <map> 标签里可以使用多个 <entry> 作为子标签. 每个条目包含一个键和一个值.
必须在 <key> 标签里定义键
因为键和值的类型没有限制, 所以可以自由地为它们指定 <value>, <ref>, <bean> 或 <null> 元素.
可以将 Map 的键和值作为 <entry> 的属性定义: 简单常量使用 key 和 value 来定义; Bean 引用通过 key-ref 和 value-ref 属性定义
使用 <props> 定义 java.util.Properties, 该标签使用多个 <prop> 作为子标签. 每个 <prop> 标签必须定义 key 属性.
<!--Map集合--> <bean id="person2" class="com.guigu.spring.beans.Person"> <property name="name" value="p1"></property> <property name="mcars"> <map> <!--引入bean--> <entry key="AA" value-ref="car1"/> <!--内部bean--> <entry key="BB" > <bean class="com.guigu.spring.beans.Car"> <constructor-arg value="car2" type="java.lang.String"></constructor-arg> <constructor-arg value="2000" type="java.lang.Integer"></constructor-arg> </bean> </entry> </map> </property> </bean>
<!--properties--> <bean id="dataSource" class="com.guigu.spring.beans.DataSource"> <property name="pro"> <props> <prop key="user">root</prop> <prop key="password"></prop> <prop key="jdbcUrl">jdbc:mysql:///test</prop> <prop key="driverClass">com.mysql.jdbc.Driver</prop> </props> </property> </bean>
使用 utility scheme 定义集合
使用基本的集合标签定义集合时, 不能将集合作为独立的 Bean 定义, 导致其他 Bean 无法引用该集合, 所以无法在不同 Bean 之间共享集合.
可以使用 util schema 里的集合标签定义独立的集合 Bean. 需要注意的是, 必须在 <beans> 根元素里添加 util schema 定义
<!--配置单例的集合bean,以供多个bean进行引用--> <util:list id="cars"> <ref bean="car1"></ref> <ref bean="car1"></ref> <bean class="com.guigu.spring.beans.Car"> <constructor-arg value="car2" type="java.lang.String"></constructor-arg> <constructor-arg value="2000" type="java.lang.Integer"></constructor-arg> </bean> </util:list> <bean id="person3" class="com.guigu.spring.beans.Person"> <property name="name" value="p1"></property> <property name="cars" ref="cars"></property> </bean>
XML 配置里的 Bean 自动装配(开发很少使用,但整合第三方框架时会用到,需要了解)
Spring IOC 容器可以自动装配 Bean. 需要做的仅仅是在 <bean> 的 autowire 属性里指定自动装配的模式
byType(根据类型自动装配): 若 IOC 容器中有多个与目标 Bean 类型一致的 Bean. 在这种情况下,Spring 将无法判定哪个 Bean 最合适该属性, 所以不能执行自动装配.
byName(根据名称自动装配): 必须将目标 Bean 的名称和属性名设置的完全相同.
<!--autowire byName 根据bean的名字和当前bean的setter风格的属性名进行自动装配,有匹配则装配,没就不装配。 byType 根据bean的类型和当前bean的属性的类型进行自动装配,若ioc容器中有一个以上的类型匹配的bean,就抛异常。 --> <bean id="person4" class="com.guigu.spring.beans.Person" autowire="byName"> <property name="name" value="p1"></property> <!--因为cars属性的setter方法名为cars,所以自动寻找id=cars的bean来装配--> <!--<property name="cars" ref="cars"></property>--> </bean>
缺点:
在 Bean 配置文件里设置 autowire 属性进行自动装配将会装配 Bean 的所有属性. 然而, 若只希望装配个别属性时, autowire 属性就不够灵活了.
autowire 属性要么根据类型自动装配, 要么根据名称自动装配, 不能两者兼而有之.
一般情况下,在实际的项目中很少使用自动装配功能,因为和自动装配功能所带来的好处比起来,明确清晰的配置文档更有说服力一些
继承 Bean 配置
Spring 允许继承 bean 的配置, 被继承的 bean 称为父 bean. 继承这个父 Bean 的 Bean 称为子 Bean
子 Bean从父 Bean 中继承配置, 包括 Bean 的属性配置
子 Bean 也可以覆盖从父 Bean 继承过来的配置
父 Bean可以作为配置模板, 也可以作为 Bean 实例. 若只想把父 Bean 作为模板, 可以设置 <bean> 的abstract 属性为 true, 这样 Spring 将不会实例化这个 Bean
并不是 <bean> 元素里的所有属性都会被继承. 比如: autowire, abstract 等. 也可以忽略父 Bean 的 class 属性, 让子 Bean 指定自己的类, 而共享相同的属性配置. 但此时 abstract 必须设为 true
<!--bean之间的关系--> <bean id="car1" class="com.guigu.spring.beans.Car"> <property name="name" value="大众"></property> <property name="price" value="10000"></property> </bean> <bean id="car2" class="com.guigu.spring.beans.Car" parent="car1"> <property name="price" value="20000"></property> </bean>
Car{name=\'大众\', price=10000}
Car{name=\'大众\', price=20000}
把一个bean定为配置模板。
<bean id="car1" abstract="true">
抽象bean不能被实例化(在ioc容器里)。main方法里不再能实例化car1(运行时会抛异常)。
如果bean的class属性没有指定,则该bean必须是一个抽象bean。
Car{name=\'大众\', price=20000}
依赖 Bean 配置
Spring 允许用户通过 depends-on 属性设定 Bean 前置依赖的Bean,前置依赖的 Bean 会在本 Bean 实例化之前创建好
如果前置依赖于多个 Bean,则可以通过逗号,空格或的方式配置 Bean 的名称
<!--要求配置person时,必须有一个关联的car。换句话说person的bean必须依赖car这个bean 如果没有这个bean那么初始化时就报错--> <bean id="person" class="com.guigu.spring.beans.Person" depends-on="car2"> <property name="name" value="p1"></property> </bean>
只是依赖关系,不会赋值。
使用singleton,main方法里实例化car01和car02,car01=car02;
使用prototype,main方法里实例化car01和car02,car01!=car02;
在 Spring 中, 可以在 <bean> 元素的 scope 属性里设置 Bean 的作用域.
默认情况下, Spring 只为每个在 IOC 容器里声明的 Bean 创建唯一一个实例, 整个 IOC 容器范围内都能共享该实例:所有后续的 getBean() 调用和 Bean 引用都将返回这个唯一的 Bean 实例.该作用域被称为 singleton, 它是所有 Bean 的默认作用域.
例子:创建连接池连接数据库。
<!--连接数据库--> <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"> <property name="username" value="root"></property> <property name="password" value=""></property> <property name="driverClassName" value="com.mysql.jdbc.Driver"></property> <property name="url" value="jdbc:mysql://localhost:3306/girls?rewriteBatchedStatements=true"></property> </bean>
main方法里
这样可以连接数据库,但是不利于修改和维护。
在配置文件里配置 Bean 时, 有时需要在 Bean 的配置里混入系统部署的细节信息(例如: 文件路径, 数据源配置信息等). 而这些部署细节实际上需要和 Bean 配置相分离
Spring 提供了一个 PropertyPlaceholderConfigurer 的 BeanFactory 后置处理器, 这个处理器允许用户将 Bean 配置的部分内容外移到属性文件中. 可以在 Bean 配置文件里使用形式为 ${var} 的变量, PropertyPlaceholderConfigurer 从属性文件里加载属性, 并使用这些属性来替换变量.
Spring 还允许在属性文件中使用 ${propName},以实现属性之间的相互引用。
具体方法:
Spring 2.0:已经不建议使用了。
Spring 2.5 之后: 可通过 <context:property-placeholder> 元素简化: <beans> 中添加 context Schema 定义 在配置文件中加入如下配置:
在beans里做配置,虽然直接写在idea里会自动导入,不过还是要了解一下。补充一下,p标签是用来在bean里赋值的,简化代码。但是我这里没有使用。
<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:context="http://www.springframework.org/schema/context">
<!--连接数据库--> <!-- <context:property-placeholder location="classpath:druid.properties"/>--> <bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"> <property name="location" value="classpath:druid.properties"></property> </bean> <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"> <property name="username" value="${username}"></property> <property name="password" value="${password}"></property> <property name="driverClassName" value="${driverClassName}"></property> <property name="url" value="${url}"></property> </bean>
我不知道是什么地方出了问题,反正我写<context:property-placeholder>没有用,要么是mysql的版本要么是druid数据连接池包不兼容的问题。反正我代码是没问题。
Spring表达式语言:SpEL
语法类似于 EL:SpEL 使用 #{…} 作为定界符,所有在大框号中的字符都将被认为是 SpEL
字面量的表示:
整数:<property name="count" value="#{5}"/>
小数:<property name="frequency" value="#{89.7}"/>
科学计数法:<property name="capacity" value="#{1e4}"/>
String可以使用单引号或者双引号作为字符串的定界符号:<property name=“name” value="#{\'Chuck\'}"/> 或 <property name=\'name\' value=\'#{"Chuck"}\'/>
Boolean:<property name="enabled" value="#{false}"/>
SpEL:引用 Bean、属性和方法(1)
上面的if-else运算符真的没用,其实就是三元运算符而已了,像ppt里那样写直接报错啊。
注意:下面例子里引用的是car3的价格,不能引用自己bean的。
<!--spEl--> <bean id="car4" class="com.guigu.spring.beans.Car" > <property name="name" value="oooo"></property> <property name="price" value="9999"></property> <property name="info" value="#{car3.price>1000?\'豪车\':\'代步车\'}"></property> </bean>
public final class Math {//java.lang.math的源码
public static final double PI = 3.14159265358979323846;
Spring IOC 容器可以管理 Bean 的
以上是关于JavaEE基础01-spring的主要内容,如果未能解决你的问题,请参考以下文章