草庐IT

day01-Spring基本介绍

liyuelian 2023-04-16 原文

Spring基本介绍

1.官方资料和下载

1.1Spring5下载

直接访问 https://repo.spring.io/ui/native/release/org/springframework/spring/,选择相应版本即可

  1. 进入官网 https://spring.io/

  2. 进入Spring5

  3. 进入Spring5 的github

    也可以直接访问 https://github.com/spring-projects/spring-framework

  4. 在github仓库README.md文件往下拉,看到Access to Binaries,点击里面的链接

  5. 在新页面往下拉,看到如下标题,点击链接

  6. 点击Artifacts

  7. 在左边的弹窗中依次点击release-->org-->springframework-->spring

  8. 在右边的窗口点击复制url

  9. 在浏览器新页面访问该地址,在页面中即可选择你想要下载的版本,我这里选择下载5.3.8版本

  10. 点击对应版本,在新页面中选择第一个,点击即可下载

1.2Spring文档

  • 在线文档

    Spring Framework Documentation

  • 离线文档

    解压缩后,在\spring-framework-5.3.8\docs\reference目录下分别提供了spring介绍文档的html和pdf版本

  • 离线API

    在\spring-framework-5.3.8\docs\javadoc-api\index.html中可以查看Spring的API

2.Spring学习的核心内容

  1. Spring核心学习内容:IOC,AOP,JDBCTemplate
  2. IOC:控制反转,可以管理java对象
  3. AOP:切面编程
  4. JDBCTemplate:是Spring提供的一套访问数据库的技术,应用性强,相对好理解
  5. 声明式事务:基于ioc/aop实现事务管理
  6. 其中ioc,aop是重点以及难点

3.Spring几个重要概念

  1. Spring可以整合其他的框架(Spring是管理框架的框架)
  2. Spring有两个核心的概念:IOC 和 AOP

  1. IOC [Inversion Of Control 反转控制]
  • 传统的开发模式 [ JdbcUtils/反射 ]

    程序------->环境 //程序读取环境配置,然后自己创建对象

    传统的开发模式:(以连接到数据库为例说明)

    1. 程序员编写程序,在程序中读取数据库配置信息

    2. 创建对象(反射或者new)

      各种对象,如:Connection,PreparedStatement,ResultSet等等

    3. 使用对象完成任务

  • IOC的开发模式

    程序<------容器 //容器创建好对象,程序直接使用

    如,现在有几个类:EmpAction EmpService EmpDao Emp

    传统的方式是通过手动new创建对象,然后在程序中使用

    现在,可以把要使用到的对象先配置到一个文件中(xml或者注解,这里以xml配置为例),这个文件可以理解成一个“容器文件”。配置好后,当spring启动以后,它就可以直接在程序中来获取容器创建好的对象,并进行使用:

    1. Spring根据配置文件xml/注解,创建对象,并放入到容器中(类似ConcurrentHashMap),并且可以完成对象之间的依赖(对象之间的依赖关系也在xml配置或者注解中完成)

      依赖:即对象间的引用关系。例如有A,B两个类。A类中的某个属性是B类,通过容器创建的两个类的对象a,b,它们之间的依赖/引用关系,将会自动完成(当然,也需要配置)

    2. 当需要使用某个对象实例时,直接从容器中获取即可

    3. 这样程序员可以更加专注于使用对象完成相应的业务

      这样创建对象的方式就从 new ===> 注解/配置方式


  1. DI(Dependency Injection 依赖注入),可以理解成是IOC的另外叫法

  2. Spring最大的价值:通过配置,给程序员提供需要使用的 web层[Servlet(Action/Controller)]/Service/Dao/JavaBean等对象。这是Spring的核心价值所在,也是ioc的具体体现,实现解耦

    • 原先的开发模式,Servlet是tomcat创建的,然后Servlet中如果你要用什么对象实例(如Service),就new一个,在Service中你要使用什么实例(如Dao),也同样是通过new的方式来创建实例……以此类推

    • 当使用Spring以后,web层的Servlet,Service层,Dao层,Javabean[entity]中的所有对象,我们都可以在配置文件中配置(或者通过注解指定),并且指定好对象间的依赖关系,放入到容器中。当这个流程结束后,我们想在程序中使用哪个对象,都可以直接在容器中直接获取。

4.Spring快速入门

4.1需求说明

通过Spring的方式[配置文件],获取JavaBean:Monster的对象,并给该对象的属性赋值,输出该对象信息

4.2完成步骤

  1. 下载spring 5 开发包

    详见1.1.1spring5下载

  2. 创建Java工程

    为了清晰spring5 的各个jar包的作用,这里使用Java工程

    image-20230114181550153

  3. 新建一个lib文件夹,引入开发spring5的基本包

    commons-logging.jar包需要另外下载,不在spring5包中

4.3代码实现

  1. 创建Javabean:Monster.java

    package com.li.bean;
    
    /**
     * @author 李
     * @version 1.0]
     * Javabean / Entity
     */
    public class Monster {
        private Integer monsterId;
        private String name;
        private String skill;
    
        //无参构造器一定要有,spring底层反射创建对象时需要使用
        public Monster() {
        }
    
        public Monster(Integer monsterId, String name, String skill) {
            this.monsterId = monsterId;
            this.name = name;
            this.skill = skill;
        }
    
        public Integer getMonsterId() {
            return monsterId;
        }
    
        public void setMonsterId(Integer monsterId) {
            this.monsterId = monsterId;
        }
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public String getSkill() {
            return skill;
        }
    
        public void setSkill(String skill) {
            this.skill = skill;
        }
    
        @Override
        public String toString() {
            return "Monster{" +
                    "monsterId=" + monsterId +
                    ", name='" + name + '\'' +
                    ", skill='" + skill + '\'' +
                    '}';
        }
    }
    
  2. 在src目录下创建配置文件:鼠标右击src目录--->new--->XML Configuration File--->Spring Config,我这里起名为beans.xml

  3. 创建好配置文件后,文件上方显示"Applilcation context not configured for this file",点击右边的Create Spring facet

    在弹出的窗口中直接点击右下方的ok

    然后重新点击文件的Create Spring facet

    在新窗口选中beans.xml文件,然后点击ok,之后文件就不再提示了。

  4. 在beans.xml文件中配置monster对象

    <?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">
        <!--
            1. 配置monster对象/javabean
            2. 在 beans标签中可以配置多个bean
            3. 一个bean就是一个Javabean对象
            4. class属性用于指定类的全路径->spring底层反射要用
            5. id属性表示该java对象在spring容器中的id(将来在程序中通过id在容器中获取相应对象,因此id是唯一的)
            6. <property name="monsterId" value="100"/> 用于给该对象的属性赋值,没有的话就是默认值
        -->
        <bean class="com.li.bean.Monster" id="monster01">
            <property name="monsterId" value="100"/>
            <property name="name" value="牛魔王"/>
            <property name="skill" value="芭蕉扇"/>
        </bean>
    
    </beans>
    
  5. 创建测试类SpringBeanTest

    package com.li.test;
    
    import com.li.bean.Monster;
    import org.springframework.context.ApplicationContext;
    import org.springframework.context.support.ClassPathXmlApplicationContext;
    import org.testng.annotations.Test;
    
    /**
     * @author 李
     * @version 1.0
     */
    public class SpringBeanTest {
    
        @Test
        public void getMonster() {
            // 1.创建容器 ApplicationContext
    
            // 2.这个容器是和配置文件关联的。也就是说,将来可能会有多个容器,因为配置文件可能会有多个
            ApplicationContext ioc = new ClassPathXmlApplicationContext("beans.xml");
    
            // 3.通过getBean获取对应的对象
            //   默认返回的是Object类型(编译类型),但是运行类型是Monster
            // Object monster01 = ioc.getBean("monster01");
            Monster monster01 = (Monster) ioc.getBean("monster01");//这样就可以在编译时获取对象属性
    
            //4.输出
            System.out.println("monster01=" + monster01 + " 运行类型=" + monster01.getClass());
            System.out.println("monster01.id=" + monster01.getMonsterId() +
                    " monster01.name=" + monster01.getName() +
                    " monster01.skill=" + monster01.getSkill());
    
            //5.相比于强转,也可以直接在获取的时候指定Class类型
            Monster monster011 = ioc.getBean("monster01", Monster.class);
            System.out.println("monster011=" + monster011);
            System.out.println("monster011.name=" + monster011.getName());
    
    
            System.out.println("ok~~~");
    
        }
    }
    

4.4注意事项和细节

类加载路径

一个问题:为什么下面可以读取到beans.xml文件?或者说,读取的文件是src目录下的beans.xml吗?

ApplicationContext ioc =new ClassPathXmlApplicationContext("beans.xml");

我们可以在测试类中输出一下路径:

//验证类加载路径
@Test
public void classPath() {
    File file = new File(this.getClass().getResource("/").getPath());
    System.out.println("file=" + file);
}

可以看到类的加载路径并不在src目录,而是在一个out目录下:

可以看到在out\production\spring目录下有一个beans.xml文件:

当运行后,会将src目录下编译好的class文件放到out目录下。同时,将资源文件(这里指beans.xml)也放到out目录,读取的时候是按照out目录来读取的。

因此运行时,真正读取的配置文件不是在src目录下的beans.xml,而是在out目录下的beans.xml,只是两个文件是一样的。(类的加载路径)


回到之前的问题,为什么下面的语句,我们直接写“beans.xml”?

ApplicationContext ioc =new ClassPathXmlApplicationContext("beans.xml");

因为默认读取的是.....\out\production\spring目录,而一旦运行过后,在src目录下的beans.xml文件会被直接放在.....\out\production\spring目录下,因此可以直接读取到。

假如beans.xml文件是放到src下面的某个子文件夹,那么在语句中就要根据子文件夹的结构来进行修改。

4.5.spring容器的结构/机制

注意配置debugger,通过配置指定哪一些数据在debug的时候会展示,哪一些不展示

1.如下,以4.3为例,打上断点,点击debug

2.如图所示,ioc就是我们创建的容器对象,是一个”重量级对象“,因为它的内容很多。因此创建该对象比较耗费资源,所以通常情况下,我们应该只有一个这样的对象。

容器对象中有一个重要的属性:beanFactory对象,beanFactory对象有一个beanDefinitionMap属性,该属性的类型是ConcurrentHashMap集合,用于保存配置文件的对象信息(注意不是保存对象,而是保存对象信息)

在我们的配置文件中,通常会有很多的java对象,spring会把这些java对象的信息保存下来,以便将来spring重复创建对象时使用

注意:这里的bean对象指的是任意Java对象,不仅仅是指Javabean!

4.5.1beanDefinitionMap保存Java对象的信息

3.点击展开beanDefinitionMap对象,可以看到一个table数组(ConcurrentHashMap$Node类型),初始的大小为512,Spring会将所有的Java对象的信息放到table中。

4.在这个例子中,table数组的index=127位置以ConcurrentHashMap$Node类型保存了Monster01对象信息(Node是ConcurrentHashMap的内部类)

其中key=“monster01”就是beans.xml中配置的bean的id,value存放了很多数据,例如:monster01对象的信息(属性,属性值,类全路径,是否懒加载)

非懒加载:在使用前已经创建好对象;懒加载:使用到时再动态创建对象

value的propertyValues就是记录beans.xml中配置的monster01对象的属性名和属性值。其中的elementData属性是一个Object类型的数组。点击展开elementData数组的一个元素,可以看到该元素记录了对象信息的name、value,即属性名称和属性值。

当spring再次创建对象时,就会到这里来获取对象的属性值

4.5.2singletonObjects保存Java对象

5.此外beanFactory还有一个重要属性singletonObjects,singletonObjects也是一个ConcurrentHashMap集合,singleonObjects有一个table数组,类型是ConcurrentHashMap$Node.

如下,在singleonObjects的table.index=217处存放了spring创建的monster对象

4.5.3beanDefinitionNames记录配置文件的bean名称

6.beanFactory还有一个beanDefinitionNames属性,存放配置文件的bean-id,主要是为了方便快速查找。


beanFactory的beanDefinitionMap用于保存Java对象的信息;

beanFactory的singletonObjects对象用于保存Java对象;

beanFactory的beanDefinitionNames用于记录配置文件的bean的名称(即Java对象的id)

4.5.4容器结构练习

如下在配置文件配置两个bean对象

练习:查看容器注入了哪些bean对象,输出bean的id

根据5.1-5.3的分析,获取bean的id可以输出beanDefinitionNames的属性值。

package com.li.test;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.testng.annotations.Test;

import java.io.File;

/**
 * @author 李
 * @version 1.0
 */
public class SpringBeanTest {

    @Test
    public void getMonster() {
        //创建容器 ApplicationContext
        ApplicationContext ioc =
                new ClassPathXmlApplicationContext("beans.xml");

        //查看容器注入了哪些bean对象,输出bean的id
        String[] beanDefinitionNames = ioc.getBeanDefinitionNames();
        for (String beanDefinitionName : beanDefinitionNames) {
            System.out.println("beanDefinitionName=" + beanDefinitionName);
        }
    }
}

输出如下:

有关day01-Spring基本介绍的更多相关文章

  1. ruby-on-rails - 带 Spring 锁的 Rails 4 控制台 - 2

    我正在使用Ruby2.1.1和Rails4.1.0.rc1。当执行railsc时,它被锁定了。使用Ctrl-C停止,我得到以下错误日志:~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.2/lib/spring/client/run.rb:47:in`gets':Interruptfrom~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.2/lib/spring/client/run.rb:47:in`verify_server_version'from~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.

  2. Unity 热更新技术 | (三) Lua语言基本介绍及下载安装 - 2

    ?博客主页:https://xiaoy.blog.csdn.net?本文由呆呆敲代码的小Y原创,首发于CSDN??学习专栏推荐:Unity系统学习专栏?游戏制作专栏推荐:游戏制作?Unity实战100例专栏推荐:Unity实战100例教程?欢迎点赞?收藏⭐留言?如有错误敬请指正!?未来很长,值得我们全力奔赴更美好的生活✨------------------❤️分割线❤️-------------------------

  3. 计算机毕业设计ssm+vue基本微信小程序的小学生兴趣延时班预约小程序 - 2

    项目介绍随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱小学生兴趣延时班预约小程序的设计与开发被用户普遍使用,为方便用户能够可以随时进行小学生兴趣延时班预约小程序的设计与开发的数据信息管理,特开发了小程序的设计与开发的管理系统。小学生兴趣延时班预约小程序的设计与开发的开发利用现有的成熟技术参考,以源代码为模板,分析功能调整与小学生兴趣延时班预约小程序的设计与开发的实际需求相结合,讨论了小学生兴趣延时班预约小程序的设计与开发的使用。开发环境开发说明:前端使用微信微信小程序开发工具:后端使用ssm:VU

  4. spring.profiles.active和spring.profiles.include的使用及区别说明 - 2

    转自:spring.profiles.active和spring.profiles.include的使用及区别说明下文笔者讲述spring.profiles.active和spring.profiles.include的区别简介说明,如下所示我们都知道,在日常开发中,开发|测试|生产环境都拥有不同的配置信息如:jdbc地址、ip、端口等此时为了避免每次都修改全部信息,我们则可以采用以上的属性处理此类异常spring.profiles.active属性例:配置文件,可使用以下方式定义application-${profile}.properties开发环境配置文件:application-dev

  5. ruby-on-rails - 使用 HTTParty 的非常基本的 Rails 4.1 API 调用 - 2

    Rails相对较新。我正在尝试调用一个API,它应该向我返回一个唯一的URL。我的应用程序中捆绑了HTTParty。我已经创建了一个UniqueNumberController,并且我已经阅读了几个HTTParty指南,直到我想要什么,但也许我只是有点迷路,真的不知道该怎么做。基本上,我需要做的就是调用API,获取它返回的URL,然后将该URL插入到用户的数据库中。谁能给我指出正确的方向或与我分享一些代码? 最佳答案 假设API为JSON格式并返回如下数据:{"url":"http://example.com/unique-url"

  6. ruby-on-rails - Rails 基本 Base64 身份验证 - 2

    我正在尝试复制此GETcurl请求:curl-D--XGET-H"Authorization:BasicdGVzdEB0YXByZXNlYXJjaC5jb206NGMzMTg2Mjg4YWUyM2ZkOTY2MWNiNWRmY2NlMTkzMGU="-H"Content-Type:application/json"http://staging.example.com/api/v1/campaigns在Ruby中,通过电子邮件+apikey生成身份验证:auth="Basic"+Base64::encode64("test@example.com:4c3186288ae23fd9661c

  7. ruby-on-rails - Spring 不起作用。 [未初始化常量 Spring::SID::DL] - 2

    我无法运行Spring。这是错误日志。myid-no-MacBook-Pro:myid$spring/Users/myid/.rbenv/versions/1.9.3-p484/lib/ruby/gems/1.9.1/gems/spring-0.0.10/lib/spring/sid.rb:17:in`fiddle_func':uninitializedconstantSpring::SID::DL(NameError)from/Users/myid/.rbenv/versions/1.9.3-p484/lib/ruby/gems/1.9.1/gems/spring-0.0.10/li

  8. ruby-on-rails - rails : Find tasks that were created on a certain day? - 2

    我有一个任务列表(名称、starts_at),我试图在每日View中显示它们(就像iCal)。deftodays_tasks(day)Task.find(:all,:conditions=>["starts_atbetween?and?",day.beginning,day.ending]end我不知道如何将Time.now(例如“2009-04-1210:00:00”)动态转换为一天的开始(和结束),以便进行比较。 最佳答案 deftodays_tasks(now=Time.now)Task.find(:all,:conditio

  9. 什么是0day漏洞?如何预防0day攻击? - 2

    什么是0day漏洞?0day漏洞,是指已经被发现,但是还未被公开,同时官方还没有相关补丁的漏洞;通俗的讲,就是除了黑客,没人知道他的存在,其往往具有很大的突发性、破坏性、致命性。0day漏洞之所以称为0day,正是因为其补丁永远晚于攻击。所以攻击者利用0day漏洞攻击的成功率极高,往往可以达到目的并全身而退,而防守方却一无所知,只有在漏洞公布之后,才后知后觉,却为时已晚。“后知后觉、反应迟钝”就是当前安全防护面对0day攻击的真实写照!为了方便大家理解,中科三方为大家梳理当前安全防护模式下,一个漏洞从发现到解决的三个时间节点:T0:此时漏洞即0day漏洞,是已经被发现,还未被公开,官方还没有相

  10. Simulink方法总结和避坑指南(一)——Simulink入门与基本调试方法 - 2

    文章目录一、项目场景二、基本模块原理与调试方法分析——信源部分:三、信号处理部分和显示部分:四、基本的通信链路搭建:四、特殊模块:interpretedMATLABfunction:五、总结和坑点提醒一、项目场景  最近一个任务是使用simulink搭建一个MIMO串扰消除的链路,并用实际收到的数据进行测试,在搭建的过程中也遇到了不少的问题(当然这比vivado里面的debug好不知道多少倍)。准备趁着这个机会,先以一个很基本的通信链路对simulink基础和相关的debug方法进行总结。  在本篇中,主要记录simulink的基本原理和基本的SISO通信传输链路(QPSK方式),计划在下篇记

随机推荐