框架-Spring(上)
1、Spring
1.1简介
-
Spring:春天——>给软件行业带来了春天
-
2002,首次推出了Spring框架的雏形:interface21框架!
-
Spring框架即以interface21框架为基础,经过重新设计,并不断丰富其内涵,于2004年3月24日,发布了1.0正式版
-
Rod Johnson,Spring Framework创始人,著名作者。很难想象Rod Johnson的学历,真的让好多人大吃一惊,他是悉尼大学的博士,然而他的专业不是计算机,而是音乐学。
-
spring理念:使现有的技术更加容易使用,本身是一个大杂烩。
-
SSH:Struct2 + Spring + Hibernate
-
SSM: SpringMVC + Spring + Mybatis
-
官方下载地址:https://repo.spring.io/ui/native/release/org/springframework/spring
-
官方文档:https://docs.spring.io/spring-framework/docs/5.2.0.RELEASE/spring-framework-reference/index.html
-
maven包:
-
spring-webmvc
-
spring-jdbc
1.2优点
-
Spring是一个开源的免费框架
-
Spring是一个轻量级的,非入侵式的框架
-
控制反转(IOC),面向切面编程(AOP)
-
支持事务的处理,对框架整合的支持
-
总结:Spring是一个轻量级的控制反转(IOC)和面向切面编程(AOP)的框架
1.3组成

1.4扩展
在spring的官网有这个介绍,现代化的java开发,说白了就是基于spring的开发

-
SpringBoot
一个快速开发的脚手架
基于SpringBoot可以快速开发单个微服务
约定大于配置 -
SpringCloud
基于SpringBoot实现
大多数公司都在使用SpringBoot进行快速开发,学习SpringBoot的前提,需要完全掌握Spring及SpringMVC!承上启下的作用!
2.IOC理论推导
1.5
1.UserDao接口

2.UserDaoImpl实现类



3.UserService业务接口

4.UserServiceImpl业务实现类

在之前的业务中,用户的需求可能会影响原本的代码,程序员需要根据用户的需求去修改源代码!如果程序代码量十分大,修改一次的成本代价十分昂贵!
使用Set接口实现,发生了革命性的变化
private UserDao userDao;//*利用set进行动态实现值的注入 public void setUserDao(UserDao userDao) { this.userDao = userDao; }-
之前,程序时主动创建对象,控制权在程序猿手上
-
现在,使用set注入后,程序不再具有主动性,而是变成了被动接收对象
这种思想,从本质上解决了问题,程序猿不用再去管理对象的创建了
系统的耦合性大大降低,可以更加专注的在业务的实现上! 这是IOC的原型!
2.1 IOC本质
控制反转IoC(Inversion of Control),是一种设计思想,DI(依赖注入)是实现IoC的一种方法,也有人认为DI只是IoC的另一种说法。在没有IoC的程序中,我们使用面向对象编程,对象的创建与对象间的依赖关系完全硬编码在程序中,对象的创建由程序自己控制,控制反转后将对象的创建转移给第三方,个人认为所谓控制反转就是:获得依赖对象的方式反转了。
采用XML方式配置Bean的时候,Bean的定义信息是和实现分离的,而采用注解的方式可以把二者合为一体,Bean的定义信息直接以注解的形式定义在实现类中,从而达到了零配置的目的。
控制反转是一种通过描述(XML或注解)并通过第三方去生产或获取特定对象的方式。在Spring中实现控制反转的是IoC容器,其实现方法是依赖注入(Dependenccy Injection,DI)。
3.HelloSpring
3.1
<dependencies> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-webmvc</artifactId> <version>6.1.3</version> </dependency> </dependencies>项目目录:

Hello.java
import com.yan.pojo.Hello; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class MyTest { public static void main(String[] args) { //获取spring的上下文对象 ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml"); //我们的对象现在都在spring中管理了,我们要使用的话直接去里面取出来就可以了 Hello hello = (Hello)context.getBean("hello"); System.out.println(hello.toString()); } }applicationContext.xml
<?xml version="1.0" encoding="UTF-8"?> <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"> <!-- 使用spring对象创建,在spring中都叫beans --> <!-- bean = 对象 new Hello() id = 对象名 class = new 的对象 property 相当于给对象中的属性设置一个值 name = 对象中的属性名 value = 设置的值 --> <bean id="hello" class ="com.yan.pojo.Hello"> <property name="str" value="spring"/> </bean> </beans>MyTest.java
import com.yan.pojo.Hello; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class MyTest { public static void main(String[] args) { //获取spring的上下文对象 ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml"); //我们的对象现在都在spring中管理了,我们要使用的话直接去里面取出来就可以了 Hello hello = (Hello)context.getBean("hello"); System.out.println(hello.toString()); } }这个过程就叫控制反转。
-
-
控制:谁来控制对象的创建,传统应用程序的对象是由程序本身控制创建的,使用spring后,对象是由spring来创建的
-
反转:程序本身不创建对象,而变成被动的接收对象
-
依赖注入:就是利用set方法来进行注入的。
IOC是一种编程思想,由主动编程变成了被动接收
可以通过newClassPathXmlApplicationContext去浏览一下底层源码,
ok,到了现在,我们彻底不用再去程序中进行改动了,要实现不同的操作,只需要在xml配置文件中进行修改,所谓的IOC,一句话搞定: -
对象由Spring创建,管理,装配!
4、IOC创建对象的方式
4.1、使用无参构造创建对象,默认!
*必须有无参构造方法
4.2、假设我们要使用有参构造构建对象
-
下标赋值
<!--下标赋值--> <bean id="user" class="com.gty.pojo.User"> <constructor-arg index="0" value="tiantian"/> </bean> -
类型赋值
<!--类型赋值(不建议使用)--> <bean id="user" class="com.gty.pojo.User"> <constructor-arg type="java.lang.String" value="tiantian"/> </bean> -
参数名赋值
<!--参数名赋值--> <bean id="user" class="com.gty.pojo.User"> <constructor-arg name="name" value="tiantian"/> </bean>当在xml配置文件加载时,容器中管理里的对象就已经初始化了(注册bean时就已经被实例化了),无需等到getBean(),
5、Spring配置
5.1、别名
<!--别名,如果添加了别名,也可以通过别名获取到这个对象--> <alias name="user" alias="usernew"/>ApplicationContext context=new ClassPathXmlApplicationContext("beans.xml"); User user=(User)context.getBean("usernew"); user.show();5.2、Bean的配置
<bean id="user" class="com.yan.pojo.User" name = "user1,user2"> <!-- 设置属性 --> <!-- id: bean的唯一标识符,也就是相当于我们学的对象名 class: bean对象所对应的全限定名:包名+类型 name: 也是别名,而且name可以同时取多个别名 --> <property name="name" value="yan"/> </bean>5.3、import
import一般用于团队开发使用,他可以将多个配置文件导入合并为一个。
假设现在项目中有多个人开发,这三个人负责不同的类开发,不同的类需要注册在不同的bean中,我们可以利用import将所有人的beans.xml合并为一个总的applicationContext.xml
-
张三
-
李四
-
王五
applicationContext.xml
<import resource="beans.xml"/> <import resource="bean1.xml"/> <import resource="bean2.xml"/> <import resource="bean3.xml"/>使用时直接使用总的配置。
6、依赖注入
6.1、构造器注入
前面已经提及
6.2、*Set方式注入
依赖注入:set注入依赖注入:set注入
-
依赖:bean对象的创建依赖于容器
-
注入:bean对象中的所有属性,由容器来注入
环境搭建环境搭建 -
复杂类型
public void setName(String name) { this.name = name; } public String getAddress() { return address; } -
真实测试对象
public class Student { private String name; private Address address; private String[] books; private List<String> hobbys; private Map<String,String> card; private Set<String> games; private String wifi; private Properties info; } -
beans.xml
<bean id = "student" class = "com.yan.pojo.Student"> <!--第一种,普通值注入,value--> <property name="name" value = "张三"/> </bean>测试类
public class MyTest { public static void main(String[] args) { // 加载Spring配置文件 ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml"); Student student = (Student)context.getBean("student"); System.out.println(student.getName()); } }完善注入信息
<bean id = "address" class = "com.yan.pojo.Address"/> <bean id = "student" class = "com.yan.pojo.Student"> <!--第一种,普通值注入,value--> <property name="name" value = "张三"/> <!--第二种,Bean注入,ref--> <property name="address" ref = "address"/> <!--数组注入--> <property name="books"> <array> <value>红楼梦</value> <value>西游记</value> <value>水浒传</value> <value>三国演义</value> </array> </property> <!--List注入--> <property name="hobbys"> <list> <value>听歌</value> <value>敲代码</value> <value>看电影</value> </list> </property> <!--Map注入--> <property name="card"> <map> <entry key="身份证" value="123456789012345678"/> <entry key="银行卡" value="123456789012345678"/> </map> </property> <!--Set注入--> <property name="games"> <set> <value>LOL</value> <value>COC</value> <value>BOB</value> </set> </property> <!--Null注入--> <property name="wifi"> <null/> </property> <!--Properties注入--> <property name="info"> <props> <prop key="学号">2019001</prop> <prop key="性别">男</prop> <prop key="姓名">小明</prop> </props> </property> </bean>6.3、扩展方式注入
我们可以使用p命令空间和c命令空间进行注入
官方解释: -
使用
<!--p命名空间注入,可以直接注入属性的值:property--> <bean id = "user" class="com.yan.pojo.User" p:name = "name" p:age = "18"/> <!--c命名空间注入,通过构造器注入:construct-args--> <bean id = "user" class="com.yan.pojo.User" c:name = "name" c:age = "18"/>注意:p命令和c命令不能同时使用,需要导入xml约束,并且c命令需要构造对应的有参方法
6.4、Bean的作用域
-
单例模式(spring默认机制)
<bean id="user2" class="com.gty.pojo.User" c:age="23" c:name="gty" scope="singleton"/> -
原型模式:每次从容器中get时,都会产生一个新的对象
<bean id="user2" class="com.gty.pojo.User" c:age="23" c:name="gty" scope="prototype"/>7、自动装配
-
自动装配是Spring满足bean依赖的一种方式
-
Spring会在上下文中自动寻找,并自动给bean装配属性
在Spring中有三种自动装配方式:
1.在xml中显式配置
2.在java中显式配置
3.隐式的自动装配bean(重要)
7.1、测试
1、环境搭建:一个人有两个宠物
自己进行装配:
<!--byName:会自动在容器上下文查找,和自己对象set方法后面的值对应的bean id--> <bean id = "cat" class="com.yan.pojo.Cat"/> <bean id = "dog" class="com.yan.pojo.Dog"/> <bean id = "people" class="com.yan.pojo.People"> <property name="name" value="Yan"/> <property name="cat" ref="cat"/> <property name="dog" ref="dog"/> </bean>7.2、ByName自动装配
<!-- byName:会自动在上下文中查找和自己对象set方法后面的值对应的beanid --> <bean id = "people" class="com.yan.pojo.People" autowire="byName"> <property name="name" value="Yan"/> </bean>7.3、ByType自动装配
<!-- byType:会自动在上下文中查找和自己对象属性类型相同的bean --> <bean id = "people" class="com.yan.pojo.People" autowire="byType"> <property name="name" value="Yan"/> </bean>注:如果有两个
则找不到,必须保持全局唯一 小结:
-
byName,需要保证所有bean的id唯一,并且这个bean需要和自动注入的属性的set方法的值一致。
-
byType,需要保证所有bean的class唯一,并且这个bean需要和自动注入的属性的类型一致。
7.4 使用注解实现自动装配
jdk1.5支持的注解,Spring2.5就支持注解了
使用注解须知:
1.导入约束:context约束
2.配置注解的支持:context:annotation-config/<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"> <context:annotation-config/> </beans>@Autowired:
直接在属性上使用即可!也可以在set方式上使用!public class People { //如果显示定义了Autowired的required属性为false,说明这个对象可以为null,否则不允许为null @Autowired private Dog dog; @Autowired private Cat cat; private String name; } -
使用Autowired我们可以不用编写Set方法,前提是这个自动装配的属性在IOC(Spring)容器中存在,且符合名字byName!
-
如果@Autowired自动装配的环境比较复杂,自动装配无法通过一个注解@Autowired完成的时候 ,可以使用@Qualifier(“xxx”)去配合@Autowired的使用,即既无法通过byName又无法通过byType时,指定一个唯一的bean对象注入
<bean id="cat1111" class="com.gty.pojo.Cat"/> <bean id="cat222" class="com.gty.pojo.Cat"/> <bean id="dog1111" class="com.gty.pojo.Dog"/> <bean id="dog222" class="com.gty.pojo.Dog"/> public class People { @Autowired @Qualifier("cat222") private Cat cat; @Autowired @Qualifier("dog222") private Dog dog; }public class People { @Resource(name="cat1111") private Cat cat; @Resource private Dog dog; } -
都是用来自动装配的,都可以放在属性字段上
-
@Autowired通过byType实现,必须要求这个对象存在
-
@Resoutce默认通过byName方式实现,如果找不到名字,则通过byType实现
-
执行顺序不同:@Autowired通过byType实现,@Resoutce通过byName实现
8、使用注解开发
在spring4之后,要使用注解开发,必须要保证aop的包导入了

使用注解需要导入context约束,增加注解的支持
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"> <context:annotation-config/> </beans>1、bean
<!--指定要扫描的包,这个包下的注解就会生效--> <context:component-scan base-package="com.gty.pojo"/> <!--开启注解支持--> <context:annotation-config/>
//<bean id = "user" class="com.yan.pojo.User"/> @Component //组件 public class User { //方法与属性 }2、属性如何注入
可以放在属性上或者set方法上@Componentpublic class User { //相当于<property name="name" value="tiantian"/> @Value("Yan") public String name; //相当于<property name="name" value="tiantian"/> @Value("Yan") public void setName(String name) { this.name = name; } }3、衍生的注解
@Component有几个衍生注解,我们在web开发中会按照mvc三层架构分层
dao [@Repository]
pojo [@Component]
service [@Service]
controller [@Controller]
这四个注解功能都是一样的,都是代表将某个类注册到Spring中,装配bean4、自动装配
@Autowired:自动装配通过类型和名字
如果@Autowired不能唯一自动装配上属性,则需要通过@Qualifier(value=”xxx”)
@Nullable:某个字段标记了这个注解,说明这个字段可以为空
@Resoure:自动装配通过名字、类型
5、作用域
@Scope(“”)@Component @Scope("prototype") public class User { public String name; //相当于<property name="name" value="tiantian"/> @Value("Yan") public void setName(String name) { this.name = name; } }6、小结
xml与注解:xml更加万能,适用于任何场合!维护简单方便 注解不是自己类使用不了,维护相对复杂!xml与注解最佳实践:
xml用来管理bean 注解只负责完成属性的注入 我们在使用的过程中,只需要注意一个问题:必须让注解生效,开启注解支持//@Component意思是这个类被Spring托管了,注册到了容器中@Componentpublic class User { private String name; public String getName() { return name; } @Value("tiantian") public void setName(String name) { this.name = name; } @Override public String toString() { return "User{" + "name='" + name + '\'' + '}'; } }9、使用Java的方式配置Spring
可以完全不使用Spring的xml配置,全权交给Java实现!
JavaConfig是Spring的一个子项目,在Spring4之后变为核心功能。 -
实体类
//@Component意思是这个类被Spring托管了,注册到了容器中 @Component public class User { private String name; public String getName() { return name; } @Value("Yan") public void setName(String name) { this.name = name; } @Override public String toString() { return "User{" + "name='" + name + '\'' + '}'; }} -
配置类
//@Configuration是一个配置类,和之前的beans.xml是一样的 @Configuration//这个也会被Spring容器托管,注册到容器中,因为它本来也是一个@Component @Import(Config2.class) public class GtyConfig { //注册一个Bean,就相当于之前写的一个Bean标签 //这个方法的名字,相当于bean标签中的id属性 //这个方法的返回值,就相当于bean标签中的class属性 @Bean public User getUser(){ return new User();//就是返回要注入到bean的对象 } } -
测试类
public class MyTest { public static void main(String[] args) { //如果完全使用了配置类方式去做,我们就只能通过AnnotationConfig上下文来获取容器,通过配置类的class对象加载 ApplicationContext context=new AnnotationConfigApplicationContext(GtyConfig.class); User getUser=context.getBean("getUser",User.class); System.out.println(getUser.getName()); } }这种纯java的配置,在SpringBoot中随处可见

浙公网安备 33010602011771号