Spring学习笔记一 IOC、AOP原理
Spring的核心是:
IOC:控制反转也叫依赖注入(DI);AOP:面向切面编程,这两个是Spring的灵魂。简单来说,Spring是一个分层的一站式轻量级开源框架。
IOC和DI由什么关系呢?其实它们是同一个概念的不同角度描述,由于控制反转概念比较含糊(ioc不是什么技术,而是一种设计思想,Ioc意味着将你设计好的对象交给容器控制,而不是传统的在你的对象内部直接控制。)
原理: 主要用到的设计模式有工厂模式和代理模式。
1.IOC就是典型的工厂模式,通过sessionfactory去注入实例。
2.AOP就是典型的代理模式的体现。cglib代理和动态代理>>> invoke反射机制。
IOC说明:将对象交给容器管理,你只需要在spring配置文件总配置相应的bean(ioc-控制生成bean),以及设置相关的属性,让spring容器来生成类的实例对象以及管理对象。//在spring容器启动的时候,spring会把你在配置文件中配置的bean都初始化好,然后在你需要调用的时候,就把它已经初始化好的那些bean分配给你需要调用这些bean的类(假设这个类名是A),动态的向某个对象提供它所需要的其他对象(DI依赖注入-获取对象这时候反转了),而不需要你在A里面new这些bean了。
AOP说明:动态地将代码切入到类的指定方法、指定位置上的编程思想就是面向切面的编程 。
AOP可以说是对OOP的补充和完善。OOP引入封装、继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合。
实现aop的两种方式:一是采用动态代理技术,利用截取消息的方式,对该消息进行装饰,以取代原有对象行为的执行;二是采用静态织入的方式,引入特定的语法创建“方面”,从而使得编译器可以在编译期间织入有关“方面”的代码。
简单点解释举例:比方说你想在你的biz层所有类中都加上一个打印‘你好’的功能,这时就可以用aop思想来做.你先写个类写个类方法,方法经实现打印‘你好’,然后Ioc这个类 ref=“biz.*”让每个类都注入即可实现。 比如事务管理、权限控制、日志记录、性能统计等。
spring项目搭建:
1、导包
除了上面的4个包之外还需要日志包:
下面的包可选,老版本有需要:
com.springsource.org.apache.log4j-1.2.15.jar
2、创建对象:
public class Car { private String name; private String color; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getColor() { return color; } public void setColor(String color) { this.color = color; } @Override public String toString() { return "Car [name=" + name + ", color=" + color + "]"; } }
public class User { public User() { System.out.println("User对象空参构造方法!!!!"); } private String name; private Integer age; private Car car; public User(String name, Car car) { System.out.println("User(String name, Car car)!!"); this.name = name; this.car = car; } public User(Car car,String name) { System.out.println("User(Car car,String name)!!"); this.name = name; this.car = car; } public User(Integer name, Car car) { System.out.println("User(Integer name, Car car)!!"); this.name = name+""; this.car = car; } //get set 方法 public void init(){ System.out.println("我是初始化方法!"); } public void destory(){ System.out.println("我是销毁方法!"); } @Override public String toString() { return "User [name=" + name + ", age=" + age + ", car=" + car + "]"; } }
3、书写配置注册对象到容器:
配置文件放的位置任意(建议放到src下);配置文件名任意(建议applicationContext.xml)。
在配置文件中添加如下代码:
4、代码测试
package com.yyb.springTest;
import com.yyb.bean.User;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
* Created by Administrator on 2017/8/8.
*/
public class SpringTest {
@Test
public void func1(){
ApplicationContext ac =new ClassPathXmlApplicationContext("applicationContext.xml");
User user = (User)ac.getBean("user");
System.out.println(user);
}
}
打印结果如下,可已发现User对象创建成功了:
User对象空参构造方法!!!!
User [name=null, age=null, car=null]
IOC和AOP:
IOC: Inverse Of Control ,即反转控制。指的是对象的创建权反转(交给)给Spring。作用是实现了程序的解耦合。
以前对象的创建是由我们自己创建,比如Service层中需要使用Dao层的类,则直接在Service层中使用new创建Dao层的对象。两层之间严重耦合。使用了Spring之后,对象的创建以及依赖关系可以由spring完成创建以及注入。
注入方式:
- set方法注入
- 构造方法注入
- 接口注入
注入类型:
- 值类型注入 (8大基本数据类型)
- 引用类型注入(将依赖对象注入)
Spring配置详解:
1、Bean元素
使用Bean元素描述需要spring容器管理的对象。
- class属性:被管理对象的完整类名。
- name属性:给被管理的对象起个名字.获得对象时根据该名称获得对象。 可以重复;可以使用特殊字符。
- id属性:与name属性一模一样.。但名称不可重复;不能使用特殊字符。尽量使用name属性。
-
<bean name="user" class="cn.itcast.bean.User" ></bean>
2、Bean元素进阶:
scope属性
- singleton(默认值):单例对象,被标识为单例的对象在spring容器中只会存在一个实例。
- prototype:多例原型,被标识为多例的对象,每次在获得才会创建。每次创建都是新的对象。整合struts2时,ActionBean必须配置为多例的。
- request:web环境下,对象与request生命周期一致。
- session:web环境下,对象与session生命周期一致。
-
<bean name="user" class="com.yyb.bean.User" scope="singleton"></bean>
测试代码:
@Test
//scope:singleton 单例
//scope:prototype 多例
public void fun4(){
//1 创建容器对象
ApplicationContext ac = new ClassPathXmlApplicationContext("cn/itcast/b_create/applicationContext.xml");
//2 向容器"要"user对象
User u = (User) ac.getBean("user");
User u2 = (User) ac.getBean("user");
User u3 = (User) ac.getBean("user");
User u4 = (User) ac.getBean("user");
System.out.println(u2==u4);//单例:true
//多例:false
//3 打印user对象
System.out.println(u);
}
生命周期属性:
通过配置<bean>标签上的init-method作为Bean的初始化的时候执行的方法,配置destory-method作为Bean的销毁的时候执行的方法。销毁的方法想要执行,需要是单利创建的Bean,而且在工厂关闭的时候,Bean才会被销毁。
<bean name="user" class="cn.itcast.bean.User" init-method="init" destroy-method="destory" ></bean>
3、Spring创建对象的方式:
1:空参构造创建
<bean name="user" class="cn.itcast.bean.User"></bean>
创建方式2:静态工厂创建 :
<!-- 创建方式2:静态工厂创建 调用UserFactory的createUser方法创建名为user2的对象.放入容器--> <bean name="user2" class="cn.itcast.b_create.UserFactory" factory-method="createUser" ></bean>
UserFactory类
package cn.itcast.b_create;
import cn.itcast.bean.User;
public class UserFactory {
public static User createUser(){
System.out.println("静态工厂创建User");
return new User();
}
}
测试代码:
//创建方式2:静态工厂
@Test
public void fun2(){
//1 创建容器对象
ApplicationContext ac = new ClassPathXmlApplicationContext("cn/itcast/b_create/applicationContext.xml");
//2 向容器"要"user对象
User u = (User) ac.getBean("user2");
//3 打印user对象
System.out.println(u);
}
创建方式3:实例工厂创建:
<!-- 创建方式3:实例工厂创建 调用UserFactory对象的createUser2方法创建名为user3的对象.放入容器--> <bean name="user3" factory-bean="userFactory" factory-method="createUser2" ></bean> <bean name="userFactory" class="cn.itcast.b_create.UserFactory" ></bean>
UserFactory类
package cn.itcast.b_create;
import cn.itcast.bean.User;
public class UserFactory {
public User createUser2(){
System.out.println("实例工厂创建User");
return new User();
}
}
测试代码:
@Test
public void fun3(){
//1 创建容器对象
ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
//2 向容器"要"user对象
User u = (User) ac.getBean("user3");
//3 打印user对象
System.out.println(u);
}
Spring属性注入:
1.set方法注入 : User类中必须存在SetXXX()方法。
<!-- set方式注入: --> <bean name="user" class="cn.itcast.bean.User" > <!--值类型注入: 为User对象中名为name的属性注入tom作为值 --> <property name="name" value="tom" ></property> <property name="age" value="18" ></property> <!-- 引用类型注入: 为car属性注入下方配置的car对象 --> <property name="car" ref="car" ></property> </bean>
<!-- 将car对象配置到容器中 --> <bean name="car" class="cn.itcast.bean.Car" > <property name="name" value="兰博基尼" ></property> <property name="color" value="黄色" ></property> </bean>
2.构造方法注入:
<!-- 构造函数注入 --> <bean name="user2" class="cn.itcast.bean.User" > <!-- name属性: 构造函数的参数名 --> <!-- index属性: 构造函数的参数索引 --> <!-- type属性: 构造函数的参数类型--> <constructor-arg name="name" index="0" type="java.lang.Integer" value="999" ></constructor-arg> <constructor-arg name="car" ref="car" index="1" ></constructor-arg> </bean>
前提,必须有相应的构造函数:
public User(Integer name, Car car) {
System.out.println("User(Integer name, Car car)!!");
this.name = name+"";
this.car = car;
}
3. p名称空间注入:
<!-- p名称空间注入, 走set方法 1.导入P名称空间 xmlns:p="http://www.springframework.org/schema/p" 2.使用p:属性完成注入 |-值类型: p:属性名="值" |-对象类型: p:属性名-ref="bean名称"--> <bean name="user3" class="cn.itcast.bean.User" p:name="jack" p:age="20" p:car-ref="car" > </bean>
spel注入:
<!-- spel注入: spring Expression Language sping表达式语言--> <bean name="user4" class="cn.itcast.bean.User" > <property name="name" value="#{user.name}" ></property> <!--取对象为user的name值--> <property name="age" value="#{user3.age}" ></property> <!--取对象为user3的age值--> <property name="car" ref="car" ></property> </bean>
复杂类型注入:
<!-- 复杂类型注入 --> <bean name="cb" class="cn.itcast.c_injection.CollectionBean" > <!-- 如果数组中只准备注入一个值(对象),直接使用value|ref即可 <property name="arr" value="tom" ></property>--> <!-- array注入,多个元素注入 --> <property name="arr"> <array> <value>tom</value> <value>jerry</value> <ref bean="user4" /> </array> </property> <!-- 如果List中只准备注入一个值(对象),直接使用value|ref即可 <property name="list" value="jack" ></property>--> <property name="list" > <list> <value>jack</value> <value>rose</value> <ref bean="user3" /> </list> </property> <!-- map类型注入 --> <property name="map" > <map> <entry key="url" value="jdbc:mysql:///crm" ></entry> <entry key="user" value-ref="user4" ></entry> <entry key-ref="user3" value-ref="user2" ></entry> </map> </property> <!-- prperties 类型注入 --> <property name="prop" > <props> <prop key="driverClass">com.jdbc.mysql.Driver</prop> <prop key="userName">root</prop> <prop key="password">1234</prop> </props> </property> </bean>
CollectionBean类:
package cn.itcast.c_injection;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.Properties;
public class CollectionBean {
private Object[] arr;//数组类型注入
private List list;//list/set 类型注入
private Map map;//map类型注入
private Properties prop;//properties类型注入
public Object[] getArr() {
return arr;
}
public void setArr(Object[] arr) {
this.arr = arr;
}
public List getList() {
return list;
}
public void setList(List list) {
this.list = list;
}
public Map getMap() {
return map;
}
public void setMap(Map map) {
this.map = map;
}
public Properties getProp() {
return prop;
}
public void setProp(Properties prop) {
this.prop = prop;
}
@Override
public String toString() {
return "CollectionBean [arr=" + Arrays.toString(arr) + ", list=" + list + ", map=" + map + ", prop=" + prop
+ "]";
}
}