Spring

104 阅读6分钟

持续创作,加速成长!这是我参与「掘金日新计划 · 6 月更文挑战」的第7天,点击查看活动详情

1.1、简介

spring理念:使现有的技术更加容易使用,本身就是一个大杂烩,整合了现有的技术框架

  • SSH :Struct2+Spring+Hibernate
  • SSM :SpringMVC+Spring+MyBatis

官网:spring.io/projects/sp…

官方下载地址:repo.spring.io/ui/native/r…

github地址:github.com/spring-proj…

 <!-- https://mvnrepository.com/artifact/org.springframework/spring-webmvc -->
 <dependency>
     <groupId>org.springframework</groupId>
     <artifactId>spring-webmvc</artifactId>
     <version>5.3.18</version>
 </dependency>
 ​
 <!-- https://mvnrepository.com/artifact/org.springframework/spring-webmvc -->
 <dependency>
     <groupId>org.springframework</groupId>
     <artifactId>spring-jdbc</artifactId>
     <version>5.3.18</version>
 </dependency>
 ​

1.2、优点

  • Spring是一个免费的开源的框架(容器)
  • Spring是一个轻量级的非入侵式的框架
  • 控制反转(IOC),面向切面编程(AOP)
  • 支持事务的处理,对框架整合的支持

总结

Spring就是一个轻量级的控制反转(IOC)和面向切面编程的(AOP)框架

1.3、组成

image-20220411220518414.png

1.4、拓展

Spring官网的介绍:现代化的Java开发,基于Spring开发

image-20220411220715230.png

  • Spring Boot

    • 一个快速开发的脚手架
    • 基于SpringBoot可以快速开发单个微服务
    • 约定大于配置
  • Spring Cloud

    • 基于SpringBoot实现

2、IOC理论推导

原来:

  1. UserDao 接口

     public interface UserDao {
         void getUser();
     }
    
  2. UserDaoImpl 实现类

     public class UserDaoImpl implements UserDao{
         public void getUser() {
             System.out.println("默认获取用户的数据");
         }
     }
    
  3. UserService 业务接口

     public interface UserService {
         void getUser();
     }
    
  4. UserServiceImpl 业务实现类

     public class UserServiceImpl implements UserService {
        //原先写法
        //private UserDao userDao = new UserDaoImpl;
        
        private UserDao userDao;
     ​
         //利用set实现值的动态注入
         public void setUserDao(UserDao userDao) {
             this.userDao = userDao;
         }
     ​
         public void getUser() {
             userDao.getUser();
         }
     }
    

在之前的业务中,用户的需求可能会影响原来的代码,需要根据用户需求去修改源代码

使用一个set接口实现

 private UserDao userDao;
 ​
 //利用set实现动态的值的注入
 public void setUserDao(UserDao userDao) {
    this.userDao = userDao;
 }
  • 之前,程序是主动创建对象,控制权在程序员手里
  • 使用了set注入后,程序不在具有主动性,被动接收对象

这种思想,本质上解决了问题,程序员不用再管理对象的创建。系统的耦合性大大降低,可以更加专注在业务的实现。

IOC本质

控制反转IoC(Inversion of Control),是一种设计思想,DI(依赖注入)是实现IoC的一种方法,也有人认为DI只是引oC的另一种说法。没有引oC的程序中,我们使用面向对象编程,对象的创建与对象间的依赖关系完全硬编码在程序中,对象的创建由程序自己控制,控制反转后将对象的创建转移给第三方,所谓控制反转就是:获得依赖对象的方式反转了。

采用XML方式配置Bean的时候,Bean的定义信息是和实现分离的,而采用注解的方式可以把两者合为一体Bean的定义信息直接以注解的形式定义在实现类中,从而达到了零配置的目的。

控制反转是一种通过描述(XML或注解)并通过第三方去生产或获取特定对象的方式。在Spig中实现控制反转的是IoC容器,其实现方法是依赖注入(Dependency Injection,Dl)

3、HelloSpring

  1. 实体类

     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 + ''' +
                     '}';
         }
     }
    
  2. beans.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
             https://www.springframework.org/schema/beans/spring-beans.xsd">
     ​
         <!--使用Spring创建对象,在Spring这些都称为Bean-->
         <bean id="hello" class="com.rop.pojo.Hello">
             <property name="str" value="HelloSpring"/>
         </bean>
     ​
     </beans>
    
  3. 测试

     public class Test {
         public static void main(String[] args) {
             //获取Spring的上下文对象
             ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
             //在context中取出对象
             Hello hello = (Hello) context.getBean("hello");
             System.out.println(hello.toString());
         }
     }
    

这个过程就叫控制反转:

控制:谁来控制对象的创建,传统应用程序的对象是由程序本身控制创建的,使用Spring后,对象是由Spring来创建的。

反转:程序本身不创建对象,而变成被动的接收对象。

依赖注入:就是利用set方法来进行注入的。

IOC是一种编程思想,由主动的编程变成被动的接收。

可以通过newClassPathXmlApplicationContext去浏览一下底层源码。

IOC:对象由Spring来创建,管理,装配!

4、IOC创建对象的方式

  1. 使用无参构造创建对象,默认

  2. 假如要使用有参构造创建对象

    1. 下标赋值

       <bean id="User" class="com.rop.pojo.User">
          <!--有参构造,方式一 通过下标赋值-->
          <constructor-arg index="0" value="SpringStudy"/>
       </bean>
      
    2. 通过类型赋值

       <bean id="User" class="com.rop.pojo.User">
          <!--方式二 通过类型创建, 不建议使用,假如有两个string参数,方法无效-->
          <constructor-arg type="java.lang.String" value="rop"/>
       </bean>
      
    3. 参数名

       <bean id="User" class="com.rop.pojo.User">
          <!--方式三,直接通过参数名来设置-->
          <constructor-arg name="name" value="rao"/>
       </bean>
      

总结:在配置文件加载的时候,容器中管理的对象就已经初始化了。

5、Spring配置

5.1、别名

 <!--别名,如果添加了别名,我们也可以使用别名来获取这个对象-->
 <alias name="User" alias="rop"/>

5.2、Bean的配置

 <!--
 id:bean的唯一标识符。相当于我们学的对象名
 class:bean对象所对应的全限定名:包名+类名
 name:也是一个别名,而且name更高级,可以同时取多个别名
 -->
 <bean id="userT" class="com.rop.pojo.UserT" name="user2 u2,u3;u4">
     <property name="name" value="Spring"/>
 </bean>

5.3、import

一般用于团队开发,可以将多个配置文件导入合并为一个。

image-20220415220445776.png

选择第一个Beans。

再在applicationContext.xml导入其他的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
         https://www.springframework.org/schema/beans/spring-beans.xsd">
 ​
     <import resource="beans.xml"/>
     <import resource="beans2.xml"/>
     <import resource="beans3.xml"/>
 </beans>

6、DI(依赖注入)

6.1、构造器注入

上面已经叙述了

6.2、Set方式注入【重点】

  • 依赖注入:Set注入!

  • 依赖

    • bean对象的创建依赖于容器
  • 注入

    • bean对象中的所有属性由容器注入

环境搭建

  1. 复杂类型
  2. 真实测试对象

环境搭建代码

 public class Address {
     private String address;
 ​
     public String getAddress() {
         return address;
     }
 ​
     public void setAddress(String address) {
         this.address = address;
     }
 }
 public class Student {
     private String name;
     private Address address;
     private String[] books;
     private List<String> hobbies;
     private Map<String,String> cards;
     private Set<String> games;
     private String wife;
     private Properties info;
 }
 <?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
         https://www.springframework.org/schema/beans/spring-beans.xsd">
 ​
     <bean id="student" class="com.rop.pojo.Student">
         <!--第一种:普通值注入,value-->
         <property name="name" value="Mr.rop"/>
     </bean>
 </beans>
 public class DiTest {
     public static void main(String[] args) {
         ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
 ​
         Student student = (Student) context.getBean("student");
         System.out.println(student.getName());
     }
 }

注入信息

 <?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
         https://www.springframework.org/schema/beans/spring-beans.xsd">
 ​
     <bean id="address" class="com.rop.pojo.Address">
         <property name="address" value="123123@qq.com"/>
     </bean>
 ​
     <bean id="student" class="com.rop.pojo.Student">
         <!--第一种:普通值注入,value-->
         <property name="name" value="Mr.rop"/>
         <!--第二种:bean注入,ref-->
         <property name="address" ref="address"/>
         <!--第三种:数组注入,-->
         <property name="books">
             <array>
                 <value>三国演义</value>
                 <value>红楼梦</value>
                 <value>水浒传</value>
                 <value>西游记</value>
             </array>
         </property>
         <!--第四种:List-->
         <property name="hobbies">
             <list>
                 <value>听歌</value>
                 <value>篮球</value>
                 <value>游戏</value>
                 <value>跑步</value>
             </list>
         </property>
         <!--第五种:map-->
         <property name="cards">
             <map>
                 <entry key="手机号" value="12345678901"/>
                 <entry key="身份证" value="111111222222223333"/>
             </map>
         </property>
         <!--第六种:set-->
         <property name="games">
             <set>
                 <value>LOL</value>
                 <value>BOB</value>
                 <value>COC</value>
             </set>
         </property>
         <!--第七种:同第一种-->
         <property name="wife">
             <null/>
         </property>
         <!--第八种:properties-->
         <property name="info">
             <props>
                 <prop key="学号">123123</prop>
             </props>
         </property>
     </bean>
 </beans>

6.3、拓展方式注入

可以使用p命名空间和c命名空间进行注入

p命名空间

 xmlns:p="http://www.springframework.org/schema/p"

c命名空间

 xmlns:c="http://www.springframework.org/schema/c"

测试

 public class User {
     private int age;
     private String name;
 ​
     public User() {
     }
 ​
     public User(int age, String name) {
         this.age = age;
         this.name = name;
     }
 ​
     public int getAge() {
         return age;
     }
 ​
     public void setAge(int age) {
         this.age = age;
     }
 ​
     public String getName() {
         return name;
     }
 ​
     public void setName(String name) {
         this.name = name;
     }
 ​
     @Override
     public String toString() {
         return "User{" +
                 "age=" + age +
                 ", name='" + name + ''' +
                 '}';
     }
 }
 <!--p命名空间注入,可以直接注入属性的值:property,本质:set注入-->
 <bean id="user" class="com.rop.pojo.User" p:age="18" p:name="rop"/>
 <!--c命名空间注入,通过构造器注入-->
 <bean id="user2" class="com.rop.pojo.User" c:age="20" c:name="Mr.rop"/>
 @Test
 public void getUser(){
     ApplicationContext context = new ClassPathXmlApplicationContext("userbeans.xml");
     User user = context.getBean("user",User.class); //p命名空间注入
     User user2 = context.getBean("user2",User.class); //c命名空间注入
     System.out.println(user.toString());
     System.out.println();
     System.out.println(user2.toString());
 }

p命名空间和c命名空间不能直接使用,需要使用上面的约束。

6.4、bean的作用域

image-20220418190338324.png

  1. 单例模式(Spring默认机制)

     <bean id="user2" class="com.rop.pojo.User" c:age="20" c:name="Mr.rop" scope="singleton"/>
    
  2. 原型模式:每次从容器中get的时候,都会产生一个新对象

     <bean id="user2" class="com.rop.pojo.User" c:age="20" c:name="Mr.rop" scope="prototype"/>
    
  3. 其余的request、session、application这些个在web开发中使用到

7、Bean的自动装配

  • 自动装配,是Spring满足bean依赖的一种方式
  • Spring会在上下文中自动寻找,并自动给bean装配属性

在Spring中有三种装配的方式

  1. 在xml中显示的配置
  2. 在java中显示配置
  3. 隐式的自动装配【重要】

7.1、测试

环境搭建:一个人有两个宠物

7.2、ByName自动装配

 <!--
 byName:会自动在容器的上下文查找,与自己对象set方法后面的值对应的beanId
 -->
 <bean id="people" class="com.rop.pojo.People" autowire="byName">
     <property name="name" value="rop"/>
     <!--<property name="cat" ref="cat"/>-->
     <!--<property name="dog" ref="dog"/>-->
 </bean>

7.3、ByType自动装配

 <bean class="com.rop.pojo.Cat"/>
 <bean class="com.rop.pojo.Dog"/>
 <!--
 byType:会自动在容器的上下文查找,与自己对象属性类型相同的bean
 -->
 <bean id="people" class="com.rop.pojo.People" autowire="byType">
     <property name="name" value="rop"/>
     <!--<property name="cat" ref="cat"/>-->
     <!--<property name="dog" ref="dog"/>-->
 </bean>

小结:

  • byName的时候,需要保证所有bean 的id唯一且这个bean需要和自动注入的属性的set方法的值一致
  • byType的时候,需要保证所有bean 的class唯一且这个bean需要和自动注入的属性的类型一致

7.4、使用注解实现自动装配

使用注解须知:

  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
             https://www.springframework.org/schema/beans/spring-beans.xsd
             http://www.springframework.org/schema/context
             https://www.springframework.org/schema/context/spring-context.xsd">
     ​
         <context:annotation-config/>
     </beans>
    

@Autowired

直接在属性上使用即可!也可以在set方法上使用。

使用autowired后,可以选择不编写set方法,前提是这个自动装配的属性在IOC(Spring)容器中存在且优先符合byType。

 @Autowired
 ​
 private Cat cat;
 @Autowired
 ​
 private Dog dog;
 private String name;

如果@Autowired自动装配的环境比较复杂,自动装配无法通过一个注解【@Autowired】完成的时候,那就使用@Qualifier(value = "id名")去配合@Autowired使用,指定唯一的bean对象注入。

@Resource注解

 // @Autowired
 @Resource(name = "cat2")
 private Cat cat;
 // @Autowired
 @Resource
 private Dog dog;
 private String name;

小结:

@Resource与@Autowired的区别:

  • 都是用来自动装配的,都可以放在属性字段上
  • @Autowired默认通过byType的方式,当遇到多个同类型时,使用@Qualifier配合byName进行装配
  • @Resource默认通过byName方式,如果找不到名字,则通过byType实现。如果两个都找不到,就报错