Spring是一个轻量级的控制反转(IoC)和面向切面(AOP)的容器(框架)。
控制反转是一种通过描述(XML或注解)并通过第三方去生产或获取特定对象的方式。在Spring中实现控制反转的是IoC容器,其实现方法是依赖注入(Dependency Injection,DI)。
控制 : 谁来控制对象的创建 , 传统应用程序的对象是由程序本身控制创建的 , 使用Spring后 , 对象是由Spring来创建的。
反转 : 程序本身不创建对象 , 而变成被动的接收对象 。
beans.xml可根据index参数下标、参数名字、参数类型设置。
可用import实现多个xml文件整合。
DI注入
1.构造器注入
2.set注入(重点实现)
Student.java
package org.example;
import java.util.*;
public class Student {
private String name;
private Address address;
private String[] books;
private List hobbys;
private Map card;
private Set games;
private String wife;
private Properties info;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Address getAddress() {
return address;
}
public void setAddress(Address address) {
this.address = address;
}
public String[] getBooks() {
return books;
}
public void setBooks(String[] books) {
this.books = books;
}
public List getHobbys() {
return hobbys;
}
public void setHobbys(List hobbys) {
this.hobbys = hobbys;
}
public Map getCard() {
return card;
}
public void setCard(Map card) {
this.card = card;
}
public Set getGames() {
return games;
}
public void setGames(Set games) {
this.games = games;
}
public String getWife() {
return wife;
}
public void setWife(String wife) {
this.wife = wife;
}
public Properties getInfo() {
return info;
}
public void setInfo(Properties info) {
this.info = info;
}
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", address=" + address.toString() +
", books=" + Arrays.toString(books) +
", hobbys=" + hobbys +
", card=" + card +
", games=" + games +
", wife='" + wife + '\'' +
", info=" + info +
'}';
}
}
Address.java
package org.example;
public class Address {
private String address;
public String getAddress() {
return address;
}
public void setAddress(String address) {
this.address = address;
}
@Override
public String toString() {
return "Address{" +
"address='" + address + '\'' +
'}';
}
}
注入:
红楼梦
西游记
听歌
敲代码
LOL
2020202
小李
测试:
public class Mytest {
public static void main(String[] args) {
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
Student student = (Student) context.getBean("student");
System.out.println(student.toString());
}
}
结果:
Student{name='哈哈', address=Address{address='null'}, books=[红楼梦, 西游记], hobbys=[听歌, 敲代码], card={身份证=1234566788}, games=[LOL], wife='null', info={学号=2020202, 姓名=小李}}
3.p命名和c命名注入
自动装配
byname自动装配
当一个bean节点带有 autowire byName的属性时。
将查找其类中所有的set方法名,例如setCat,获得将set去掉并且首字母小写的字符串,即cat。
去spring容器中寻找是否有此字符串名称id的对象。
如果有,就取出注入;如果没有,就报空指针异常。
此外,还有byType
同一类型的对象,在spring容器中唯一。如果不唯一,会报不唯一的异常。
使用注解
@Autowired
注意配置注解支持。
过程中遇见Autowired members must be defined in valid Spring bean
前面添加@component。
自动装配无法通过一个注解完成,搭配@Qualifier,指定一个唯一的bean对象注入。
@Resource
如有指定的name属性,先按该属性进行byName方式查找装配;其次再进行默认的byName方式进行装配;如果以上都不成功,则按byType的方式自动装配。两个都找不到则报错。
注解开发
配置文件
User.java
@Component
public class User {
@Value("哈哈")
public String name;
}
提供了set方法,能在set方法上添加@value(“值”);
@Component三个衍生注解
@Controller:web层
@Service:service层
@Repository:dao层
JavaConfig
User.java
@Component
public class User {
private String name="哈哈";
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
'}';
}
}
Config.java
@Configuration
public class Config {
@Bean
public User getUser(){
return new User();
}
}
测试
public class Mytest {
public static void main(String[] args) {
ApplicationContext context = new AnnotationConfigApplicationContext(Config.class);
User getUser = (User) context.getBean("getUser");
System.out.println(getUser.getName());
}
}
静态代理
Rent . java 即抽象角色
public interface Rent {
public void rent();
}
Host . java 即真实角色
public class Host implements Rent {
public void rent() {
System.out.println("房出租");
}
}
Proxy . java 即代理角色
public class Proxy implements Rent{
private Host host;
public Proxy(){
}
public Proxy(Host host){
this.host = host;
}
@Override
public void rent() {
host.rent();
}
}
Client . java 即客户
public class Client {
public static void main(String[] args) {
Host host = new Host();
Proxy proxy = new Proxy(host);
proxy.rent();
}
}
动态代理
动态代理的角色和静态代理的一样。
动态代理的代理类是动态生成的,静态代理的代理类是我们提前写好的。
动态代理分为两类 : 一类是基于接口动态代理 ,一类是基于类的动态代理。
AOP
AOP(Aspect Oriented Programming)意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。 AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
横切关注点:跨越应用程序多个模块的方法或功能。即是,与我们业务逻辑无关的,但是我们需要关注的部分,就是横切关注点。如日志 , 安全 , 缓存 , 事务等等 …
切面(ASPECT):横切关注点 被模块化 的特殊对象。即,它是一个类。
通知(Advice):切面必须要完成的工作。即,它是类中的一个方法。
目标(Target):被通知对象。
代理(Proxy):向目标对象应用通知之后创建的对象。
切入点(PointCut):切面通知 执行的 “地点”的定义。
连接点(JointPoint):与切入点匹配的执行点。