大家好,我是【1+1=王】, 热爱java的计算机(人工智能)渣硕研究生在读。
如果你也对java、人工智能等技术感兴趣,欢迎关注,抱团交流进大厂!!!
Good better best, never let it rest, until good is better, and better best.
往期精彩:
- 【每天一个java设计模式(完)】 - 四万字实现23种设计模式(附示例源码)
- 【攻克java集合系列(完结)】Java集合全面总结
- 【CSDN 年终总结】结束与开始,一直在路上—— “1+1=王”的2021总结
近期会重新温习一下SSM的相关知识,相应的博客会更新至专栏【SSM框架】中,欢迎大家关注!
SSM专栏:https://blog.csdn.net/weixin_43598687/category_11652306.html
百度百科
Spring框架是一个开放源代码的J2EE应用程序框架,由Rod Johnson发起,是针对bean的生命周期进行管理的轻量级容器(lightweight container)。 Spring解决了开发者在J2EE开发中遇到的许多常见的问题,提供了功能强大IOC、AOP及Web MVC等功能。Spring可以单独应用于构筑应用程序,也可以和Struts、Webwork、Tapestry等众多Web框架组合使用,并且可以与 Swing等桌面应用程序AP组合。因此, Spring不仅仅能应用于J2EE应用程序之中,也可以应用于桌面应用程序以及小应用程序之中。Spring框架主要由七部分组成,分别是 Spring Core、 Spring AOP、 Spring ORM、 Spring DAO、Spring Context、 Spring Web和 Spring Web MVC。
简而言之,Spring是分层结构的java应用全栈轻量级开源框架,它以IoC(Inverse Of Control:控制反转,或者DI(Dependency Injection:依赖注入))和AOP(Aspect Oriented Programming:面向切面编程)为核心,提供展现层、持久层、事务层管理等技术,并且整合众多第三方框架和类库的开源框架。
回到到2002年,当时正是Java EE和EJB大行其道的时候,很多知名公司都是采用此技术方案进行项目开发。这时候Rod Johnson认为 EJB 太过臃肿,并不是所有的项目都需要使用 EJB 这种大型框架,应该会有一种更好的方案来解决这个问题。
他为了证明自己的想法是正确的,在2001年10月写了一本书《Expert One-on-One J2EE》,介绍了当时Java企业应用程序开发的情况,并指出了 Java EE 和 EJB 组件框架中存在的一些主要缺陷。在这本书中,他提出了一个基于普通 Java 类和依赖注入的更简单的解决方案。在书中,他展示了如何在不使用 EJB 的情况下构建高质量、可扩展的在线座位预留系统。为了构建应用程序,他编写了超过 30,000 行的基础结构代码,项目中的根包命名为 com.interface21,这就是 Spring 的前身。
2003 年 Rod Johnson 和同伴在此框架的基础上开发了一个全新的框架命名为 Spring。
2004 年 03 月,1.0 版发布。
2006 年 10 月,2.0 版发布。
2007 年 11 月,更名为 SpringSource,同时发布了 Spring 2.5。
2009 年 12 月,Spring 3.0 发布。
2013 年 12 月,Pivotal 宣布发布 Spring 框架 4.0。
2017 年 09 月,Spring 5.0 发布。
在进行下面的步骤之前,我们先打开idea创建一个新的maven工程。如下:
项目结构如下:
public interface UserDao {
public void save();
}
public class UserDaoImpl implements UserDao {
@Override
public void save() {
System.out.println("这是一个save方法");
}
}
<dependency>
<groupId>junitgroupId>
<artifactId>junitartifactId>
<version>4.11version>
<scope>testscope>
dependency>
@Test
public void test(){
UserDao userDao = new UserDaoImpl();
userDao.save();
}
<dependency>
<groupId>org.springframeworkgroupId>
<artifactId>spring-contextartifactId>
<version>5.0.5.RELEASEversion>
dependency>
public interface UserDao {
public void save();
}
public class UserDaoImpl implements UserDao {
@Override
public void save() {
System.out.println("这是一个save方法");
}
}
applicationContext.xml
,并将UserDaoImpl配置进applicationContext.xml。
<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">
<bean id="userDao" class="com.wang.dao.impl.UserDaoImpl">
bean>
beans>
关于applicationContext.xml
中具体配置的内容是什么意思,在下一部分会进行详细讲解。
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:applicationContext.xml");
UserDao userDao = (UserDao) applicationContext.getBean("userDao");
userDao.save();
将配置的对象交给Spring进行管理,默认调用类中的无参构造函数进行创建。
基本属性
scope的取值主要有singleton(单例)和prototype(多例)两种,默认取值为singleton。
scope="singleton"
时,多次通过ApplicationContext获得的Bean 实例是一样的。 <bean id="userDao" class="com.wang.dao.impl.UserDaoImpl" scope="singleton">
bean>
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:applicationContext.xml");
UserDao userDao1 = (UserDao) applicationContext.getBean("userDao");
System.out.println("userDao1:" + userDao1);
UserDao userDao2 = (UserDao) applicationContext.getBean("userDao");
System.out.println("userDao2:" + userDao2);
scope="prototype"
时,多次通过ApplicationContext获得的Bean 实例是不同的对象。 <bean id="userDao" class="com.wang.dao.impl.UserDaoImpl" scope="prototype">
bean>
singleton | prototype |
---|---|
实例化1个 | 实例化多个 |
当配置文件加载时实例化Bean | 当调用getBean()方法时实例化Bean |
创建容器时,创建对象 | 使用对象时,创建新的实例化对象 |
容器在,对象就存活 | 对象还在使用中,对象就存活 |
销毁容器时,对象被销毁 | 对象长时间不适用,垃圾回收器回收 |
// 创建工厂和静态方法
public class StaticFactory {
public static UserDao getUserDao(){
return new UserDaoImpl();
}
}
修改核心配置文件
<bean id="userDao" class="com.wang.factory.StaticFactory"
factory-method="getUserDao">
bean>
//创建工厂和非静态方法
public class InstanceFactory {
public UserDao getUserDao(){
return new UserDaoImpl();
}
}
修改核心配置文件
<bean id="factoryBean" class="com.wang.factory.InstanceFactory">bean>
<bean id="userDao" factory-bean="factoryBean" factory-method="getUserDao">bean>
上面的讲解中,对于Bean配置只是在DAO层,那如果我们加上业务层(Service、Controller)的需求该怎么做呢?
创建Service接口和实现类
public interface UserService {
public void save();
}
public class UserServiceImpl implements UserService {
@Override
public void save() {
ApplicationContext appContext = new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao)appContext.getBean("userDao");
userDao.save();
}
}
模拟controller
public class UserController {
public static void main(String[] args) {
UserService userService = new UserServiceImpl();
userService.save();
}
}
将userDao注入给userService
修改UserServiceImpl
public class UserServiceImpl implements UserService {
private UserDao userDao;
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
}
@Override
public void save() {
userDao.save();
}
}
修改核心配置文件
<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">
<bean id="userDao" class="com.wang.dao.impl.UserDaoImpl">bean>
<bean id="userService" class="com.wang.service.impl.UserServiceImpl">
<property name="userDao" ref="userDao">property>
bean>
beans>
修改模拟controller
public class UserController {
public static void main(String[] args) {
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = (UserService) applicationContext.getBean("userService");
userService.save();
}
}
UserServiceImpl中添加构造方法
public class UserServiceImpl implements UserService {
private UserDao userDao;
public UserServiceImpl() {
}
public UserServiceImpl(UserDao userDao) {
this.userDao = userDao;
}
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
}
@Override
public void save() {
userDao.save();
}
}
修改核心配置文件
<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">
<bean id="userDao" class="com.wang.dao.impl.UserDaoImpl">bean>
<bean id="userService" class="com.wang.service.impl.UserServiceImpl">
<constructor-arg name="userDao" ref="userDao">constructor-arg>
bean>
beans>
当我们的项目足够大的时候,Spring的配置内容非常多,这就导致Spring配置很繁杂且体积很大,所以,可以将部分配置拆解到其他配置文件中,而在Spring主配置文件通过import标签进行加载。
假设我们现在按照层次结构创建了多个配置文件,如下:
当有多个配置文件时,可以在主配置文件中通过import标签配置加载其他配置文件。
<?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">
<import resource="applicationContext-dao.xml"/>
<import resource="applicationContext-service.xml"/>
<import resource="applicationContext-web.xml"/>
</beans>
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:applicationContext.xml");
上面在通过核心配置文件获取对象实例的时候使用到了一个ApplicationContext类。
打开ApplicationContext类的源码我们可以看到,它其实是一个接口:
ClassPathXmlApplicationContext是ApplicationContext的一个实现类。它表示从类的根路径下加载配置文件。
除了ClassPathXmlApplicationContext之外,ApplicationContext还有两个实现类分别是:
getBean方法的参数主要可以有两种,分别是String、Class。
近期会重新温习一下SSM的相关知识,相应的博客会更新至专栏【SSM框架】中,欢迎大家关注!
SSM专栏:https://blog.csdn.net/weixin_43598687/category_11652306.html