day01-Spring基本介绍
Spring基本介绍
1.官方资料和下载
1.1Spring5下载
直接访问 https://repo.spring.io/ui/native/release/org/springframework/spring/,选择相应版本即可
-
进入官网 https://spring.io/
-
进入Spring5
-
进入Spring5 的github
-
在github仓库README.md文件往下拉,看到Access to Binaries,点击里面的链接
-
在新页面往下拉,看到如下标题,点击链接
-
点击Artifacts
-
在左边的弹窗中依次点击release-->org-->springframework-->spring
-
在右边的窗口点击复制url
-
在浏览器新页面访问该地址,在页面中即可选择你想要下载的版本,我这里选择下载5.3.8版本
-
点击对应版本,在新页面中选择第一个,点击即可下载
1.2Spring文档
-
在线文档
-
离线文档
解压缩后,在\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学习的核心内容
- Spring核心学习内容:IOC,AOP,JDBCTemplate
- IOC:控制反转,可以管理java对象
- AOP:切面编程
- JDBCTemplate:是Spring提供的一套访问数据库的技术,应用性强,相对好理解
- 声明式事务:基于ioc/aop实现事务管理
- 其中ioc,aop是重点以及难点
3.Spring几个重要概念
- Spring可以整合其他的框架(Spring是管理框架的框架)
- Spring有两个核心的概念:IOC 和 AOP
- IOC [Inversion Of Control 反转控制]:
-
传统的开发模式 [ JdbcUtils/反射 ]
程序------->环境 //程序读取环境配置,然后自己创建对象
传统的开发模式:(以连接到数据库为例说明)
-
程序员编写程序,在程序中读取数据库配置信息
-
创建对象(反射或者new)
各种对象,如:Connection,PreparedStatement,ResultSet等等
-
使用对象完成任务
-
-
IOC的开发模式
程序<------容器 //容器创建好对象,程序直接使用
如,现在有几个类:EmpAction EmpService EmpDao Emp
传统的方式是通过手动new创建对象,然后在程序中使用
现在,可以把要使用到的对象先配置到一个文件中(xml或者注解,这里以xml配置为例),这个文件可以理解成一个“容器文件”。配置好后,当spring启动以后,它就可以直接在程序中来获取容器创建好的对象,并进行使用:
-
Spring根据配置文件xml/注解,创建对象,并放入到容器中(类似ConcurrentHashMap),并且可以完成对象之间的依赖(对象之间的依赖关系也在xml配置或者注解中完成)
依赖:即对象间的引用关系。例如有A,B两个类。A类中的某个属性是B类,通过容器创建的两个类的对象a,b,它们之间的依赖/引用关系,将会自动完成(当然,也需要配置)
-
当需要使用某个对象实例时,直接从容器中获取即可
-
这样程序员可以更加专注于使用对象完成相应的业务
这样创建对象的方式就从 new ===> 注解/配置方式
-
-
DI(Dependency Injection 依赖注入),可以理解成是IOC的另外叫法
-
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完成步骤
-
下载spring 5 开发包
详见1.1.1spring5下载
-
创建Java工程
为了清晰spring5 的各个jar包的作用,这里使用Java工程
-
新建一个lib文件夹,引入开发spring5的基本包
commons-logging.jar包需要另外下载,不在spring5包中
4.3代码实现
-
创建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 + '\'' + '}'; } }
-
在src目录下创建配置文件:鼠标右击src目录--->new--->XML Configuration File--->Spring Config,我这里起名为beans.xml
-
创建好配置文件后,文件上方显示"Applilcation context not configured for this file",点击右边的Create Spring facet
在弹出的窗口中直接点击右下方的ok
然后重新点击文件的Create Spring facet
在新窗口选中beans.xml文件,然后点击ok,之后文件就不再提示了。
-
在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>
-
创建测试类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);
}
}
}
输出如下: