spring理念:使现有的技术更加容易使用,本身是一个大杂烩,整合了现有的技术框架
Spring就是一个轻量级的控制反转(IOC)和面向切面编程的框架
Spring是一个分层架构,由7个定义良好的模块组成。Spring模块构建在核心容器之上,核心容器定义了创建、配置和管理bean
的方式。
组成Spring框架的每个模块(或组件)都可以单独存在,或者与其他一个或多个模块联合实现。每个模块的功能如下:
BeanFactory
,它是工厂模式的实现。BeanFactory
使用控制反转(IOC) 模式将应用程序的配置和依赖性规范与实际的应用程序代码分开。Spring Boot与Spring Cloud
新建一个空白的maven项目
先用JavaWeb的方式写一个需求:
UserDao接口
public interface UserDao {
public void getUser();
}
Dao的实现类
public class UserDaoImpl implements UserDao {
@Override
public void getUser() {
System.out.println("获取用户数据");
}
}
UserService接口
public interface UserService {
public void getUser();
}
UserService实现类
public class UserServiceImpl implements UserService {
private UserDao userDao = new UserDaoImpl();
@Override
public void getUser() {
userDao.getUser();
}
}
测试
@Test
public void test(){
UserService service = new UserServiceImpl();
service.getUser();
}
当增加一个UserDao的实现类时:
public class UserDaoMySqlImpl implements UserDao {
@Override
public void getUser() {
System.out.println("MySql获取用户数据");
}
}
如果要使用MySql的话,就需要去service实现类里修改对应的实现:
public class UserServiceImpl implements UserService {
private UserDao userDao = new UserDaoMySqlImpl();
@Override
public void getUser() {
userDao.getUser();
}
}
如果再增加一个UserDao的实现类:
public class UserDaoOracleImpl implements UserDao {
@Override
public void getUser() {
System.out.println("Oracle获取用户数据");
}
}
如果要使用Oracle,又需要去service实现类里面修改对应的代码。假设这种需求非常大,这种方式就不适用了,每次变动都需要修改大量代码,这种设计的耦合性太高了。
如果我们可以在需要dao层实现类的地方,不去实现它,而是留出一个接口,我们可以使用set
方法实现:
public class UserServiceImpl implements UserService {
private UserDao userDao;
// 利用set实现
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
}
@Override
public void getUser() {
userDao.getUser();
}
}
再去测试:
@Test
public void test(){
UserServiceImpl service = new UserServiceImpl();
service.setUserDao( new UserDaoMySqlImpl() );
service.getUser();
//那我们现在又想用Oracle去实现呢
service.setUserDao( new UserDaoOracleImpl() );
service.getUser();
}
这时已经发生了根本性的变化:之前程序是主动创建对象,控制权在程序员手上。使用set注入后,程序不再具有主动性,而是变成了被动的接收对象。
这种思想,从本质上解决了问题,程序员不用再去管理对象的创建,降低耦合性,可以更加专注业务的实现,这是IOC的原型。
控制反转IOC是一种设计思想,**DI(依赖注入)**是实现IOC的一种方法。
在没有IOC的程序中,使用面向对象编程,对象的创建与对象间的依赖关系完全硬编码在程序中,对象的创建由程序自己控制,控制反转后将对象的创建转移给第三方。所谓控制反转就是:获得依赖对象的方式反转了。
IOC是Spring框架的核心内容,使用很多种方式完美的实现了IOC,可以使用XML配置,也可以使用注解。
Spring容器在初始化时先读取配置文件,根据配置文件或元数据创建与组织对象存入容器中,程序使用时再从IOC容器中取出需要的对象。
采用XML方式配置Bean的时候,Bean的定义信息是和实现分离的,而采用注解的方式可以把两者合为
一体,Bean的定义信息直接以注解的形式定义在实现类中,从而达到了零配置的目的。
控制反转是一种通过描述(XML或注解)并通过第三方去生产或获取特定对象的方式。在Spring中实现控制反转的是IoC容器,其实现方法是依赖注入(Dependency Injection,DI)。
<dependencies>
<dependency>
<groupId>org.springframeworkgroupId>
<artifactId>spring-webmvcartifactId>
<version>5.2.10.RELEASEversion>
dependency>
dependencies>
实体类
public class Hello {
private String str;
public String getStr() {
return str;
}
public void setStr(String str) {
this.str = str;
}
@Override
public String toString() {
return "Hello{" +
"str='" + str + '\'' +
'}';
}
}
编写spring文件 , 这里命名为beans.xml
<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
https://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="hello" class="com.study.pojo.Hello">
<property name="str" value="Spring"/>
bean>
beans>
测试
public static void main(String[] args) {
// 获取Spring的上下文对象
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
// 对象现在都在Spring中管理了,要使用直接去里面取出来就可以
Hello hello = (Hello)context.getBean("hello");
System.out.println(hello.toString());
}
Hello对象是谁创建的【Hello对象是由Spring创建的】
Hello对象属性值怎么设置的【Hello对象属性值是由Spring容器设置的】
这个过程就叫控制反转:
依赖注入:就是利用set方法来进行注入的。
IOC是一种编程思想,由主动的编程变成被动的接收。
<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="MysqlImpl" class="com.kuang.dao.impl.UserDaoMySqlImpl"/>
<bean id="OracleImpl" class="com.kuang.dao.impl.UserDaoOracleImpl"/>
<bean id="ServiceImpl" class="com.kuang.service.impl.UserServiceImpl">
<property name="userDao" ref="OracleImpl"/>
bean>
beans>
@Test
public void test2(){
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
UserServiceImpl serviceImpl = (UserServiceImpl)context.getBean("ServiceImpl");
serviceImpl.getUser();
}
要实现不同的操作,不需要再去改程序中的代码了,只需要改xml文件中的数据。
User实体类
public class User {
private String name;
public User() {
System.out.println("user无参构造方法");
}
public void setName(String name) {
this.name = name;
}
public void show(){
System.out.println("name="+ name );
}
}
beans.xml
<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="user" class="com.kuang.pojo.User">
<property name="name" value="kuangshen"/>
bean>
beans>
测试
@Test
public void test(){
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
//在执行getBean的时候, user已经创建好了 , 通过无参构造
User user = (User) context.getBean("user");
//调用对象的方法 .
user.show();
}
结果可以发现,在调用show方法之前,User对象已经通过无参构造初始化了。本质是先通过无参构造创建出对象,然后调用setter
方法赋值。
public class UserT {
private String name;
public UserT(String name) {
this.name = name;
}
public void setName(String name) {
this.name = name;
}
public void show(){
System.out.println("name="+ name );
}
}
依赖注入的三种方式
<bean id="User" class="com.study.pojo.User">
<constructor-arg index="0" value="还好"/>
bean>
<bean id="user" class="com.study.pojo.User">
<constructor-arg type="java.lang.String" value="哈哈"/>
bean>
<bean id="User" class="com.study.pojo.User">
<constructor-arg name="name" value="哈哈">constructor-arg>
bean>
测试
@Test
public void testT(){
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
UserT user = (UserT) context.getBean("userT");
user.show();
}
结论: