1.Spring作用:
1.生态体系庞大,全能型选手!【springmvc是其一个子模块,jdbcTemplate能直接操作数据库!】
2.将其他组件粘合在一起
比如将SpringMVC和Mybaits连在一起
3.包括:IOC容器和AOP【面向切面编程】
Spring的IOC机制(控制反转和依赖注入)正是用在此处。
Spring的IOC(控制反转和依赖注入)
控制反转[IOC]:就是由容器控制程序之间的(依赖)关系,而非传统实现中,由程序代码直接操控。
控制反转是一种思想,其具体实现就是依赖注入!
1.生态体系庞大,全能型选手!【springmvc是其一个子模块,jdbcTemplate能直接操作数据库!】
2.将其他组件粘合在一起
比如将SpringMVC和Mybaits连在一起
3.包括:IOC容器和AOP【面向切面编程】
Spring的IOC机制(控制反转和依赖注入)正是用在此处。
Spring的IOC(控制反转和依赖注入)
控制反转[IOC]:就是由容器控制程序之间的(依赖)关系,而非传统实现中,由程序代码直接操控。
控制反转是一种思想,其具体实现就是依赖注入!
依赖注入[DI:Dependency Injection]:组件之间的依赖关系由容器在运行期决定 ,由容器动态的将某种依赖关系注入到组件之中。
比如在controller层要调用service层方法 不需要new一个对象,用注解的形式,将对象放入容器里,进行调用,这就用到了依赖注入
2.IOC容器细节
1.使用IOC容器创建对象
2.使用IOC容器在创建对象的同时,给对象的属性赋值
1.使用IOC容器创建对象
2.使用IOC容器在创建对象的同时,给对象的属性赋值
3.在创建对象的过程中,实现组件对象之间的自动装配
3.搭建Spring IOC容器需要的开发环境
1.导入IOC容器需要的jar包
spring-beans-4.0.0.RELEASE.jar
spring-context-4.0.0.RELEASE.jar
1.导入IOC容器需要的jar包
spring-beans-4.0.0.RELEASE.jar
spring-context-4.0.0.RELEASE.jar
spring-core-4.0.0.RELEASE.jar
spring-expression-4.0.0.RELEASE.jar
commons-logging-1.1.3.jar
2.创建Spring配置文件[Spring bean Configuration File]
实验1:通过IOC容器创建对象,并为属性赋值
<!-- 需要由IOC容器创建对象的全类名 -->
<property>是通过 set 方法注入的,所以实体类必须有set方法
<!-- class指定类对象的全类名,交给服务器创建一个对象 id是一个唯一标识,在IOC只能出现一个id为book的bean对象 --> <bean id="book" class="com.neuedu.spring.bean.Book"> <property name="bookName" value="JAVA"></property> <property name="author" value="you"></property> <property name="price" value="123.123"></property> </bean>
public class TestIOC { @Test public void test() { //1.获取IOC容器本身这个对象 ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml"); //2.从IOC容器中获取bean对象 Object bean = ioc.getBean("book"); System.out.println(bean); } }
注意:
①IOC容器本身对象创建时,会将配置文件中配置好的bean先创建出来
②默认是单实例的,只创建bean的一个对象
③如果设置bean的scope属性为prototype,那么创建bean的对象就是多实例的,在获取的时候创建,每次获取对象都会创建新的
④.从IOC容器中获取对象
①根据bean的id获取
①IOC容器本身对象创建时,会将配置文件中配置好的bean先创建出来
②默认是单实例的,只创建bean的一个对象
③如果设置bean的scope属性为prototype,那么创建bean的对象就是多实例的,在获取的时候创建,每次获取对象都会创建新的
④.从IOC容器中获取对象
①根据bean的id获取
②根据bean的类型获取:要求容器中指定类型的bean是唯一的
bean的后置处理器:
1.指的是在bean的初始化方法前后执行操作的专门的对象。
2.自定义的后置处理器:
1) 需要实现接口:org.springframework.beans.factory.config.BeanPostProcessor .
2) 做相应的配置就好!
1.指的是在bean的初始化方法前后执行操作的专门的对象。
2.自定义的后置处理器:
1) 需要实现接口:org.springframework.beans.factory.config.BeanPostProcessor .
2) 做相应的配置就好!
实验2:根据bean的类型从IOC容器中获取bean的实例
public class TestIOC { @Test public void test() { //1.获取IOC容器本身这个对象 ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml"); //2.从IOC容器中获取bean对象 Book bean = ioc.getBean(Book.class); System.out.println(bean.getBookName()+"---"+bean.getAuthor()+"---"+bean.getPrice()); } }
要注意,创建实体类的时候要有无参构造器,没有无参的话会默认有一个无参构造器,但是这时要是有一个有参构造器,就构建不成对象,获取不到bean实例。
实验3:通过构造器为bean的属性赋值、通过index属性指定参数的位置、通过类型不同区分重载的构造器
通过<constructor-arg> 给构造器赋值,通过index给属性排序,让该属性与有参构造器的顺序一直,index 是从 0 开始的,type 指定 你输入的值的类型,进而将该值付给对应的实体类的属性中,比如“22.22”会付给double 修饰的属性,而不是其他属性
<bean id="book01" class="com.neuedu.spring.bean.Book"> <constructor-arg value="you" index="1"></constructor-arg> <constructor-arg value="C++" index="0"></constructor-arg> <constructor-arg value="22.22" index="2" type="double"></constructor-arg> </bean>
实验4:给bean的级联属性赋值
创建三个实体类,第一个类包含第二个类,第二个类包含第三个类,第三个类有一个 username 属性
分别 get、set、toString 方法
public class StudentController { private StudentService studentService; public StudentService getStudentService() { return studentService; } public void setStudentService(StudentService studentService) { this.studentService = studentService; } @Override public String toString() { return "StudentController [studentService=" + studentService + "]"; } }
public class StudentService { private StudentDao studentDao; public StudentDao getStudentDao() { return studentDao; } public void setStudentDao(StudentDao studentDao) { this.studentDao = studentDao; } @Override public String toString() { return "StudentService [studentDao=" + studentDao + "]"; } }
public class StudentDao { private String username; public String getUsername() { return username; } public void setUsername(String username) { this.username = username; } @Override public String toString() { return "StudentDao [username=" + username + "]"; } }
value 只能给字符串赋值,ref 指向对象 ,与该对象的 id 相配
最后页面调用 studentController
<!-- 给bean的级联属性赋值 --> <bean id="studentDao" class="com.neuedu.spring.bean.StudentDao"></bean> <bean id="studentService" class="com.neuedu.spring.bean.StudentService"></bean> <bean id="studentController" class="com.neuedu.spring.bean.StudentController"> <property name="studentService" ref="studentService"></property> <property name="studentService.studentDao" ref="studentDao"></property> <property name="studentService.studentDao.username" value="zhangsan"></property> </bean>
public class TestIOC { @Test public void test() { //1.获取IOC容器本身这个对象 ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml"); //2.从IOC容器中获取bean对象 Object bean = ioc.getBean("studentController"); System.out.println(bean); } }
实验5:通过p名称空间为bean赋值
在spring.xml 文件中的namespace 中选中 p 标签
<!-- 通过p名称空间为bean赋值 --> <bean id="student" class="com.neuedu.spring.bean.Student" p:name="zhangsan" p:gender="1" p:address="China" > </bean>
实验6:测试bean的作用域,分别创建单实例和多实例的bean
在两个实体类中的无参构造器中分别输入一句话,判断对象创建几次
通过 scope 控制单例还是多例:singleton是单例;prototype是多例
单例 是在IOC容器创建的时候创建
多例 是每一次获取bean对象的时候创建
<!-- 测试bean的作用域,分别创建单实例和多实例的bean --> <bean id="book2" scope="singleton" class="com.neuedu.spring.bean.Book"></bean> <bean id="student2" scope="prototype" class="com.neuedu.spring.bean.Student"></bean>
显示 一个book ,三个 student
public class TestIOC { //1.获取IOC容器本身这个对象 private ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml"); @Test public void test() { ioc.getBean("book"); ioc.getBean("book"); ioc.getBean("book"); ioc.getBean("student"); ioc.getBean("student"); ioc.getBean("student"); } }
实验7:创建带有生命周期方法的bean
在两个实体类中的无参构造器中分别输入一句话,判断对象是否被创建
<bean id="book" class="com.neuedu.spring.bean.Book"></bean> <bean id="student" class="com.neuedu.spring.bean.Student"></bean>
test() 中啥都不写,运行之后发现两个对象都被创建了
说明当 ioc 容器被创建时,对象就被创建了
public class TestIOC { private ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml"); @Test public void test() { } }
实验8:创建带有生命周期方法的bean
在实体类中创建两个方法:init()和destroy()
分别输入“init方法执行”、“destroy方法执行”
在 bean 中 添加 init-method 和 destroy-method
分别对应 init 方法和 destroy 方法
<bean id="teacher" class="com.neuedu.spring.bean.Teacher" init-method="teacherinit" destroy-method="destroy"></bean>
但是调用不了 destroy 方法,通过 ApplicationContext 的子类的 close 方法调用 destroy 方法
public class TestIOC { private ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml"); @Test public void test() { ConfigurableApplicationContext cac = (ConfigurableApplicationContext) ioc; ioc.getBean("teacher"); cac.close(); } }
实验9:bean之间的依赖 depends-on="order"被依赖的对象会先创建
如果没有depends-on 会按照 bean 顺序创建对象
但是有了之后,被依赖的先创建,本例中 book 依赖 student ,所以 student 会先创建
<bean id="book" class="com.neuedu.spring.bean.Book" depends-on="student"></bean> <bean id="student" class="com.neuedu.spring.bean.Student" ></bean> <bean id="teacher" class="com.neuedu.spring.bean.Teacher" init-method="teacherinit" destroy-method="destroy"></bean>
实验10:通过继承实现bean配置信息的重用
通过 parent 进行继承
book1 继承 book 的 bookName 和 author 其余的两个进行了修改
输出 book1 会输出
Book [bookName=JAVA, author=you, price=34.23 , isbn=house]
<bean id="book" class="com.neuedu.spring.bean.Book"> <property name="bookName" value="JAVA"></property> <property name="author" value="you"></property> <property name="price" value="43.43"></property> <property name="isbn" value="home"></property> </bean> <bean id="book1" class="com.neuedu.spring.bean.Book" parent="book"> <property name="price" value="34.23"></property> <property name="isbn" value="house"></property> </bean>
实验11:通过abstract属性创建一个模板bean
通过 abstract 使 book 对象抽象化,这样就不能调用 book 对象
但是可以调用子对象 book1
<bean id="book" class="com.neuedu.spring.bean.Book" abstract="true"> <property name="bookName" value="JAVA"></property> <property name="author" value="you"></property> <property name="price" value="43.43"></property> <property name="isbn" value="home"></property> </bean> <bean id="book1" parent="book"> <property name="price" value="43.43"></property> <property name="isbn" value="home"></property> </bean>
实验12:测试使用null值
book 属性少了 author 所以会显示 null
<bean id="book" class="com.neuedu.spring.bean.Book"> <property name="bookName" value="java"></property> <property name="price" value="33.5"></property> <property name="isbn" value="house"></property> </bean>
也可以这么写,将<null/> 写进<property> 中
<bean id="book" class="com.neuedu.spring.bean.Book"> <property name="bookName" value="java"></property> <property name="author"> <null/> </property> <property name="price" value="33.5"></property> <property name="isbn" value="house"></property> </bean>
以上都是 String 类型,如果是 int/double 类型,就不能用<null/>
应该直接 <value="0"> 这样就会显示0/0.0
包装类 会显示 null
实验13:引用其他bean
新建 BookShop ,包含 book 属性, 通过 ref 连接上
调用bookShop,即可
<bean id="book" class="com.neuedu.spring.bean.Book"> <property name="bookName" value="java"></property> <property name="isbn" value="house"></property> </bean> <bean id="bookShop" class="com.neuedu.spring.bean.BookShop"> <property name="address" value="河北"></property> <property name="book" ref="book"></property> </bean>
实验14:引用内部bean
在 bean 中在声明一个 bean
<bean id="bookShop" class="com.neuedu.spring.bean.BookShop"> <property name="address" value="河北"></property> <property name="book"> <bean class="com.neuedu.spring.bean.Book"> <property name="bookName" value="java"></property> <property name="author" value="you"></property> <property name="price" value="43.32"></property> <property name="isbn" value="house"></property> </bean> </property> </bean>
实验15:使用List类型的集合属性
在 BookShop 类中添加 bookList 属性,并添加set/get方法
bookList 分别 ref 上 book1、book2、book3、book4
<bean id="book1" class="com.neuedu.spring.bean.Book"> <property name="bookName" value="java"></property> </bean> <bean id="book2" class="com.neuedu.spring.bean.Book"> <property name="author" value="you"></property> </bean> <bean id="book3" class="com.neuedu.spring.bean.Book"> <property name="price" value="43.43"></property> </bean> <bean id="book4" class="com.neuedu.spring.bean.Book"> <property name="isbn" value="home"></property> </bean> <bean id="bookShop" class="com.neuedu.spring.bean.BookShop"> <property name="bookList"> <list> <ref bean="book1"/> <ref bean="book2"/> <ref bean="book3"/> <ref bean="book4"/> </list> </property> </bean>
public class TestIOC { private ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml"); @Test public void test() { BookShop bean = ioc.getBean(BookShop.class); List<Book> bookList = bean.getBookList(); for(Book book : bookList){ System.out.println(book); } } }
实验16:使用Map类型的集合属性
在 BookShop 类中添加 Map 属性,并添加set/get方法
<map> 里面用 <entry>,一个键值对就是一个 <entry>
<bean id="bookShop" class="com.neuedu.spring.bean.BookShop"> <property name="bookMap"> <map> <entry> <key> <value>book1</value> </key> <ref bean="book1"/> </entry> <entry> <key> <value>book2</value> </key> <ref bean="book2"/> </entry> <entry> <key> <value>book3</value> </key> <ref bean="book3"/> </entry> <entry> <key> <value>book4</value> </key> <ref bean="book4"/> </entry> </map> </property> </bean>
用 .entry 遍历 map 集合
public class TestIOC { private ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml"); @Test public void test() { BookShop bean = ioc.getBean(BookShop.class); Map<String, Book> bookMap = bean.getBookMap(); Set<Entry<String,Book>> entrySet = bookMap.entrySet(); for (Entry<String, Book> entry : entrySet) { System.out.println(entry.getKey()+"==="+entry.getValue()); } } }
实验17:使用prop子元素为Properties类型的属性赋值
在 BookShop 类中添加 Properties 属性,并添加set/get方法
<bean id="bookShop" class="com.neuedu.spring.bean.BookShop"> <property name="p"> <props> <prop key="book1">value1</prop> <prop key="book2">value2</prop> <prop key="book3">value3</prop> <prop key="book4">value4</prop> </props> </property> </bean>
public class TestIOC { private ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml"); @Test public void test() { BookShop bean = ioc.getBean(BookShop.class); java.util.Properties p = bean.getP(); System.out.println(p.get("book1")); System.out.println(p.get("book2")); System.out.println(p.get("book3")); System.out.println(p.get("book4")); } }