我们为什么要学习Spring框架?
Spring技术是JavaEE开发必备技能,企业开发技术选型命中率>90%
专业角度
企业级开发:Spring使用率>90%
简化开发
框架整合
目前我们使用的是Spring几版本?
spring家族:
通过系统架构图,Spring能不能进行数据层开发?Spring能不能进行web层开发?
4.x架构图趋于成熟
Aspect不是spring原创,是别人的东西spring觉得非常之好,拿过来直接用,所以导包时要单独导Aspect包(依赖)
问题1:目前我们的代码存在什么问题以及怎么解决这些问题?
问题2:请描述什么是IOC,什么是DI?
IOC(Inversion of Control)控制反转
使用对象时,由主动new产生对象转换为由外部提供对象,此过程中对象创建控制权由程序转移到外部,此思想称为控制反转。通俗的讲就是“将new对象的权利交给Spring,我们从Spring中获取对象使用即可”
Spring技术对IoC思想进行了实现
Spring提供了一个容器,称为IOC容器,用来充当IoC思想中的“外部”
IOC容器负责对象的创建、初始化等一系列工作,被创建或被管理的对象在IoC容器中统称为Bean
IOC容器也叫Spring容器,就是上图中的Core Container
IoC容器充当Ioc思想中的“外部”就是说:主动new对象改成由IoC容器提供对象
DI(Dependency Injection)依赖注入
service和bean都在IoC容器中被管理,他们之间又有关系,那么IoC容器不妨直接将他们之间关系绑定好,这个过程成为:DI
高内聚,低耦合,一切的一切就是要减少类与类(代码与代码)之间的关系,但是类与类之间又有着不可分割的关系,那么全部不能写死,全部关系变成动态的有关,代码都抽象到接口层面,都不写死,就充分解耦了。最后维护时改了A不影响B,改了B不影响C…
【第一步】导入Spring坐标
【第二步】定义Spring管理的类(接口)
【第三步】创建Spring配置文件,配置对应类作为Spring管理的bean对象
【第四步】初始化IOC容器(Spring核心容器/Spring容器),通过容器获取bean对象
先创建module
创建目录结构如下:
最终目录结构:
注意新项目maven可能变了,需要修改下配置:
【第一步】导入Spring坐标
org.springframework spring-context 5.2.10.RELEASE
第一次导入直接复制,然后刷新一下maven让其下载,然后下次就会有代码提示了
【第二步】定义Spring管理的类(接口)
public interface BookDao {public void save();
}public class BookDaoImpl implements BookDao {public void save() {System.out.println("book dao save ...");}
}
public interface BookService {public void save();
}public class BookServiceImpl implements BookService {private BookDao bookDao = new BookDaoImpl();public void save() {System.out.println("book service save ...");bookDao.save();}
}
【第三步】创建Spring配置文件,配置对应类作为Spring管理的bean对象
先导spring坐标,然后resources根目录下右键创建
注意事项:bean定义时id属性在同一个上下文中(IOC容器中)不能重复
【第四步】初始化IOC容器(Spring核心容器/Spring容器),通过容器获取Bean对象
public class App {public static void main(String[] args) {//1. 创建IoC容器对象,加载spring核心配置文件ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");//2. 在IoC容器里获取Bean对象//2.1 拿daoBookDao bookDao = (BookDao) ioc.getBean("bookDao");bookDao.save(); // 正常执行方法//2.2 拿serviceBookService bookService = (BookService) ioc.getBean("bookService");bookService.save(); // 正常执行方法}
}
【第一步】删除使用new的形式创建对象的代码
【第二步】提供依赖对象对应的setter方法
【第三步】配置service与dao之间的关系
【第一步】删除使用new的形式创建对象的代码
public class BookServiceImpl implements BookService {private BookDao bookDao; //【第一步】删除使用new的形式创建对象的代码public void save() {System.out.println("book service save ...");bookDao.save();}
}
【第二步】提供依赖对象对应的setter方法
public class BookServiceImpl implements BookService {private BookDao bookDao;public void save() {System.out.println("book service save ...");bookDao.save();}//【第二步】提供依赖对象对应的setter方法public void setBookDao(BookDao bookDao) {this.bookDao = bookDao;}
}
【第三步】配置service与dao之间的关系
在applicationContext.xml中配置
DI注入两个bookDao都指的啥
问题1:在
问题2:Bean的默认作用范围是什么?如何修改?
见上面《IOC入门案例》applicationContext.xml配置
见上面《IOC入门案例》运行结果
下面图片中的com.itheima我都换成了cn.whu
name作用域范围很大,甚至可以取代id作为ref值
扩展:scope的取值不仅仅只有singleton和prototype,还有request、session、application、 websocket ,表示创建出的对象放置在web容器(tomcat)对应的位置。比如:request表示保存到request域中。
不修改配置,默认情况下单例,打印的对象地址值完全一样
最后给大家说明一下:在我们的实际开发当中,绝大部分的Bean是单例的,也就是说绝大部分Bean不需要配置scope属性。
比如dao,之前new一次调用一个方法,下次再new再调用方法,完全不必要,复用一个对象就行了的,有参数也可以一个对象方法传递不同参数,springIOC帮我们管理的就是这类可以复用的对象 都是单例模式,很轻量化
Bean的实例化方式有几种?
bean本质上就是对象,创建bean使用构造方法完成
之前的module复制一份,然后稍作修改(service直接删除 这个知识点不需要)
public class BookDaoImpl implements BookDao {public BookDaoImpl() {System.out.println("book dao constructor is running ....");}public void save() {System.out.println("book dao save ...");}
}
构造方法private也都能正常创建,说明内部是用反射实现的
public class AppForInstanceBook {public static void main(String[] args) {ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");BookDao bookDao = (BookDao) ctx.getBean("bookDao");bookDao.save();}
}
注意:无参构造方法如果不存在,将抛出异常BeanCreationException
(比如给上面构造方法加个参数,再执行就会报错)
public interface OrderDao {public void save();
}
public class OrderDaoImpl implements OrderDao {public void save() {System.out.println("order dao save ...");}
}
cn.whu.factory.OrderDaoFactory
//静态工厂创建对象
public class OrderDaoFactory {public static OrderDao getOrderDao(){System.out.println("factory setup....");return new OrderDaoImpl();}
}
public class AppForInstanceOrder {public static void main(String[] args) {ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");OrderDao orderDao = (OrderDao) ctx.getBean("orderDao");orderDao.save();}
}
public class AppForInstanceOrder {public static void main(String[] args) {ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");//OrderDao orderDao = (OrderDao) ioc.getBean("orderDao");//orderDao.save();}
}
public interface UserDao {public void save();
}
public class UserDaoImpl implements UserDao {public void save() {System.out.println("user dao save ...");}
}
//实例工厂创建对象
public class UserDaoFactory {public UserDao getUserDao(){System.out.println("UserDaoFactory ....");return new UserDaoImpl();}
}
前面的配置都注释了
区别:静态工厂直接写类名
实例工厂得先创建工厂对象
public class AppForInstanceUser {public static void main(String[] args) {// //创建实例工厂对象// UserDaoFactory userDaoFactory = new UserDaoFactory();// //通过实例工厂对象创建对象// UserDao userDao = userDaoFactory.getUserDao();// userDao.save();ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");UserDao userDao = (UserDao) ctx.getBean("userDao");userDao.save();}
}
UserDaoFactoryBean中实例化什么类型的对象泛型就是该类型。
//FactoryBean创建对象
public class UserDaoFactoryBean implements FactoryBean {//代替原始实例工厂中创建对象的方法//好处:以后工厂获取对象的方法定死了,统统都叫getObject,不需要变了public UserDao getObject() throws Exception {return new UserDaoImpl();}//对象Class类型public Class> getObjectType() {return UserDao.class;}
}
其他配置都注释了
使用之前的AppForInstanceUser测试类去运行看结果就行了。注意配置文件中id="userDao"是否重复。
public class AppForInstanceUser {public static void main(String[] args) {ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");UserDao userDao1 = (UserDao) ioc.getBean("userDao");UserDao userDao2 = (UserDao) ioc.getBean("userDao");System.out.println(userDao1);System.out.println(userDao2);}
}
如何改成多例:继续重写一个方法isSingleton()
public boolean isSingleton() {return false;//多例//return true;//单例
}
问题1:多例的Bean能够配置并执行销毁的方法?
问题2:如何做才执行Bean销毁的方法?
复制一下之前的module,修改成如下形式:
public class BookDaoImpl implements BookDao {public void save() {System.out.println("book dao save~");}//自定义两个无参方法,用来监视生命周期(applicationContext.xml中配置下即可)//表示bean初始化对应的操作public void init(){System.out.println("init...");}//表示bean销毁前对应的操作public void destroy(){System.out.println("destroy...");}
}
public class AppForLifeCycle {public static void main( String[] args ) {//此处需要使用实现类类型,接口类型没有close方法ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");BookDao bookDao = (BookDao) ctx.getBean("bookDao");bookDao.save();//关闭容器,执行销毁的方法ctx.close();}
}
还有一种温和的关闭虚拟机的方式,为IoC容器注册关闭钩子registerShutdownHook,也就是让虚拟机在IoC容器彻底关闭了之后才关闭:
public static void main(String[] args) {ClassPathXmlApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");ioc.registerShutdownHook();BookDao bookDao = (BookDao) ioc.getBean("bookDao");bookDao.save();//ioc.close();
}
执行结果同上~
约定优于配置: 按照人家的约定来写代码,就不用配置了(不用配置init-method和destroy-method了)
public class BookServiceImpl implements BookService, InitializingBean, DisposableBean {private BookDao bookDao;public void setBookDao(BookDao bookDao) {System.out.println("set .....");this.bookDao = bookDao;}public void save() {System.out.println("book service save ...");bookDao.save();}public void destroy() throws Exception {System.out.println("service destroy");}public void afterPropertiesSet() throws Exception {System.out.println("service init");}
}
ClassPathXmlApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");
ioc.registerShutdownHook();
//容器加载就会创建对象 方法执不执行无所谓
分析:销毁方法也叫destroy,很正常,但是初始化方法叫afterPropertiesSet,有点奇怪。
仔细分析,发现after Properties Set =》属性设置之后,集合运行结果"service init"正好在"set …"之后,这就明白了,该方法执行时机为,该类所有属性设置(注入)完毕之后
ConfigurableApplicationContext
接口close()
操作ConfigurableApplicationContext
接口registerShutdownHook()
操作public class AppForLifeCycle {public static void main( String[] args ) {//此处需要使用实现类类型,接口类型没有close方法ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");BookDao bookDao = (BookDao) ctx.getBean("bookDao");bookDao.save();//注册关闭钩子函数,在虚拟机退出之前回调此函数,关闭容器ctx.registerShutdownHook();//关闭容器//ctx.close();}
}
依赖注入有几种方式?
setter方式注入使用什么子标签?
构造方式注入使用什么子标签?
名称不必一致了,有时候真的挺好用的
复制一份代码,修改为如下形式:
具体代码很简单,都是打印一句话,没有复杂逻辑
pom.xml引入lombok依赖,省得老是写set方法
org.projectlombok lombok 1.18.26
@Data
public class BookServiceImpl implements BookService {private BookDao bookDao;private UserDao userDao;public void save() {System.out.println("book service save ...");bookDao.save();userDao.save();}//有lombok 所以这里其实提供了所有的Get/Set方法
}
其中:
//BookDao接口
public interface BookDao {public void save();
}
//BookDaoImpl实现类
public class BookDaoImpl implements BookDao {public void save() {System.out.println("book dao save~");}
}//UserDao接口
public interface UserDao {public void save();
}
//UserDaoImpl实现类
public class UserDaoImpl implements UserDao {public void save() {System.out.println("user dao save ...");}
}
applicationContext.xml
AppForDISet 测试方法
public class AppForDISet {public static void main(String[] args) {//1. 创建IoC容器对象,加载spring核心配置文件ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");//2. 在IoC容器里获取Bean对象(BookServiceImpl)BookService bookService = (BookService) ioc.getBean("bookService");bookService.save(); // 正常执行方法}
}
可以注入多个
@Data
public class BookDaoImpl implements BookDao {private int connectionNum;private String databaseName;public void save() {System.out.println("book dao save~ "+connectionNum+" "+databaseName);}//有lombok @Data 这里其实有了所有属性的get/set
}
applicationContext.xml
AppForDISet_Simple 测试方法
public class AppForDISet_Simple {public static void main(String[] args) {ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");BookDao bookDao = (BookDao) ioc.getBean("bookDao");bookDao.save();}
}
先复制一份还原环境
有构造方法,不需要set方法了
public class BookServiceImpl implements BookService {private BookDao bookDao;private UserDao userDao;//构造器注入 不需要set方法了public BookServiceImpl(BookDao bookDao,UserDao userDao){this.bookDao = bookDao;this.userDao = userDao;}public void save() {System.out.println("book service save ...");bookDao.save();userDao.save();}
}
其中:
//BookService接口
public interface BookService {public void save();
}//BookDao 接口
public interface BookDao {public void save();
}
//BookDaoImpl实现类
public class BookDaoImpl implements BookDao {public void save() {System.out.println("book dao save~");}
}//UserDao 接口
public interface UserDao {public void save();
}
//UserDaoImpl 实现类
public class UserDaoImpl implements UserDao {public void save() {System.out.println("user dao save ...");}
}
applicationContext.xml
AppForDIConstructor测试类
public class AppForDIConstructor {public static void main(String[] args) {ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");BookService bookService = (BookService) ioc.getBean("bookService");bookService.save();}
}
public class BookDaoImpl implements BookDao {private String databaseName;private int connectionNum;//同样,有构造方法就不需要set方法了 (这里用不着lombok了)public BookDaoImpl(String databaseName, int connectionNum) {this.databaseName = databaseName;this.connectionNum = connectionNum;}public void save() {System.out.println("book dao save ... " + databaseName+" "+connectionNum);}
}
其中:
//BookDao 接口
public interface BookDao {public void save();
}
applicationContext.xml
AppForDIConstructor_Simple 测试方法
public class AppForDIConstructor_Simple {public static void main(String[] args) {ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");BookDao bookDao = (BookDao) ioc.getBean("bookDao");bookDao.save();}
}
【参数适配 了解】
前面的问题在于配置文件里的name和字段名或属性名name必须一直,属性名改了配置文件里也要改
参数适配就没有这个问题了,常用的还是上面name匹配,但有时候这个适配还真的挺好用的
上面配置最终结果如下:
下面直接在此基础上修改,其他代码不必动
其中:
//BookDaoImpl构造器
public BookDaoImpl(String databaseName, int connectionNum) {this.databaseName = databaseName;this.connectionNum = connectionNum;
}//BookServiceImpl构造器
public BookServiceImpl(BookDao bookDao,UserDao userDao){this.bookDao = bookDao;this.userDao = userDao;
}
constructor-arg按参数类型注入 【type=“int,String,…形参类型”】
等价修改为:
个人猜想:肯定有取别名简化全全类名的方式,但是其实写简单类名,IDEA自动帮你变成了全类名
弊端: 万一两个相同类型(eg:两个String类型) 就没办法用了
constructor-arg按参数位置注入 【index=“0,1,2…参数位置”】
等价修改为:
小结:
如何配置按照类型自动装配?
配置中使用bean标签autowire属性设置自动装配的类型
总得来说一句话: 只能对引用类型自动装配且该引用类型在bean中有且只有一个对象(其实以后配置实现类不可能写两个的,springIOC管理的都是单例对象呀)
(推荐的按类型装配)
// BookDao 接口
public interface BookDao {public void save();
}
// BookDaoImpl 实现类
public class BookDaoImpl implements BookDao {public void save() {System.out.println("book dao save ... ");}
}// UserDao 接口
public interface UserDao {public void save();
}
// UserDaoImpl 实现类
public class UserDaoImpl implements UserDao {public void save() {System.out.println("user dao save ...");}
}// BookService 接口
public interface BookService {public void save();
}// BookServiceImpl 实现类
@Data //set方法还是要给的
public class BookServiceImpl implements BookService {private BookDao bookDao;private UserDao userDao;public void save() {System.out.println("book service save ...");bookDao.save();userDao.save();}
}
applicationContext.xml 使用自动装配 (按类型装配)
测试方法仍然正常执行:
public static void main(String[] args) {ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");BookService bookService = (BookService) ioc.getBean("bookService");bookService.save();
}
环境很简单:一个BookDao(Impl),一个测试方法AppForDICollection
applicationContext.xml也只是配置了1个dao:
pom.xml 里有lombok依赖
BookDaoImpl.java
import java.util.*;
@Data
public class BookDaoImpl implements BookDao {private int[] array;private List list;private Set set;private Map map;private Properties properties;public void save() {System.out.println("book dao save ...");System.out.println("遍历数组: "+ Arrays.toString(array));System.out.println("遍历List: "+list);System.out.println("遍历Set: "+set);System.out.println("遍历Map: "+map);System.out.println("遍历Properties: "+properties);}//这里有所有的get/set
}
100 200 300
itcast itheima boxuegu chuanzhihui
itcast itheima boxuegu boxuegu
china henan kaifeng
说明:property标签表示setter方式注入,构造方式注入constructor-arg标签内部也可以写
、 、
、
最终配置:
100 200 300 www whu edu cn
whu edu cn 青鸾峰 天墉城 幻溟界
测试方法
public class AppForDICollection {public static void main(String[] args) {//1. 创建IoC容器对象,加载spring核心配置文件ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");BookDao bookDao = (BookDao) ioc.getBean("bookDao");bookDao.save();}
}
第三方资源配置管理
还是复制一份之前的修改吧
App.java
public class App {public static void main(String[] args) {//1. 创建IoC容器对象,加载spring核心配置文件ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");}
}
applicationContext.xml
pom.xml
4.0.0 cn.whu spring_09_datasource war 1.0-SNAPSHOT org.springframework spring-context 5.2.10.RELEASE com.alibaba druid 1.1.16
测试druid
public static void main(String[] args) {//1. 创建IoC容器对象,加载spring核心配置文件ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");DataSource dataSource = (DataSource) ioc.getBean("dataSource");System.out.println(dataSource);
}
获取到了就OK
先pom.xml中引入依赖:
c3p0 c3p0 0.9.1.2
mysql mysql-connector-java 5.1.16
不知道坐标,可以百度或者maven仓库搜索
https://mvnrepository.com/search?q=mysql
applicationContext.xml 核心配置里写管理
测试方法:
public class TestC3P0 {public static void main(String[] args) {//1. 创建IoC容器对象,加载spring核心配置文件ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");DataSource dataSource = (DataSource) ioc.getBean("comboPooledDataSource");System.out.println(dataSource);}
}
运行结果:
com.mchange.v2.c3p0.ComboPooledDataSource [ acquireIncrement -> 3, acquireRetryAttempts -> 30, acquireRetryDelay -> 1000, autoCommitOnClose -> false, automaticTestTable -> null, breakAfterAcquireFailure -> false, checkoutTimeout -> 0, connectionCustomizerClassName -> null, connectionTesterClassName -> com.mchange.v2.c3p0.impl.DefaultConnectionTester, dataSourceName -> 2zr8rdau1tyh71w1bihje8|bd8db5a, debugUnreturnedConnectionStackTraces -> false, description -> null, driverClass -> com.mysql.jdbc.Driver, factoryClassLocation -> null, forceIgnoreUnresolvedTransactions -> false, identityToken -> 2zr8rdau1tyh71w1bihje8|bd8db5a, idleConnectionTestPeriod -> 0, initialPoolSize -> 3, jdbcUrl -> jdbc:mysql://localhost:3306/spring_db, maxAdministrativeTaskTime -> 0, maxConnectionAge -> 0, maxIdleTime -> 0, maxIdleTimeExcessConnections -> 0, maxPoolSize -> 15, maxStatements -> 0, maxStatementsPerConnection -> 0, minPoolSize -> 3, numHelperThreads -> 3, numThreadsAwaitingCheckoutDefaultUser -> 0, preferredTestQuery -> null, properties -> {user=******, password=******}, propertyCycle -> 0, testConnectionOnCheckin -> false, testConnectionOnCheckout -> false, unreturnedConnectionTimeout -> 0, usesTraditionalReflectiveProxies -> false ]
jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://127.0.0.1:3306/spring_db
jdbc.username=root
jdbc.password=1234
创建一个模板文件,一件生成文件, 文件名都不用写的
最终配置:
测试效果
1)直接运行上面的测试方法,输出结果完全一样。
2)下面将数据注入到BookDaoImpl看看效果
BookDaoImpl.java
public class BookDaoImpl implements BookDao {private String name;public void save() {System.out.println("book dao save ... " + name);}public void setName(String name) {this.name = name;}
}
applicationContext.xml 里最下面加入一行配置
TestC3P0DI.java 测试方法
public class TestC3P0DI {public static void main(String[] args) {//1. 创建IoC容器对象,加载spring核心配置文件ApplicationContext ioc = new ClassPathXmlApplicationContext("applicationContext.xml");BookDao bookDao = (BookDao) ioc.getBean("bookDao");bookDao.save();}
}
这就简单试了一下参数的注入
加载properties文件注意点:
- 为什么写jdbc.username而不直接写username?系统有环境变量username了,优先级比我写的高,自己写的username就被屏蔽了
除非引入properties文件时手动将系统环境变量给屏蔽了:
- 可以导入多个properties配置文件,location里逗号隔开就行了
- 最好的方式:一次性加载所有的配置文件:
location="classpath*:*.properties"
classpath*:
能省略,但是不建议
加上classpath*:
就不仅可以从当前项目读properties文件,还可以读导入的jar包里的properties文件了
小结:
下一篇:2025端午及暑假上海旅游攻略