Spring5.0(IOC基于xml和注解)

Spring5.0

Spring5 框架概述

1、Spring 是轻量级的开源的 JavaEE 框架
2、Spring 可以解决企业应用开发的复杂性
3、Spring 有两个核心部分:IOC 和 Aop
(1)IOC:控制反转,把创建对象过程交给 Spring 进行管理
(2)Aop:面向切面,不修改源代码进行功能增强
4、Spring 特点
(1)方便解耦,简化开发
(2)Aop 编程支持
(3)方便程序测试
(4)方便和其他框架进行整合
(5)方便进行事务操作
(6)降低 API 开发难度

Spring入门案例

  1. 引入依赖
    引入spring框架的依赖和日志文件的依赖
    <!-- https://mvnrepository.com/artifact/org.springframework/spring-context -->
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-context</artifactId>
      <version>5.2.9.RELEASE</version>
    </dependency>
    <!-- https://mvnrepository.com/artifact/commons-logging/commons-logging -->
    <dependency>
      <groupId>commons-logging</groupId>
      <artifactId>commons-logging</artifactId>
      <version>1.2</version>
    </dependency>
    <dependency>
  1. 创建普通类
public class User {
    public void add() {
        System.out.println("spring第一次~");
    }
}
  1. 创建 Spring 配置文件,在配置文件配置创建的对象
    resource下创建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">

    <!--    配置user对象的创建-->
    <bean id="user" class="com.spring5.User"></bean>
</beans>
  1. 进行测试代码的编写
    ①加载spring配置文件
    ②获取配置创建的对象
public class TetsSpring5 {

    @Test
    public void testadd(){
        //加载spring配置文件
        BeanFactory applicationContext = new ClassPathXmlApplicationContext("bean1.xml");
        //获取配置创建的对象
        User user = applicationContext.getBean("user", User.class);

        System.out.println(user);
        user.add();
    }

IOC(概念和原理)

什么是ioc

(1)控制反转,把对象创建和对象之间的调用过程,交给 Spring 进行管理
(2)使用 IOC 目的:为了耦合度降低
(3)做入门案例就是 IOC 实现

ioc底层原理

xml 解析、工厂模式、反射
xml解析bean中的class,反射出对应的对象,通过工厂模式返回
在这里插入图片描述

BeanFactory 接口和ApplicationContext接口

1、IOC 思想基于 IOC 容器完成,IOC 容器底层就是对象工厂

2、Spring 提供 IOC 容器实现两种方式:(两个接口)
(1)BeanFactory:IOC 容器基本实现,是 Spring 内部的使用接口,不提供开发人员进行使用

  • 加载配置文件时候不会创建对象,在获取对象(使用)才去创建对象

(2)ApplicationContext:BeanFactory 接口的子接口,提供更多更强大的功能(国际化、访问资源,如URL和文件、载入多个(有继承关系)上下文 ,使得每一个上下文都专注于一个特定的层次,比如应用的web层、消息发送、响应机制、AOP),一般由开发人员进行使用

  • 加载配置文件时候就会把在配置文件对象进行创建

3、ApplicationContext 接口的实现类:
在这里插入图片描述
从名字上可以看出区别:
第一个是在系统盘中的路径

第二个是在resource(非maven是src)下的路径

IOC操作Bean管理

概念
什么是Bean管理

Bean 管理指的是两个操作
(1)Spring 创建对象
(2)Spirng 注入属性

Bean 管理操作有两种方式

(1)基于 xml 配置文件方式实现
(2)基于注解方式实现

基于 xml 方式
基于 xml 方式创建对象

在这里插入图片描述
(1)在 spring 配置文件中,使用 bean 标签,标签里面添加对应属性,就可以实现对象创建

(2)在 bean 标签有很多属性,常用的属性有:

  • id 属性:唯一标识
  • class 属性:类全路径(包类路径)

(3)创建对象时候,默认也是执行无参数构造方法完成对象创建

基于 xml 方式注入属性

DI:依赖注入,就是注入属性
DI就是ioc的实现

第一种注入方式:使用 set 方法进行注入

(1)创建类,定义属性和对应的 set 方法

/**
 * 演示使用 set 方法进行注入属性
 */
public class Book {
    //创建属性
    private String bname;
    private String bauthor;
    //创建属性对应的 set 方法
    public void setBname(String bname) {
        this.bname = bname;
    }
    public void setBauthor(String bauthor) {
        this.bauthor = bauthor;
    }
}

(2)在 spring 配置文件配置对象创建,配置属性注入

property 向set方法完成属性注入

  • name:类里面属性名称
  • value:向属性注入的值
  • ref:外部bean注入,创建另一个对象 bean 标签 id 值
    <!--    set方法注入属性-->
    <bean id="book" class="com.spring5.Book">
        <!--使用 property 完成属性注入
        name:类里面属性名称
        value:向属性注入的值
        -->
        <property name="bname" value="yzl"></property>
        <property name="bauthor" value="xml"></property>
    </bean>

调用:

    @Test
    public void testBook(){
        ApplicationContext applicationContext=new ClassPathXmlApplicationContext("bean1.xml");
        Book book = applicationContext.getBean("book", Book.class);
        System.out.println(book);
    }
第二种注入方式:使用有参构造器进行注入

(1)创建类,定义属性,创建属性对应有参数构造方法

/**
 * 使用有参数构造注入
 */
public class Order {
    //属性
    private String oname;
    private String address;
    //有参数构造
    public Order(String oname,String address) {
        this.oname = oname;
        this.address = address;
    }
}

(2)在 spring 配置文件中进行配置
constructor-arg向有参构造完成属性注入

  • name:类里面属性名称
  • value:向属性注入的值
  • index:构造器内属性值的序号
    <!--    有参构造注入属性-->
    <bean id="Order" class="com.spring5.Order">
        <constructor-arg name="oname" value="xml"></constructor-arg>
        <constructor-arg name="address" value="yzl"></constructor-arg>

<!--        <constructor-arg index="0" value="xml"></constructor-arg>-->
    </bean>

调用:

    @Test
    public void testOrder(){
        ApplicationContext applicationContext=new ClassPathXmlApplicationContext("bean1.xml");
        Order order = applicationContext.getBean("Order", Order.class);
        System.out.println(order);
    }
p 名称空间注入(了解)

使用 p 名称空间注入,可以简化基于 xml 配置方式,本质是set方法注入属性

第一步 添加 p 名称空间在配置文件中
在这里插入图片描述
第二步 进行属性注入,在 bean 标签里面进行操作

    <!--    set方法注入属性-->
    <bean id="book" class="com.spring5.Book" p:bname="xml" p:bauthor="yzl"></bean>
xml注入其他类型属性
字面量(null和特殊符号)
  1. null值
        <property name="bename">
            <null></null>
        </property>
  1. 属性值包含特殊符号
    1 把<>进行转义 &lt ;(左括号) &gt ;(右括号)
    2 把带特殊符号内容写到 CDATA(idea输入大写的CD)
        <property name="bauthor">
            <value>&lt;南京&gt;</value>
        </property>

			
		//或者
		<property name="bauthor">
            <value><![CDATA[
南京            
            ]]></value>
        </property>
注入属性-外部bean

示例:
(1)创建两个类 service 类和 dao 类
(2)在 service 调用 dao 里面的方法
(3)在 spring 配置文件中进行配置

dao:

public class Dao {
    public void add(){
        System.out.println("xmlsz");
    }
}

service:

public class Service {
    private Dao dao;

    public void setDao(Dao dao) {
        this.dao = dao;
    }

    public void add(){
        System.out.println("yzlss");
        dao.add();
    }
}

bean配置文件:

< property name=“dao” ref=“dao1”></ property>中ref是外部注入的关键

    <bean id="service" class="com.spring5.Service">
        <property name="dao" ref="dao1"></property>
    </bean>
    <bean id="dao1" class="com.spring5.Dao"></bean>

调用:

    @Test
    public void testService(){
        ApplicationContext applicationContext=new ClassPathXmlApplicationContext("bean1.xml");
        Service service = applicationContext.getBean("service", Service.class);
        service.add();
    }
注入属性-内部 bean

案例:
(1)一对多关系:部门和员工
一个部门有多个员工,一个员工属于一个部门
部门是一,员工是多
(2)在实体类之间表示一对多关系,员工表示所属部门,使用对象类型属性进行表示

//部门类
public class Dept {
    private String dname;

    public void setDname(String dname) {
        this.dname = dname;
    }

    @Override
    public String toString() {
        return "Dept{" +
                "dname='" + dname + '\'' +
                '}';
    }
}
//员工类
public class Emp {
    private String ename;
    private String gender;
    //员工属于某一个部门,使用对象形式表示
    private Dept dept;

    public void setDept(Dept dept) {
        this.dept = dept;
    }

    public void setEname(String ename) {
        this.ename = ename;
    }

    public void setGender(String gender) {
        this.gender = gender;
    }

    @Override
    public String toString() {
        return "Emp{" +
                "ename='" + ename + '\'' +
                ", gender='" + gender + '\'' +
                ", dept=" + dept +
                '}';
    }
}

在spring配置文件中进行配置:

<!--内部bean-->
    <bean id="emp" class="com.spring5.bean.Emp">
<!--        两个普通属性-->
        <property name="ename" value="yzl"></property>
        <property name="gender" value="xml"></property>
<!--        设置对象类型属性-->
        <property name="dept">
            <bean id="dept" class="com.spring5.bean.Dept">
                <property name="dname" value="xsr"></property>
            </bean>
        </property>
    </bean>
注入属性-级联赋值

(1)第一种写法:
其实也就是外部bean注入

<!--级联赋值-->
    <bean id="emp" class="com.spring5.bean.Emp">
<!--        两个普通属性-->
        <property name="ename" value="yzl"></property>
        <property name="gender" value="xml"></property>
<!--        设置对象类型属性-->
        <property name="dept" ref="dept1"></property>
    </bean>
    <bean id="dept1" class="com.spring5.bean.Dept">
        <property name="dname" value="xsr"></property>
    </bean>

(2)第二种写法:
在这里插入图片描述
重点是下面的 < property name=“dept.dname” value=“xsr1”></ property>
可以对对象进行赋值,进行这样的配置后,外部bean注入的属性值就会被这个覆盖

<!--级联赋值-->
    <bean id="emp" class="com.spring5.bean.Emp">
<!--        两个普通属性-->
        <property name="ename" value="yzl"></property>
        <property name="gender" value="xml"></property>
<!--        设置对象类型属性-->
        <property name="dept" ref="dept1"></property>
        <property name="dept.dname" value="xsr1"></property>
    </bean>
    <bean id="dept1" class="com.spring5.bean.Dept">
        <property name="dname" value="xsr"></property>
    </bean>
xml注入集合属性
注入数组、List、Map、Set集合类型属性

1、注入数组类型属性
2、注入 List 集合类型属性
3、注入 Map 集合类型属性
4、注入Set集合类型属性

(1)创建类,定义数组、list、map、set 类型属性,生成对应 set 方法

public class Stu {
    //数组类型属性
    private String[] course;

    //list集合类型属性
    private List<String> list;

    //map集合类型属性
    private Map<String, String> maps;

    //set集合类型属性
    private Set<String> set;

    public void setSet(Set<String> set) {
        this.set = set;
    }

    public void setCourse(String[] course) {
        this.course = course;
    }

    public void setList(List<String> list) {
        this.list = list;
    }

    public void setMaps(Map<String, String> maps) {
        this.maps = maps;
    }

    @Override
    public String toString() {
        return "Stu{" +
                "course=" + Arrays.toString(course) +
                ", list=" + list +
                ", maps=" + maps +
                ", set=" + set +
                ", list1=" + list1 +
                '}';
    }

}

(2)在 spring 配置文件进行配置

<!--    集合类型属性注入-->
    <bean id="stu" class="com.spring5.collectiontype.Stu">
<!--        数组类型注入-->
        <property name="course">
            <array>
                <value>1</value>
                <value>2</value>
                <value>3</value>
            </array>
        </property>
<!--        list类型属性注入-->
        <property name="list" >
            <list>
                <value>1</value>
                <value>2</value>
                <value>3</value>
            </list>
        </property>
<!--        map类型属性注入-->
        <property name="maps">
            <map>
                <entry key="1" value="1.1"></entry>
                <entry key="2" value="2.2"></entry>
                <entry key="3" value="3.3"></entry>
            </map>
        </property>
<!--        set类型属性注入-->
        <property name="set">
            <set>
                <value>2</value>
                <value>2</value>
                <value>3</value>
            </set>
        </property>

     </bean>

在集合里面设置对象类型值

java代码:

public class Stu {
    //学生所学的多门课程
    private List<Course> list1;

    public void setList1(List<Course> list1) {
        this.list1 = list1;
    }
}
public class Course {
    private String cname;

    public void setCname(String cname) {
        this.cname = cname;
    }

    @Override
    public String toString() {
        return "Course{" +
                "cname='" + cname + '\'' +
                '}';
    }
}

注入list集合类型,值是对象。通过ref注入外部的bean

	  <bean>
<!--        注入list集合类型,值是对象-->
        <property name="list1">
            <list>
                <ref bean="courser1"></ref>
                <ref bean="courser2"></ref>
            </list>
        </property>
     </bean>

<!--    创建多个Course对象-->
    <bean id="courser1" class="com.spring5.collectiontype.Course">
        <property name="cname" value="zzz"></property>
    </bean>
    <bean id="courser2" class="com.spring5.collectiontype.Course">
        <property name="cname" value="xxx"></property>
    </bean>
把集合注入部分提取出来

(1)在 spring 配置文件中引入名称空间 util
在这里插入图片描述

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

http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util.xsd"

(2)使用 util 标签完成 list 集合注入提取

<!--    提取list集合类型属性注入-->
    <util:list id="booklist">
        <value>xxx1</value>
        <value>xxx2</value>
        <value>xxx3</value>
    </util:list>

<!--    提取list集合类型属性注入使用-->
    <bean id="book" class="com.spring5.collectiontype.Book">
        <property name="list" ref="booklist"></property>
    </bean>
FactoryBean

1、Spring 有两种类型 bean,一种普通 bean,另外一种工厂 bean(FactoryBean)
2、普通 bean:在配置文件中定义 bean 类型就是返回类型
3、工厂 bean:在配置文件定义 bean 类型可以和返回类型不一样

第一步 创建类,让这个类作为工厂 bean,实现接口 FactoryBean
第二步 实现接口里面的方法,在实现的方法中定义返回的 bean 类型

public class MyBean implements FactoryBean<Course> {
    //定义返回bean,这个方法其实就是一个工厂模式
    @Override
    public Course getObject() throws Exception {
        Course course=new Course();
        course.setCname("abc");
        return course;
    }

    @Override
    public Class<?> getObjectType() {
        return null;
    }

    @Override
    public boolean isSingleton() {
        return false;
    }
}

<bean id="mybean" class="com.spring5.collectiontype.MyBean"></bean>

具体调用:
因为实现了FactoryBean接口,所以不能使用MyBean.class了,而是要用Course.class

    @Test
    public void test3(){
        ApplicationContext applicationContext=new ClassPathXmlApplicationContext("bean3.xml");
//        MyBean mybean = applicationContext.getBean("mybean", MyBean.class);
        Course mybean = applicationContext.getBean("mybean", Course.class);
        System.out.println(mybean);
    }
bean 作用域

1、在 Spring 里面,设置创建 bean 实例是单实例还是多实例
2、在 Spring 里面,默认情况下,bean 是单实例对象

默认情况下是单例的:

    @Test
    public void test2(){
        ApplicationContext applicationContext=new ClassPathXmlApplicationContext("bean2.xml");
        Book stu1 = applicationContext.getBean("book", Book.class);
        Book stu2 = applicationContext.getBean("book", Book.class);
        System.out.println(stu1==stu2);   //true
    }

如何设置单实例还是多实例
(1)在 spring 配置文件 bean 标签里面有属性(scope)用于设置单实例还是多实例
(2)scope 属性值(常用的两个)
第一个值 默认值,singleton,表示是单实例对象
第二个值 prototype,表示是多实例对象。(prototype意思是原型,有点深度clone的感觉)

    <bean id="book" class="com.spring5.collectiontype.Book" scope="prototype">
        <property name="list" ref="booklist"></property>
    </bean>
    @Test
    public void test2(){
        ApplicationContext applicationContext=new ClassPathXmlApplicationContext("bean2.xml");
        Book stu1 = applicationContext.getBean("book", Book.class);
        Book stu2 = applicationContext.getBean("book", Book.class);
        System.out.println(stu1==stu2);   //false
    }

(3)singleton 和 prototype 区别
第一 singleton 单实例,prototype 多实例
第二 设置 scope 值是 singleton 时候,加载 spring 配置文件时候就会创建单实例对象
设置 scope 值是 prototype 时候,不是在加载 spring 配置文件时候创建 对象,在调用getBean 方法时候创建多实例对象(懒汉式的感觉)

补充:
scope除了singleton 和 prototype 还有几个值
request、session和global session

  • request
    request,session和global session类型只实用于web程序,通常是和XmlWebApplicationContext共同使用。
    Spring容器,即XmlWebApplicationContext 会为每个HTTP请求创建一个全新的RequestPrecessor对象,当请求结束后,该对象的生命周期即告结束,如同java web中request的生命周期。当同时有100个HTTP请求进来的时候,容器会分别针对这10个请求创建10个全新的RequestPrecessor实例,且他们相互之间互不干扰,简单来讲,request可以看做prototype的一种特例,除了场景更加具体之外,语意上差不多。

  • session
    对于web应用来说,放到session中最普遍的就是用户的登录信息,对于这种放到session中的信息,我们可以使用如下形式的制定scope为session:
    Spring容器会为每个独立的session创建属于自己的全新的UserPreferences实例,比request scope的bean会存活更长的时间,其他的方面没区别,如果java web中session的生命周期。

  • global session
    global session只有应用在基于porlet的web应用程序中才有意义,它映射到porlet的global范围的session,如果普通的servlet的web 应用中使用了这个scope,容器会把它作为普通的session的scope对待。

bean 生命周期

生命周期:从对象创建到对象销毁的过程

bean 生命周期:
(1)通过构造器创建 bean 实例(无参数构造)
(2)为 bean 的属性设置值和对其他 bean 引用(调用 set 方法)
(3)调用 bean 的初始化的方法(需要进行配置初始化的方法)
(4)bean 可以使用了(对象获取到了)
(5)当容器关闭时候,调用 bean 的销毁的方法(需要进行配置销毁的方法)

简述为:

  • 实例化 Instantiation
  • 属性赋值 Populate
  • 初始化 Initialization
  • 销毁 Destruction

示例如下:
自定义一个类来观察

public class Orders {
    public Orders() {
        System.out.println("第一步 执行无参数构造创建 bean 实例");
    }

    private String oname;

    public void setOname(String oname) {
        this.oname = oname;
        System.out.println("第二步 调用 set 方法设置属性值");
    }

    //自定义的初始化方法
    public void init(){
        System.out.println("第三步 执行初始化的方法");
    }

    //自定义的销毁方法
    public void destroy(){
        System.out.println("第五步 执行销毁的方法");
    }
}

spring配置文件:
init-method为初始化调用的方法
destroy-method为销毁调用的方法

    <bean id="orders" class="com.spring5.collectiontype.Orders" init-method="init" destroy-method="destroy">
        <property name="oname" value="zzt"></property>
    </bean>

调用bean完成测试:

    @Test
    public void test4(){
//        ApplicationContext applicationContext=new ClassPathXmlApplicationContext("bean4.xml");
        ClassPathXmlApplicationContext context=new ClassPathXmlApplicationContext("bean4.xml");
        Orders orders = context.getBean("orders", Orders.class);
        System.out.println("第四步 获取创建 bean 实例对象");
        System.out.println(orders);
        //手动关闭bean实例
        context.close();
    }

结果是:
在这里插入图片描述
如果调用bean的后置处理器,那么bean的生命周期有7步:
(1)通过构造器创建 bean 实例(无参数构造)
(2)为 bean 的属性设置值和对其他 bean 引用(调用 set 方法)
(3)把 bean 实例传递 bean 后置处理器的方法 postProcessBeforeInitialization
(4)调用 bean 的初始化的方法(需要进行配置初始化的方法)
(5)把 bean 实例传递 bean 后置处理器的方法 postProcessAfterInitialization
(6)bean 可以使用了(对象获取到了)
(7)当容器关闭时候,调用 bean 的销毁的方法(需要进行配置销毁的方法)

注意:实现了该接口后,配置完成后,不需要调用,默认给整个bean文件下的所有bean生效

演示添加后置处理器效果:示例:
(1)创建类,实现接口 BeanPostProcessor,创建后置处理器

public class BeanPost implements BeanPostProcessor {
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("在初始化之前执行的方法");
        return bean;
    }

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("在初始化之后执行的方法");
        return bean;
    }
}

(2)在spring中配置

<bean id="beanpost" class="com.spring5.collectiontype.BeanPost"></bean>

效果如下:
在这里插入图片描述

xml 自动装配
什么是自动装配

根据指定装配规则(属性名称或者属性类型),Spring 自动将匹配的属性值进行注入

自动装配过程
 bean 标签属性 autowire,配置自动装配
 autowire 属性常用两个值:
 byName 根据属性名称注入 ,注入值 bean 的 id 值和类属性名称一样
 byType 根据属性类型注入

(1)根据属性类型自动注入

    <!--实现自动装配
     bean 标签属性 autowire,配置自动装配
     autowire 属性常用两个值:
     byName 根据属性名称注入 ,注入值 bean 的 id 值和类属性名称一样
     byType 根据属性类型注入
    -->
    <bean id="userservice" class="com.spring5.service.UserService" autowire="byType"></bean>
    <bean id="userdaoimpl" class="com.spring5.dao.UserDaoImpl"></bean>

(2)根据属性名称自动注入
上面如果存在多个实例的时候,就需要指定名字注入
byName 根据属性名称注入 ,注入值 bean 的 id 值和类属性名称一样

    <bean id="userservice" class="com.spring5.service.UserService" autowire="byName"></bean>
    <bean id="userdaoimpl" class="com.spring5.dao.UserDaoImpl"></bean>
    <bean id="userDao" class="com.spring5.dao.UserDaoImpl1"></bean>

以上两个的java代码:

public class UserService {
    private UserDao userDao;

    public void setUserDao(UserDao userDao) {
        this.userDao = userDao;
    }

    public void add() {
        System.out.println("service add...");
        userDao.add();
    }
}
public interface UserDao {
    public void add();
}
public class UserDaoImpl implements UserDao{
    @Override
    public void add() {
        System.out.println("UserDao ...");
    }
}
public class UserDaoImpl1 implements UserDao{
    @Override
    public void add() {
        System.out.println("UserDao ...");
    }
}
外部属性文件

1、直接配置数据库信息
(1)配置德鲁伊连接池

    <bean id="datesource" class="com.alibaba.druid.pool.DruidDataSource">
        <property name="driverClassName" value="com.mysql.jdbc.Driver"></property>
        <property name="url"
                  value="jdbc:mysql://localhost:3306/douyin"></property>
        <property name="username" value="root"></property>
        <property name="password" value="root"></property>
    </bean>

(2)引入德鲁伊连接池依赖 jar 包

    <!-- https://mvnrepository.com/artifact/com.alibaba/druid -->
    <dependency>
      <groupId>com.alibaba</groupId>
      <artifactId>druid</artifactId>
      <version>1.1.23</version>
    </dependency>

2、引入外部属性文件配置数据库连接池

  • 引入context名称空间
    在这里插入图片描述
  • 建立一个properties文件
url=jdbc:mysql://localhost:3306/douyin
username=root
password=root
driverClassName=com.mysql.jdbc.Driver


initialSize=10
maxActive=20
minIdle=10
maxWait=5000
  • 在 spring 配置文件使用标签引入外部属性文件
<!--    引入外部属性文件-->
    <context:property-placeholder location="classpath:druid.properties"></context:property-placeholder>
<!--    配置连接池-->
    <bean id="druiddatasource" class="com.alibaba.druid.pool.DruidDataSource">
        <property name="driverClassName" value="${driverClassName}"></property>
        <property name="url" value="${url}"></property>
        <property name="username" value="${username}"></property>
        <property name="password" value="${password}"></property>
    </bean>
基于注解的方式
什么是注解

(1)注解是代码特殊标记,格式:@注解名称(属性名称=属性值, 属性名称=属性值…)
(2)使用注解,注解作用在类上面,方法上面,属性上面
(3)使用注解目的:简化 xml 配置

Spring 针对 Bean 管理中创建对象提供注解

@Component, @Service, @Controller, @Repository是spring注解,注解后可以被spring框架所扫描并注入到spring容器来进行管理

@Component是通用注解,其他三个注解是这个注解的拓展,并且具有了特定的功能
@Repository注解在持久层中,具有将数据库操作抛出的原生异常翻译转化为spring的持久层异常的功能。
@Controller层是spring-mvc的注解,具有将请求进行转发,重定向的功能。
@Service层是业务逻辑层注解,这个注解只是标注该类处于业务逻辑层。

用这些注解对应用进行分层之后,就能将请求处理,义务逻辑处理,数据库操作处理分离出来,为代码解耦,也方便了以后项目的维护和开发。

  • 上面四个注解功能是一样的,都可以用来创建 bean 实例
基于注解方式实现对象创建

第一步 引入依赖
第二步 开启组件扫描

<!--    开启组件扫描
        1.如果扫描多个包,多个包使用逗号隔开(base-package="com.spring5.dept,com.spring5.emp")
        2.扫描包的上层目录
-->
    <context:component-scan base-package="com.spring5"></context:component-scan>

第三步 创建类,在类上面添加创建对象注解(随便一个都行)

//在注解里面value值可以不写
//默认是类名称,首字母小写
//UserService---->userService
@Component(value = "userService")  //<bean id="userService" class="..">
//@Service
//@Controller
//@Repository
public class UserService {
    public void add() {
        System.out.println("service add...");
    }
}

public class TestDemo {
    @Test
    public void test1(){
        ApplicationContext applicationContext=new ClassPathXmlApplicationContext("bean1.xml");
        UserService userService = applicationContext.getBean("userService", UserService.class);
        userService.add();
    }
}

开启组件扫描细节配置

不自动扫描,自己决定扫描哪里
use-default-filters="false"表示现在不使用默认filter,自己配置filter
context:include-filter 设置扫描哪些内容

<!--    use-default-filters="false"表示现在不使用默认filter,自己配置filter-->
<!--    context:include-filter 设置扫描哪些内容-->
    <context:component-scan base-package="com.spring5" use-default-filters="false">
        <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>

自动扫描,自己决定不扫描哪里
context:exclude-filter:设置哪些内容不进行扫描

<!--    扫描包下的所有内容-->
<!--    context:exclude-filter:设置哪些内容不进行扫描-->
    <context:component-scan base-package="com.spring5">
        <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>
基于注解方式实现属性注入

默认的value是类名的首字母小写!!!!

@Autowired

@Autowired:根据属性类型进行自动装配
示例:
第一步 把 service 和 dao 对象创建,在 service 和 dao 类添加创建对象注解
第二步 在 service 注入 dao 对象,在 service 类添加 dao 类型属性,在属性上面使用注解
原来我们需要先生成set方法,然后使用外部bean注入,ref引进来,现在只需要一个autowired注解

public interface UserDao {
    public void add();
}
@Repository
public class UserDaoImpl implements UserDao{
    @Override
    public void add() {
        System.out.println("UserDao ...");
    }
}
@Service
public class UserService {
    @Autowired
    private UserDao userDao;
    public void add() {
        System.out.println("service add...");
        userDao.add();
    }
}
@Qualifier

@Qualifier:根据名称进行注入
这个@Qualifier 注解的使用,和上面@Autowired 一起使用

示例:
如果在上面的代码中加入这个类,那么UserDao就有两个实现类了,这时候在使用属性注入,spring就不知道具体是哪个

@Repository
public class UserDaoImpl1 implements UserDao{
    @Override
    public void add() {
        System.out.println("UserDao1 ...");
    }
}

所以我们可以加上这个注解来表面具体是哪个子类,运用如下

@Service
public class UserService {
    @Autowired
    @Qualifier(value = "userDaoImpl1")
    private UserDao userDao;
    public void add() {
        System.out.println("service add...");
        userDao.add();
    }
}

@Resource

@Resource:可以根据类型注入,可以根据名称注入,不加参数是类型配置,加了是名称配置

@Service
public class UserService {
//    @Resource //类型注入
    @Resource(name = "userDaoImpl")   //名称注入
    private UserDao userDao;
    public void add() {
        System.out.println("service add...");
        userDao.add();
    }
}

注意:resource是java的原生注解,autowired是spring的注解,所以推荐使用autowired注解搭配Qualifier注解一起使用
在这里插入图片描述
在这里插入图片描述

@Value

@Value:注入普通类型属性

@Component
public class ValueTest {
    @Value("abc")
    private String name;

    public String getName() {
        return name;
    }
}

同样,和上文提到的外部属性文件来装配bean,我们也可以用这个value来实现。
有@Value(“${}”)和@Value(“#{}”)两种方式

① ${ property : default_value }
② #{ obj.property? :default_value }
第一个注入的是外部配置文件对应的property
第二个则是SpEL表达式对应的内容。 那个default_value,就是前面的值为空时的默认值。注意二者的不同,#{}里面那个obj代表对象
#{}需要get set ${}不需要

@Value(“${}”)就是,从下面的配置文件中读取他的 server.port 属性
在这里插入图片描述
在这里插入图片描述

@Value(“#{}”)就是别人实现了get和set方法,可以直接调用别人的
在这里插入图片描述
在这里插入图片描述

完全注释开发

上面可以看到,对于xml文件,我们其实只用到了一个context来扫描包,那么我们可不可以把xml也省去,用全注解的方式开发呢?是可以的

  • @Configuration:作为配置类,替代 xml 配置文件
  • @ComponentScan(basePackages = {“com.spring5”}):扫描包的范围
    等同于
<context:component-scan base-package="com.spring5"></context:component-scan>

(1)创建配置类,替代 xml 配置文件

@Configuration //作为配置类,替代 xml 配置文件
@ComponentScan(basePackages = {"com.atguigu"})
public class SpringConfig {
}

(2)编写测试类
原本的ApplicationContext 实现类是ClassPathXmlApplicationContext来加载xml文件
现在如果采用全注解,实现类是AnnotationConfigApplicationContext

    @Test
    public  void test3(){
        //加载配置类
        ApplicationContext applicationContext=new AnnotationConfigApplicationContext(SpringConfig.class);
        UserService userService = applicationContext.getBean("userService", UserService.class);
        userService.add();
    }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值