0. 框架1
0.1 什么是框架?
框架是一类工具包的组合,(方便开发)
我们知道当我每开发一个功能时如果功能比较复杂,我们很有可能把共性的东西抽取成工具类,而如果这个工具类有很多java 工具类组成某一方面的功能,而且功能比较强大的化我们可以将其制作成jar 包发布,供其他项目引用, 这个jar 我们就可以称之为框架
0.2 为什么要使用框架?
我们在学习黑马旅游网的时候,有很多工具类都是我们字节写的,比如BaseServlet , 有两个选择,
A,复制别人的代码直接使用,早点下班享受 欢乐人生
B,埋头苦干,自立更生,放弃6点下班的权力
很显然, 一个不是太固执的人都会选择 A , 因为简单方便,我们应该考虑的问题别人已经考虑好了
同样,框架也是将项目中共有的问题抽取出来,封装成工具类供大家使用,我们遇到的问题,前辈们已经帮助我们封装好了工具类,使用就好了
0.3 为什么是三大框架?
为了便于管理代码,现在流行的是分层开发,
-
控制层负责接收参数,响应数据
-
service 层负责处理业务逻辑
-
dao 层负责数据库交互
但是分层完毕后,全世界所有的java 项目开发人员发现 控制层,dao 层 所有的逻辑都使用servlect 太麻烦,需要封装工具类,而互联网有很多封装好的工具类可以使用,开发人员就会选择比较适用的下载使用,
世间本没有路,走的人多了就成了路,
三大框架就是因为使用的人多我们称为"三大框架",但是市面上依旧有很多框架,只不过用人相对较少
0.4 三大框架分别解决了什么问题?
-
mybatis 解决的是dao 层问题, 使我们访问数据库代码更加简介,开发更加快捷
-
springMVC 解决的是 表现层(web层,也叫controller 层)问题,使我们的开发变得简单快捷
-
spring 解决的的是 各个框架之间的整合问题相当于一个领导,不干活,只负责管理人与人之前的关系
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-GmB8Rjgx-1583059037686)(assets/01三层架构.png)]
0.5 框架使用起来为何比不使用框架还难?
框架使用起来为何比不使用框架还难? 不是是我们的开发更加简单了吗?
“杀鸡焉用青龙偃月刀”?
如果只是做一个登陆注册,传统方式,当然简单,但是如果让你做一个非常大的项目,你还用 原始的方式就必须使用框架了. 否则开发出来的项目可能有很多问题.
所以大家开始学习项目有这样的感觉 ,配置好麻烦, 配置麻烦是给后续项目打基础的,会使后续项目开发更加简单?
1. spring概述
1.1 Spring是什么(理解) IOC/AOP
Spring是分层的 Java SE/EE应用 full-stack 轻量级开源框架,以 IoC(Inverse Of Control:反转控制)和 AOP(Aspect Oriented Programming:面向切面编程)为内核。
提供了展现层 SpringMVC和持久层 Spring JDBCTemplate以及业务层事务管理等众多的企业级应用技术,还能整合开源世界众多著名的第三方框架和类库,逐渐成为使用最多的Java EE 企业应用开源框架
1.1 spring IOC解决了什么问题?(补)
/***
* 什么是耦合?
* 程序间的依赖关系
* 如何解耦?
* 降低程序间的依赖关系
实际开发中:
* 应该做到:编译期不依赖,运行时才依赖。
*
* 思路?
* 1)使用反射来创建对象,而避免使用new关键字。
* 2)通过读取配置文件来获取要创建的对象全限定类名
*/
public class JDBCDemo {
public static void main(String [] args) throws Exception{
//1.注册驱动
//DriverManager.registerDriver(new com.mysql.jdbc.Driver());
Class.forName("com.mysql.jdbc.Driver");
//2.获取连接
Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/spring","root","root");
//3.获取操作数据库的预处理对象
PreparedStatement pstm = conn.prepareStatement("select * from account");
//4.执行SQL,得到结果集
ResultSet rs = pstm.executeQuery();
//5.遍历结果集
while(rs.next()){
System.out.println(rs.getString("name"));
}
//6.释放资源
rs.close();
pstm.close();
conn.close();
}
传统的代码是存在耦合的,如何降低耦合? 我们可以字节写工具类利用反射的思想解决,但是我们碰到过的问题,前辈们都碰到过,已经帮我们解决了,我我们拿来用就可以了,这个工具包就是spring,也叫框架
UserService userService = new UserServiceImpl();
userService.save();
UserDao dao = new UserDaoImpl();
dao.save();
1.2 Spring发展历程 (了解)
EJB
Rod Johnson ( Spring 之父)
2017 年
9 月份发布了 Spring 的最新版本 Spring5.0
通用版(GA)
1.3 Spring的优势(理解)
方便解耦,简化开发
AOP 编程的支持
声明式事务的支持
方便程序的测试
1.4 Spring的体系结构(了解)
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-c43Trctu-1583059037692)(.\img\图片29.png)]
2. spring快速入门
2.1 Spring程序开发步骤
①导入 Spring 开发的基本包坐标
②编写 Dao 接口和实现类
③创建 Spring 核心配置文件
④在 Spring 配置文件中配置 UserDaoImpl
⑤使用 Spring 的 API 获得 Bean 实例
2.2 导入Spring开发的基本包坐标
<properties>
<spring.version>5.0.5.RELEASE</spring.version>
</properties>
<!--导入spring的context坐标,context依赖core、beans、expression-->
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>${spring.version}</version>
</dependency>
</dependencies>
2.3 编写Dao接口和实现类
public interface UserDao {
public void save();
}
public class UserDaoImpl implements UserDao {
@Override
public void save() {
System.out.println("UserDao save method running....");
}
}
2.4 创建Spring核心配置文件
在类路径下(resources)创建applicationContext.xml配置文件
<?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">
</beans>
2.5 在Spring配置文件中配置UserDaoImpl
<?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">
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"></bean>
<bean id="userDao1" name="userDao2,userDao3" class="com.itheima.dao.UserDaoImpl"/>
</beans>
2.6 使用Spring的API获得Bean实例
@Test
public void test1(){
ApplicationContext applicationContext = new
ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao) applicationContext.getBean("userDao"); userDao.save();
}
2. 快速入门解析_(补充)
2.1) IOC
什么是IOC
上述把创建对象的权力交给框架或者工厂,创建对象的权力发生了反转,我们叫做控制反转Inverse Of Control
什么是
spring IOC
spring 对上述IOC 思想进行了具体功能的实现,我们叫springIOC 如果张三同学自己开发了一个框架.实现了IOC 我们可以把这个框架叫 zsIOC 仅仅是个代号
IOC 是思想, springIOc是具体实现
3. Spring配置文件
3.1 Bean标签基本配置
用于配置对象交由Spring 来创建。
默认情况下它调用的是类中的无参构造函数,如果没有无参构造函数则不能创建成功。
基本属性:
id:Bean实例在Spring容器中的唯一标识
class:Bean的全限定名称
3.2_单利模式(补)
什么是单利模式?
一个对象从tomcat启动到tomcat 结束(假设运行24小时) ,再次期间如果对象只被创建了一次,则对象是单利的
单利有什么好处?
我们知道每个new 一个对象,对内存中就需要开辟一部分空间存储该对象,如果一个对象每次被访问都需要new 一次,极大的浪费了内存空间,导致系统性能下降
单利的好处:
1) 节约内存
弊端:
如果有全局变量则有安全问题(多线程环境下)
java 如何实现单利模式?
//懒汉式和饿汉式两种
//所谓“懒汉式”与“饿汉式”的区别,是在与建立单例对象的时间的不同。
//“懒汉式”是在你真正用到的时候才去建这个单例对象:
public class Singleton{
private Singleton(){}
private static Singleton singleton = null; //不建立对象
public static synchronized Singleton getInstance(){
if(singleton == null) { //先判断是否为空
singleton = new Singleton (); //懒汉式做法
}
return singleton ;
}
}
// “饿汉式”是在不管你用的用不上,一开始就建立这个单例对象:
public class Singleton{
public Singleton(){}
private static Singleton singleton = new Singleton(); //建立对象
public static Singleton getInstance(){
return singleton ;//直接返回单例对象
}}
3.2 Bean标签范围配置
scope:指对象的作用范围,取值如下:
取值范围 | 说明 |
---|---|
singleton | 默认值,单例的 |
prototype | 多例的 |
request | WEB 项目中,Spring 创建一个 Bean 的对象,将对象存入到 request 域中 |
session | WEB 项目中,Spring 创建一个 Bean 的对象,将对象存入到 session 域中 |
global session | WEB 项目中,应用在 Portlet 环境,如果没有 Portlet 环境那么globalSession 相当于 session |
1)当scope的取值为singleton时
Bean的实例化个数:1个
Bean的实例化时机:当Spring核心文件被加载时,实例化配置的Bean实例
Bean的生命周期:
对象创建:当应用加载,创建容器时,对象就被创建了
对象运行:只要容器在,对象一直活着
对象销毁:当应用卸载,销毁容器时,对象就被销毁了
2)当scope的取值为prototype时
Bean的实例化个数:多个
Bean的实例化时机:当调用getBean()方法时实例化Bean
对象创建:当使用对象时,创建新的对象实例
对象运行:只要对象在使用中,就一直活着
对象销毁:当对象长时间不用时,被 Java 的垃圾回收器回收了
3.3 Bean生命周期配置
init-method:指定类中的初始化方法名称
destroy-method:指定类中销毁方法名称
3.4 Bean实例化三种方式
1) 使用无参构造方法实例化
它会根据默认无参构造方法来创建类对象,如果bean中没有默认无参构造函数,将会创建失败
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"/>
2) 工厂静态方法实例化
工厂的静态方法返回Bean实例
public class StaticFactoryBean {
public static UserDao createUserDao(){
return new UserDaoImpl();
}
}
<bean id="userDao" class="com.itheima.factory.StaticFactoryBean"
factory-method="createUserDao" />
例如创建 conn 对象
<bean id="conn" class="java.sql.DriverManager" factory-method="getConnection">
<constructor-arg name="url" value="jdbc:mysql://localhost:3306/spring"/>
</bean>
3) 工厂实例方法实例化
工厂的非静态方法返回Bean实例
public class DynamicFactoryBean {
public UserDao createUserDao(){
return new UserDaoImpl();
}
}
<bean id="factoryBean" class="com.itheima.factory.DynamicFactoryBean"/>
<bean id="userDao" factory-bean="factoryBean" factory-method="createUserDao"/>
3.5 Bean的依赖注入入门
DI : Dependency Injection
依赖容器,通过注入的方式管理对象与对象之间的关系
①创建 UserService,UserService 内部在调用 UserDao的save() 方法
public class UserServiceImpl implements UserService {
@Override
public void save() {
ApplicationContext applicationContext = new
ClassPathXmlApplicationContext("applicationContext.xml"); UserDao userDao = (UserDao) applicationContext.getBean("userDao");
userDao.save();
}
}
②将 UserServiceImpl 的创建权交给 Spring
<bean id="userService" class="com.itheima.service.impl.UserServiceImpl"/>
③从 Spring 容器中获得 UserService 进行操作
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = (UserService) applicationContext.getBean("userService");
userService.save();
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-9PkpKrid-1583059037694)(assets/1564383417777.png)]
3.6 Bean的依赖注入概念
依赖容器 注入 对象或数据
依赖注入(Dependency Injection):它是 Spring 框架核心 IOC 的具体实现。
在编写程序时,通过控制反转,把对象的创建交给了 Spring,但是代码中不可能出现没有依赖的情况。
IOC 解耦只是降低他们的依赖关系,但不会消除。例如:业务层仍会调用持久层的方法。
那这种业务层和持久层的依赖关系,在使用 Spring 之后,就让 Spring 来维护了。
简单的说,就是坐等框架把持久层对象传入业务层,而不用我们自己去获取
3.7 Bean的依赖注入方式
①构造方法
创建有参构造
public class UserServiceImpl implements UserService {
private UserDao userDao;
public UserServiceImpl(UserDao userDao) {
this.userDao = userDao;
}
@Override
public void save() {
userDao.save();
}
}
配置Spring容器调用有参构造时进行注入
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"/>
<bean id="userService" class="com.itheima.service.impl.UserServiceImpl"> <constructor-arg name="userDao" ref="userDao"></constructor-arg>
</bean>
②set方法
在UserServiceImpl中添加setUserDao方法
public class UserServiceImpl implements UserService {
private UserDao userDao;
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
}
@Override
public void save() {
userDao.save();
}
}
配置Spring容器调用set方法进行注入
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"/>
<bean id="userService" class="com.itheima.service.impl.UserServiceImpl">
<property name="userDao" ref="userDao"/>
</bean>
set方法:P命名空间注入
P命名空间注入本质也是set方法注入,但比起上述的set方法注入更加方便,主要体现在配置文件中,如下:
首先,需要引入P命名空间:
xmlns:p="http://www.springframework.org/schema/p"
其次,需要修改注入方式
<bean id="userService" class="com.itheima.service.impl.UserServiceImpl"
p:userDao-ref="userDao"/>
3.8 Bean的依赖注入的数据类型
上面的操作,都是注入的引用Bean,处了对象的引用可以注入,普通数据类型,集合等都可以在容器中进行注入。
注入数据的三种数据类型
普通数据类型
引用数据类型
集合数据类型
其中引用数据类型,此处就不再赘述了,之前的操作都是对UserDao对象的引用进行注入的,下面将以set方法注入为例,演示普通数据类型和集合数据类型的注入。
Bean的依赖注入的数据类型
(1)普通数据类型的注入
public class UserDaoImpl implements UserDao {
private String company;
private int age;
public void setCompany(String company) {
this.company = company;
}
public void setAge(int age) {
this.age = age;
}
public void save() {
System.out.println(company+"==="+age);
System.out.println("UserDao save method running....");
}
}
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl">
<property name="company" value="传智播客"></property>
<property name="age" value="15"></property>
</bean>
(2)集合数据类型(List)的注入
public class UserDaoImpl implements UserDao {
private List<String> strList;
public void setStrList(List<String> strList) {
this.strList = strList;
}
public void save() {
System.out.println(strList);
System.out.println("UserDao save method running....");
}
}
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl">
<property name="strList">
<list>
<value>aaa</value>
<value>bbb</value>
<value>ccc</value>
</list>
</property>
</bean>
(3)集合数据类型(List)的注入
public class UserDaoImpl implements UserDao {
private List<User> userList;
public void setUserList(List<User> userList) {
this.userList = userList;
}
public void save() {
System.out.println(userList);
System.out.println("UserDao save method running....");
}
}
<bean id="u1" class="com.itheima.domain.User"/>
<bean id="u2" class="com.itheima.domain.User"/>
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl">
<property name="userList">
<list>
<bean class="com.itheima.domain.User"/>
<bean class="com.itheima.domain.User"/>
<ref bean="u1"/>
<ref bean="u2"/>
</list>
</property>
</bean>
(4)集合数据类型( Map<String,User> )的注入
public class UserDaoImpl implements UserDao {
private Map<String,User> userMap;
public void setUserMap(Map<String, User> userMap) {
this.userMap = userMap;
}
public void save() {
System.out.println(userMap);
System.out.println("UserDao save method running....");
}
}
<bean id="u1" class="com.itheima.domain.User"/>
<bean id="u2" class="com.itheima.domain.User"/>
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl">
<property name="userMap">
<map>
<entry key="user1" value-ref="u1"/>
<entry key="user2" value-ref="u2"/>
</map>
</property>
</bean>
(5)集合数据类型(Properties)的注入
public class UserDaoImpl implements UserDao {
private Properties properties;
public void setProperties(Properties properties) {
this.properties = properties;
}
public void save() {
System.out.println(properties);
System.out.println("UserDao save method running....");
}
}
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl">
<property name="properties">
<props>
<prop key="p1">aaa</prop>
<prop key="p2">bbb</prop>
<prop key="p3">ccc</prop>
</props>
</property>
</bean>
3.9 引入其他配置文件(分模块开发)
实际开发中,Spring的配置内容非常多,这就导致Spring配置很繁杂且体积很大,所以,可以将部分配置拆解到其他配置文件中,而在Spring主配置文件通过import标签进行加载
<import resource="applicationContext-xxx.xml"/>
4. spring相关API
4.1 ApplicationContext的继承体系
applicationContext:接口类型,代表应用上下文,可以通过其实例获得 Spring 容器中的 Bean 对象
4.2 ApplicationContext的实现类
1)ClassPathXmlApplicationContext
它是从类的根路径下加载配置文件 推荐使用这种
2)FileSystemXmlApplicationContext
它是从磁盘路径上加载配置文件,配置文件可以在磁盘的任意位置。
3)AnnotationConfigApplicationContext
当使用注解配置容器对象时,需要使用此类来创建 spring 容器。它用来读取注解。
4.3 getBean()方法使用
public Object getBean(String name) throws BeansException {
assertBeanFactoryActive();
return getBeanFactory().getBean(name);
}
public <T> T getBean(Class<T> requiredType) throws BeansException { assertBeanFactoryActive();
return getBeanFactory().getBean(requiredType);
}
其中,当参数的数据类型是字符串时,表示根据Bean的id从容器中获得Bean实例,返回是Object,需要强转。
当参数的数据类型是Class类型时,表示根据类型从容器中匹配Bean实例,当容器中相同类型的Bean有多个时,则此方法会报错
getBean()方法使用
ApplicationContext applicationContext = new
ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService1 = (UserService) applicationContext.getBean("userService");
UserService userService2 = applicationContext.getBean(UserService.class);