Spring

About Spring

开源免费框架,轻量级,非入侵式框架。Spring就是一个轻量级的控制反转(IOC)和面向切片编程(AOP)的框架

Maven repo:Spring Web MVC + spring-jdbc(整合Mybatis)

  1. <!-- https://mvnrepository.com/artifact/org.springframework/spring-webmvc -->
  2. <dependency>
  3. <groupId>org.springframework</groupId>
  4. <artifactId>spring-webmvc</artifactId>
  5. <version>5.3.9</version>
  6. </dependency>
  7. <!-- https://mvnrepository.com/artifact/org.springframework/spring-jdbc -->
  8. <dependency>
  9. <groupId>org.springframework</groupId>
  10. <artifactId>spring-jdbc</artifactId>
  11. <version>5.3.9</version>
  12. </dependency>

Spring两大特点

控制反转(IOC)

面向切片编程(AOP)支持事务处理

Spring

About IOC

控制反转:IOC是一种设计思想,通过描述(XML或注解)并通过第三方生产或获取对象的方式。之前对象的创建与对象的依赖关系完全在java硬编码程序中,控制权在程序;实现IOC后,控制权在第三方,实现降藕。

在Spring中实现控制反转的是IoC容器,实现方法是依赖注入DI(Dependency Injection,DI)

引用狂神的一个例子简单理解下

  1. private UserDao userDao = null;
  2. public UserServiceImpl(){
  3. userDao = new UserDaoImpl();
  4. }

在之前我们使用JavaWeb写service=>dao的时候,是通过如上的方式去实现的,项目构建大概如下

那么如果此时我的UserDao接口又了一个新的实现类暂且为UserDaoImpls,这个实现类中有新的功能实现,那么就需要到UserServiceImpl中再去构造方法加一段如下的代码:

  1. userDao = new UserDaoImpls();

那么这时如果该项目还没发布那到无所谓,如果是已经上线的项目是不可能这样重新去修改代码的,或者如果有n个new,就要修改n处。

解决这个问题就是通过一个set方法。如下:

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

在构造方法中实例化对象的这个操作,改为利用set封装并将需要new的实现类的名称变为set方法的参数,实现用户可控的去new一个新的实现类从而添加新的功能展示到页面。

而这个思想就是控制反转(IOC)的原型,将new实现类对象的主动权交给了用户而不是程序,从本质上解决了上面的问题,也实现了降藕。

Hello Spring

Hello.java

  1. package com.zh1z3ven.pojo;
  2. public class Hello {
  3. private String str;
  4. public Hello() {
  5. }
  6. public Hello(String str) {
  7. this.str = str;
  8. }
  9. public String getStr() {
  10. return str;
  11. }
  12. public void setStr(String str) {
  13. this.str = str;
  14. }
  15. @Override
  16. public String toString() {
  17. return "Hello{" +
  18. "str='" + str + '\'' +
  19. '}';
  20. }
  21. }

Beans.xml

一个bean标签代表一个对象, id代表在Spring中这个类要实例化生成的对象的名字, class指定这个实体类

设置对象的属性值

引用Spring容器中创建的对象

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans
  5. https://www.springframework.org/schema/beans/spring-beans.xsd">
  6. <!--使用Spring创建对象,Spring都称之为bean-->
  7. <!-- 一个bean标签代表一个对象, id代表要实例化的对象的名字, class指定这个实体类-->
  8. <bean id="hello" class="com.zh1z3ven.pojo.Hello">
  9. <!-- 设置实体类的属性值-->
  10. <property name="str" value="Spring"/>
  11. </bean>
  12. </beans>

Test.java

在Spring中也存在一个上下文,通过ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");传入xml配置文件名字来获取该xml文件的上下文对象。利用ApplicationContext#getBean()传入在xml中配置的该类的id获取该实体类对象。

  1. public class MyTest {
  2. public static void main(String[] args) {
  3. //获取Spring上下文对象
  4. ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
  5. //从Spring中取出对象
  6. Hello hello = (Hello) context.getBean("hello");
  7. System.out.println(hello);
  8. }
  9. }

IOC创建对象的几种方式

PS:在配置文件加载的时候,通过bean标签注册的对象就已经在Spring IoC容器中初始化了。

总体来说就两种方式:

  • 无参构造

    默认使用

  • 有参构造

    1. 下标赋值

      1. <!--通过下标赋值-->
      2. <bean id="hello2" class="com.zh1z3ven.pojo.Hello">
      3. <constructor-arg index="0" value="Spring2"/>
      4. </bean>
    2. 类型复制

      1. <!--通过类型赋值-->
      2. <bean id="hello3" class="com.zh1z3ven.pojo.Hello">
      3. <constructor-arg type="java.lang.String" value="Spring3"/>
      4. </bean>
    3. 属性名赋值

      1. <!--属性名赋值-->
      2. <bean id="hello4" class="com.zh1z3ven.pojo.Hello">
      3. <constructor-arg name="str" value="Spring4"/>
      4. </bean>

Spring import settings

import标签可导入其他beans.xml配置文件,而applicationContext.xml到时可作为总bean的配置文件,而不需要导入多个xml

Dependency Injection

依赖注入(Dependency Injection,DI)

PS:一定需要pojo中实现set才可以利用bean标签注入

  1. 构造器注入

    也就是上面提到的创建对象的方式,分为无参构造和有参构造

    • 无参构造

      默认使用

    • 有参构造

      1. 下标赋值

        1. <!--通过下标赋值-->
        2. <bean id="hello2" class="com.zh1z3ven.pojo.Hello">
        3. <constructor-arg index="0" value="Spring2"/>
        4. </bean>
      2. 类型复制

        1. <!--通过类型赋值-->
        2. <bean id="hello3" class="com.zh1z3ven.pojo.Hello">
        3. <constructor-arg type="java.lang.String" value="Spring3"/>
        4. </bean>
      3. 属性名赋值

        1. <!--属性名赋值-->
        2. <bean id="hello4" class="com.zh1z3ven.pojo.Hello">
        3. <constructor-arg name="str" value="Spring4"/>
        4. </bean>
  2. set注入

    依赖:bean对象的注入依赖于Spring容器

    注入:bean对象的属性,由容器来注入

  3. 拓展注入

    Student.java

    1. public class Student {
    2. private String name;
    3. private Address address;
    4. private String[] books;
    5. private List<String> hobbys;
    6. private Map<String,String> card;
    7. private Set<String> games;
    8. private String wife;
    9. private String apache;
    10. private Properties info;
    11. public String getApache() {
    12. return apache;
    13. }
    14. public void setApache(String apache) {
    15. this.apache = apache;
    16. }
    17. public String[] getBooks() {
    18. return books;
    19. }
    20. public void setBooks(String[] books) {
    21. this.books = books;
    22. }
    23. public List<String> getHobbys() {
    24. return hobbys;
    25. }
    26. public void setHobbys(List<String> hobbys) {
    27. this.hobbys = hobbys;
    28. }
    29. public Map<String, String> getCard() {
    30. return card;
    31. }
    32. public void setCard(Map<String, String> card) {
    33. this.card = card;
    34. }
    35. public Set<String> getGames() {
    36. return games;
    37. }
    38. public void setGames(Set<String> games) {
    39. this.games = games;
    40. }
    41. public String getWife() {
    42. return wife;
    43. }
    44. public void setWife(String wife) {
    45. this.wife = wife;
    46. }
    47. public Properties getInfo() {
    48. return info;
    49. }
    50. public void setInfo(Properties info) {
    51. this.info = info;
    52. }
    53. public String getName() {
    54. return name;
    55. }
    56. public void setName(String name) {
    57. this.name = name;
    58. }
    59. public Address getAddress() {
    60. return address;
    61. }
    62. public void setAddress(Address address) {
    63. this.address = address;
    64. }
    65. @Override
    66. public String toString() {
    67. return "Student{" +
    68. "name='" + name + '\'' +
    69. ", address=" + address +
    70. ", books=" + Arrays.toString(books) +
    71. ", hobbys=" + hobbys +
    72. ", card=" + card +
    73. ", games=" + games +
    74. ", wife='" + wife + '\'' +
    75. ", info=" + info +
    76. '}';
    77. }
    78. }

    beans.xml

    1. <bean id="address" class="com.zh1z3ven.pojo.Address">
    2. <property name="address" value="beijing"/>
    3. </bean>
    4. <bean id="student1" class="com.zh1z3ven.pojo.Student">
    5. <!-- 普通属性值注入-->
    6. <property name="name" value="zh1z3ven"/>
    7. <!-- ref 引用对象注入-->
    8. <property name="address" ref="address"/>
    9. <!-- 数组array注入-->
    10. <property name="books">
    11. <array>
    12. <value>红楼梦</value>
    13. <value>西游记</value>
    14. <value>水浒传</value>
    15. <value>三国演义</value>
    16. </array>
    17. </property>
    18. <!-- List注入-->
    19. <property name="hobbys">
    20. <list>
    21. <value>听音乐</value>
    22. <value>看电影</value>
    23. <value>敲代码</value>
    24. <value>写文章</value>
    25. </list>
    26. </property>
    27. <!-- Map注入-->
    28. <property name="card">
    29. <map>
    30. <entry key="银行卡" value="1"></entry>
    31. <entry key="身份证" value="2"></entry>
    32. <entry key="学生证" value="3"></entry>
    33. <entry key="电话卡" value="4"></entry>
    34. <entry key="校园卡" value="5"></entry>
    35. </map>
    36. </property>
    37. <!-- Set注入-->
    38. <property name="games">
    39. <set>
    40. <value>LOL</value>
    41. <value>CF</value>
    42. <value>qq</value>
    43. </set>
    44. </property>
    45. <!-- null注入-->
    46. <property name="wife">
    47. <null></null>
    48. </property>
    49. <!-- 空值注入-->
    50. <property name="apache" value=""/>
    51. <!-- properties-->
    52. <property name="info">
    53. <props>
    54. <prop key="id">10</prop>
    55. <prop key="city">北京</prop>
    56. </props>
    57. </property>
    58. </bean>

P-namespcae&C-namespace

  1. p命名空间注入,可以直接注入属性值,类似于bean标签中property-name-value

    Beans.xml头部需导入

    1. xmlns:p="http://www.springframework.org/schema/p"
    1. <bean id="user1" class="com.zh1z3ven.pojo.User" p:name="zh1z3ven1" p:age="18"/>
  2. c命名空间注入,通过构造器注入,类似于construct-args(需要实现有参构造)

    1. xmlns:c="http://www.springframework.org/schema/c"
    1. <bean id="user2" class="com.zh1z3ven.pojo.User" c:name="zh1z3ven2" p:age="19"/>
  1. @Test
  2. public void test(){
  3. ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
  4. User user1 = context.getBean("user1", User.class);
  5. System.out.println(user1.getName());
  6. System.out.println(user1.getAge());
  7. User user2 = context.getBean("user2", User.class);
  8. System.out.println(user2.getName());
  9. System.out.println(user2.getAge());
  10. }

Bean scopes

bean的作用域

singleton

默认bean为scope = singleton单例模式运行的

显示定义:<bean id="user1" class="com.zh1z3ven.pojo.User" p:name="zh1z3ven1" p:age="18" scope="singleton"/>

单例模式,共享一个对象,比如如下例子,getBean指向的是同一个bean,那么在Spring IoC容器中仅仅会生成一个"user2"对象保存在内存中,当调用ApplicationContext.getBean("user2")时返回该对象

  1. <bean id="user2" class="com.zh1z3ven.pojo.User" c:name="zh1z3ven2" c:age="19"/>
  1. @Test
  2. public void test(){
  3. ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
  4. User user1 = context.getBean("user2", User.class);
  5. User user2 = context.getBean("user2", User.class);
  6. System.out.println(user1==user2);
  7. }

prototype

原型模式prototype与singleton不同,每次上下文去getBean()时都会在Spring IoC容器内创建一次该对象

还是拿上面的测试代码,可以发现已经不是同一个对象了,有点类似于多态

  1. public void test2(){
  2. ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
  3. User user1 = context.getBean("user2", User.class);
  4. User user2 = context.getBean("user2", User.class);
  5. System.out.println(user1==user2);
  6. }

而其余的生命周期在Web中才会遇到。

Bean的自动装配

  • 在xml显示配置bean
  • 在java代码中配置bean
  • 隐式自动装配bean【autowire】

byName autowire

会在容器上下文中自动查找,和自己对象set方法后面的值对应的beanid。也就是这里会去上下文中寻找有无"cat"这个beanid,有则自动装配,如果并没有匹配上,比如此时beanid被修改为了"cat123" 就会跑出异常。

byType autowire

会在容器上下文中寻找该类型与属性值所对应的类型相同的bean,基于bean中的class(需要在上下文中所有类型各自只出现一次)

  1. <bean id="people" class="com.zh1z3ven.pojo.People" autowire="byType">
  2. <property name="name" value="zh1z3ven"/>
  3. </bean>

小结

  • byname需要保证所有bean的id唯一,且这个bean的id的值要和需要自动装配依赖注入的set方法的值一致。
  • bytype需要保证所有bean的class唯一,且这个bean的class的值需要和set方法的值的类型一致。

注解实现自动装配

xml配置

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xmlns:context="http://www.springframework.org/schema/context"
  5. xsi:schemaLocation="http://www.springframework.org/schema/beans
  6. https://www.springframework.org/schema/beans/spring-beans.xsd
  7. http://www.springframework.org/schema/context
  8. https://www.springframework.org/schema/context/spring-context.xsd">
  9. <context:annotation-config/>
  10. </beans>

@Autowired

默认使用byname方式去自动装配,它可以对类成员变量、方法及构造函数进行标注,完成自动装配的工作。

在使用@Autowired时,首先在容器中查询对应类型的bean(bytype)

如果查询结果刚好为一个,就将该bean装配给@Autowired指定的数据

如果查询的结果不止一个,那么@Autowired会根据名称来查找。(byname)

如果查询的结果为空,那么会抛出异常。解决方法时,使用@Autowried(required=false)

  1. public class People {
  2. @Autowired
  3. private Cat cat;
  4. @Autowired
  5. private Dog dog;
  6. private String name;

@Qualifier

如果存在多个类型且该类型有多个不同名字的对象,那么只用@Autowired会找不到该对象,可以配合@Qualifier(value="")来指定一个装配的值。

  1. @Autowired
  2. @Qualifier(value="dog222")

@Resource

java自带的一个注解,和@Autowired,@Qualifier组合用法和效果基本是一样

  1. javax.annotation.Resource
  2. @Resource //不指定名称自动装配
  3. @Resource(name="") //指定名称自动装配

使用注解开发

bean在xml里注册,属性值通过注解注入

@component

泛指各种组件,把普通pojo实例化到spring容器中,相当于配置文件中的bean,将该类在配置文件下注册到Spring容器中装配bean。类似的还有:

1、@controller 控制器(注入服务)

用于标注控制层,相当于struts中的action层

2、@service 服务(注入dao)

用于标注服务层,主要用来进行业务的逻辑处理

3、@repository(实现dao访问)

用于标注数据访问层,也可以说用于标注数据访问组件,即DAO组件.

@Scope

生命周期,用法:在目标类上面声明

  1. @Scope("singleton")

@Configuration

用于声明这是一个配置类

@Bean

相当于在配置文件中注册bean

方法名为之前的id属性

方法返回值为之前的class属性

@Import

导入其他配置类,等价于

  1. <import resource="beans.xml"/>

使用方法

  1. @Import(Config.class)

示例代码

pojo

  1. //相当于在bean中注册,相当于在Spring IoC容器new一个对象
  2. @Component
  3. public class User {
  4. @Value("CoLoo")
  5. public String name;
  6. public String getName() {
  7. return name;
  8. }
  9. public void setName(String name) {
  10. this.name = name;
  11. }
  12. @Override
  13. public String toString() {
  14. return "User{" +
  15. "name='" + name + '\'' +
  16. '}';
  17. }
  18. }

config

  1. @Configuration
  2. @ComponentScan("com.zh1z3ven.pojo")
  3. public class AppConfig {
  4. @Bean
  5. public User getUser(){
  6. return new User();
  7. }
  8. }

test

  1. public class MyTest {
  2. public static void main(String[] args) {
  3. ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
  4. User user = context.getBean("user", User.class);
  5. System.out.println(user.getName());
  6. }
  7. }

Java之Spring基础与IoC的更多相关文章

  1. Spring基础之IOC

    一.ioc能解决什么问题 1.Spring是什么 spring是以ioc和aop为核心,能整合第三方框架和类库的企业级应用开源框架. 2.程序的耦合问题 例子:Driver类必须存在,编译才通过,Jd ...

  2. Spring基础——在 IOC 容器中 Bean 之间的关系

    一.在 Spring IOC 容器中 Bean 之间存在继承和依赖关系. 需要注意的是,这个继承和依赖指的是 bean 的配置之间的关系,而不是指实际意义上类与类之间的继承与依赖,它们不是一个概念. ...

  3. 01—Spring基础配置IOC

  4. 第65节:Java后端的学习之Spring基础

    Java后端的学习之Spring基础 如果要学习spring,那么什么是框架,spring又是什么呢?学习spring中的ioc和bean,以及aop,IOC,Bean,AOP,(配置,注解,api) ...

  5. Spring笔记:IOC基础

    Spring笔记:IOC基础 引入IOC 在Java基础中,我们往往使用常见关键字来完成服务对象的创建.举个例子我们有很多U盘,有金士顿的(KingstonUSBDisk)的.闪迪的(SanUSBDi ...

  6. (转)java之Spring(IOC)注解装配Bean详解

    java之Spring(IOC)注解装配Bean详解   在这里我们要详细说明一下利用Annotation-注解来装配Bean. 因为如果你学会了注解,你就再也不愿意去手动配置xml文件了,下面就看看 ...

  7. JAVA模拟Spring实现IoC过程(附源码)

    前言:本人大四学生,第一次写博客,如果有写得不好的地方,请大家多多指正 一.IoC(Inversion of Control)反转控制 传统开发都是需要对象就new,但这样做有几个问题: 效率低下,创 ...

  8. 【已转移】【Java架构:基础技术】一篇文章搞掂:Spring

    本文篇幅较长,建议合理利用右上角目录进行查看(如果没有目录请刷新). 本文是对<SPRING实战第4版>的总结,大家也可以去仔细研读该书 [------------------------ ...

  9. Spring基础知识

    Spring基础知识 利用spring完成松耦合 接口 public interface IOutputGenerator { public void generateOutput(); } 实现类 ...

随机推荐

  1. 一道内部ctf文件包含题

    拿到题目 在burp里看下 拿到源码 很明显是一道文件包含题目,包含cookie里的值,于是构造Cookie:language=chinese试试   文件变成中文的了,说明中文语言进行了包含并替换 ...

  2. Linux中ftp服务器搭建

    一.FTP工作原理 (1)FTP使用端口 [root@localhost ~]# cat /etc/services | grep ftp ftp-data 20/tcp #数据链路:端口20 ftp ...

  3. 小刻也能看懂的Unraid系统使用手册:基础篇

    小刻也能看懂的Unraid系统使用手册 基础篇 Unraid系统简介 Unraid 的本体其实是 Linux,它主要安装在 NAS 和 All in One 服务器上,经常可以在 Linus 的视频里 ...

  4. VS Code 下载安装并设置中文面板显示

    下载: 下载地址:https://code.visualstudio.com/ 微软在2015年4月30日Build 开发者大会上正式宣布了 Visual Studio Code 项目:一个运行于 M ...

  5. Java基础00-第一个程序2

    1. 常用DOS命令 1.1 打开命令提示窗口 按下win+R 输入cmd 按下回车键 得到命令提示窗口 1.2 常用命令 2. Path环境变量的配置 2.1 为什么要配置Path环境变量 2.2 ...

  6. shell脚本(3)-格式化输出

    一个程序需要有0个或以上的输入,一个或更多输出 一.echo语法 1.功能:将内容输出到默认显示设备. echo命令功能在显示器上显示一段文字,一般提到提示的作用 2.语法:echo[-ne][字符串 ...

  7. Python+Requests+Xpath(解析)爬取某站点简历图片(数据分析三)

    1.环境安装 pip install lxml 2.解析原理 使用通用爬虫爬取网页数据 实例化etree对象,且将页面数据加载到该对象中 使用xpath函数结合xpath表达式进行标签定位和指定数据提 ...

  8. Album++:分布式事务专辑-基础概念

    (一)基础概念:↓ ↓ ↓ 1.1)什么是事务 什么是事务?举个生活中的例子:你去小卖铺买东西,"一手交钱,一手交货"就是一个事务的例子,交钱和交货必 须全部成功, 事务才算成功, ...

  9. PAT乙级:1053 住房空置率 (20分)

    PAT乙级:1053 住房空置率 (20分) 题干 在不打扰居民的前提下,统计住房空置率的一种方法是根据每户用电量的连续变化规律进行判断.判断方法如下: 在观察期内,若存在超过一半的日子用电量低于某给 ...

  10. BZOJ2457 双端队列 题解

    本题直接求解十分困难,因为在不知道整个序列的数字规律时当前所作决策都无法保证最优性. 考虑正难则反,题目转化为将一个非降序列分成尽量少的几段,让每段对应原问题的双端队列. 先将原数组排序,由于原数组下 ...