Spring学习文档

1.spring

1.1 简介

<!-- https://mvnrepository.com/artifact/org.springframework/spring-webmvc -->
<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-webmvc</artifactId>
    <version>5.2.0.RELEASE</version>
</dependency>

<!-- https://mvnrepository.com/artifact/org.springframework/spring-webmvc -->
<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-webmvc</artifactId>
    <version>5.2.0.jdbcE</version>
</dependency>

1.2 优点

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

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

1.3 组成

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fjmUBArA-1602487011658)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20200928105443332.png)]

1.4 扩展

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

现在大多数公司都在使用Spring Boot进行快速开发,学习Spring Boot的前提,需要完全掌握Spring和

Spring MVC

2. IOC 理论推导

1.UserDao 接口

2.UserDaoimpl 实现类

3.UserService 业务接口

4.UserServiceimpl 业务实现

在我们之前的业务中,用户的需求可能会影响我们原来的代码,我们需要根据用户的需求去修改源代码,如果程序代码量十分大,修改一次的成本代价十分昂贵.

我们使用一个Set接口实现

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

这种思想,从本质上解决了问题,我们程序员不用再去管理对象的创建了,

系统的耦合性大大降低,可以更加专注业务的实现上!这是IOC的原型

IOC本质

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

个人认为的所谓的控制反转就是:获得依赖对象的方式反转了

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

3.HelloSpring

hello对象是由Spring创建的

hello对象的属性是由Spring容器设置的

控制反转

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

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

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

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

所以,我们彻底不用去程序中去改动了,只需要在xml配置文件中进行修改。所谓的ioc,就是:对象由Spring来创建,管理,装配-。

4.IOC创建对象的方式

1.使用无参数创建对象,默认。

2.假设我们要使用有参数构造创建对象

  • 下标赋值
    <bean id="user" class="com.kuang.pojo.User">
        <constructor-arg index="0" value="李治玮"/>
    </bean>
  • 通过类型创建(不推荐,类型可能相同)
	<bean id="user" class="com.kuang.pojo.User">
  	 	 <constructor-arg type="java.lang.String" value="LZW"/>
	</bean>
  • 直接通过参数名来创建(推荐)
    <bean id="user" class="com.kuang.pojo.User">
        <constructor-arg type="java.lang.String" value="LZW"/>
    </bean>

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

5.Spring配置

5.1 别名

    <alias name="user" alias="asdasd"/>

5.2 Been的配置

​ id:bean的唯一标识符,也就是相当于我们的变量名

​ class:bean对象所对应的全限定名:包名+类型

​ name:别名,且可以同时取多个别名

5.3 import

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

<import resource="beans.xml"/>
<import resource="beans2.xml"/>
<import resource="beans3.xml"/>

6.依赖注入

6.1 构造器注入

6.2 Set方式注入[重点]

  • 依赖注入:Set注入!
    • 依赖:bean对象创建依赖于容器
    • 注入:bean对象中的所有属性,由容器来注入

【环境搭建】

1.复杂类型

public class Address {
    private String address;

    public String getAddress() {
        return address;
    }

    public void setAddress(String address) {
        this.address = address;
    }
}

2.真实测试对象

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 wife;
    private Properties info;
}

3.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 http://www.springframework.org/schema/beans/spring-beans.xsd">
	<!--第一种:普通值注入,value-->
    <bean id="student" class="com.kuang.pojo.Student">
        <property name="name" value="李治玮"/>
    </bean>

</beans>

4.测试类

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.getName());
    }
}

6.3 拓展方式注入

​ 我们可以使用p空间命令和c空间命令来注入

    <!--p命名空间注入,可以直接注入属性的值,property-->
    <bean id="user" class="com.kuang.pojo.User" p:name="lzw" p:age="18"/>

    <!--c命名空间注入,通过构造器注入:construct-args-->
    <bean id="user2" class="com.kuang.pojo.User" c:name="lll" c:age="10"/>

测试

@Test
public void test2(){
    ApplicationContext context = new ClassPathXmlApplicationContext("userbeans.xml");
    User user = context.getBean("user", User.class);
    System.out.println(user.toString());
    }

注意点:p命名和c命名空间不能直接使用,需要导入xml约束

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

6.4 been的作用域

1.单例模式(默认)

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-qayWMSqx-1602487011664)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20200929145832116.png)]

    <bean id="user2" class="com.kuang.pojo.User" c:name="lll" c:age="10" scope="singleton"/>

2.原型模式:每次从容器中get的时候,都会产生一个新对象

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-wJZiEuRt-1602487011669)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20200929150957431.png)]

    <bean id="user2" class="com.kuang.pojo.User" c:name="lll" c:age="10" scope="prototype"/>

3.其他的request,session,application只能在web开发中使用到

7.Bean自动装配

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

在Spring中有三种装配的方式

1.在xml中显示的配置

2.在java中显示配置

3.隐式的自动装配bean(重要)

7.1 测试

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

7.2 ByName自动装配

    <bean id="people" class="com.kuang.pojo.Peaple" autowire="byName">
        <property name="name" value="lzw"/>
    </bean>

7.3 ByType自动装配

    <bean id="people" class="com.kuang.pojo.Peaple" autowire="byType">
        <property name="name" value="lzw"/>
    </bean>

小结

byName:会自动在容器上下文中查找,和自己对象set方法后面的值对应的bean id!

要保证所有的bean的id唯一,并且这个bean需要和自动注入的属性的set方法的值一致

byName:会自动在容器上下文中查找,和自己对象set方法后面的值对应的bean id!

要保证所有的bean的class唯一,并且这个bean需要和自动注入的属性的类型一致

7.4 使用注解实现自动装配

​ 要使用注解须知:

​ 1. 导入约束

​ 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>

@Auotowired

	@Autowired
	@Qualifier(value = "dog1")
	private Dog dog;

直接在属性上使用即可

如果@Autowired自动装配的环境比较复杂,自动装配无法通过一个注解【@Autowire】完成的时候,我们可以使用**@Qualifier(value=“xxx”)**去配置@Autowire的使用,指定一个唯一的bean使用

@Resource

    @Resource(name = "dog1")
    private Dog dog;

小结:

@Resource和@Autowired的区别

  • 都是用来自动装配的,都可以放在属性字段上
  • @Autowired通过byTepe的方式实现
  • @Resource默认通过byname的方式实现,如果找不到名字,则通过byType实现!如果两个都找不到的情况下,就报错!
  • 执行顺序不同

8. 使用注解开发

1.been

2.属性如何注入

@Component
public class User {
    public String name;
    //相当于 <bean name="name" class="com.kuang.pojo.User"/>
    @Value("111")
    public void setName(String name) {
        this.name = name;
    }
}

3.衍生的注解

@Component有几个衍生注解,我们在web开发中,会按照mvc三层架构分层!

  • dao 【@Repository】

  • service【@Service】

  • controller【@Controller】这四个注解功能都是一样的,都是代表将某个类注册到Spring中,装配Been

4.自动装配

- @Autowired :自动装配类型,名字
    如果Autowired不能唯一自动装配上属性,则需要通过@Qualifier(value="xxx")
- @Resource  :自动装配名字,类型
- @Nullable  :如果字段标记了这个注解,说明这个字段可以为null

5.小结

<!--开启注解支持-->
<context:annotation-config/>
<!--指定要扫描的包,这个包下的注解就会生效-->
<context:component-scan base-package="com.Lee"/>

​ xml与注解:

  • xml更加万能,适用于任何场合,维护方便
  • 注解不是自己的类用不了,维护相对复杂

​ xml与注解最佳实践:

  • xml用来管理Been

  • 注解只负责完成属性的注入

  • 我们在使用的过程中,只需要注意一个问题,必须让注解生效

    即:开启注解的支持

<!--开启注解支持-->
<context:annotation-config/>
<!--指定要扫描的包,这个包下的注解就会生效-->
<context:component-scan base-package="com.Lee"/>

9. 使用Java的方式配置Spring

完全不使用Spring的xml配置,全权交给java来做!

JavaConfig使Spring的子项目,在Spring4之后,成了核心功能

public class MyTest {
    public static void main(String[] args) {
        //如果完全使用了配置类方式去做,我们就只能通过AnnotationConfig上下文来获取容器,通过配置类的class对象加载
        ApplicationContext context = new AnnotationConfigApplicationContext(MyConfig.class);
        User getUser = (User) context.getBean("getUser");//getUser是Been的名字
        System.out.println(getUser.getName());
    }
}

10. 代理模式

代理模式就是SpringAOP的底层【SpringAOP 和 SpringMVC】

​ 代理模式的分类:

  • 静态代理
  • 动态代理

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-12PmSEdT-1602487011675)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20201010152420109.png)]

10.1 静态代理

角色分析:

  • 抽象角色:一般会使用接口或者抽象类来解决
  • 真实角色:被代理的角色
  • 代理角色:代理真实角色,代理真实角色后,我们一般会做一些附属操作
  • 客户:访问代理对象的人

代码步骤:

​ 1.接口

//租房
public interface Rent {
    public void rent();
}

​ 2.真实角色

//房东
public class Host implements Rent{
    public void rent() {
        System.out.println("房东要出租房子");
    }
}

​ 3.代理角色

public class Proxy implements Rent {
    private Host host;

    public void rent() {
        host.rent();
    }

    public Proxy() {
    }

    public Proxy(Host host) {
        this.host = host;
    }

    //看房
    public void seeHouse() {
        System.out.println("中介带你看房");
    }

    //收费
    public void fee() {
        System.out.println("收中介费");
    }

    //签合同
    public void contract() {
        System.out.println("签合同");
    }
}

​ 4.客户端访问代理角色

public class Client {
    public static void main(String[] args) {
        //房东要出租房子
        Host host = new Host();
        //代理
        Proxy proxy = new Proxy(host);
        proxy.rent();
        proxy.seeHouse();
        proxy.fee();
        proxy.contract();
    }
}

代理模式的好处:

  • 可以使真是角色的操作更加纯粹,不用去关注一些公共的业务

  • 公共也就交给代理角色!实现了业务的分工

  • 公共业务发生拓展的时候,方便集中管理

代理模式的缺点:

  • 一个真实角色就会产生一个代理角色,代码量会翻倍,开发效率会变低

10.2 动态代理

  • 动态代理和静态代理角色一样
  • 动态代理的代理代理类是动态生成的,不是我们直接写好的!
  • 动态代理分为两大类:基于接口的动态代理,基于类的动态代理
    • 基于接口 — JDK动态代理
    • 基于类:cglib
    • java字节码实现:javasist

需要了解两个类:

​ Proxy:代理

​ InvocationHandler:调用处理程序

InvocationHandler

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值