持续创作,加速成长!这是我参与「掘金日新计划 · 6 月更文挑战」的第7天,点击查看活动详情
1.1、简介
spring理念:使现有的技术更加容易使用,本身就是一个大杂烩,整合了现有的技术框架
- SSH :Struct2+Spring+Hibernate
- SSM :SpringMVC+Spring+MyBatis
官方下载地址:repo.spring.io/ui/native/r…
github地址:github.com/spring-proj…
<!-- https://mvnrepository.com/artifact/org.springframework/spring-webmvc -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.3.18</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-webmvc -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>5.3.18</version>
</dependency>
1.2、优点
- Spring是一个免费的开源的框架(容器)
- Spring是一个轻量级的非入侵式的框架
- 控制反转(IOC),面向切面编程(AOP)
- 支持事务的处理,对框架整合的支持
总结
Spring就是一个轻量级的控制反转(IOC)和面向切面编程的(AOP)框架
1.3、组成
1.4、拓展
Spring官网的介绍:现代化的Java开发,基于Spring开发
-
Spring Boot
- 一个快速开发的脚手架
- 基于SpringBoot可以快速开发单个微服务
- 约定大于配置
-
Spring Cloud
- 基于SpringBoot实现
2、IOC理论推导
原来:
-
UserDao 接口
public interface UserDao { void getUser(); } -
UserDaoImpl 实现类
public class UserDaoImpl implements UserDao{ public void getUser() { System.out.println("默认获取用户的数据"); } } -
UserService 业务接口
public interface UserService { void getUser(); } -
UserServiceImpl 业务实现类
public class UserServiceImpl implements UserService { //原先写法 //private UserDao userDao = new UserDaoImpl; private UserDao userDao; //利用set实现值的动态注入 public void setUserDao(UserDao userDao) { this.userDao = userDao; } public void getUser() { userDao.getUser(); } }
在之前的业务中,用户的需求可能会影响原来的代码,需要根据用户需求去修改源代码
使用一个set接口实现
private UserDao userDao;
//利用set实现动态的值的注入
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
}
- 之前,程序是主动创建对象,控制权在程序员手里
- 使用了set注入后,程序不在具有主动性,被动接收对象
这种思想,本质上解决了问题,程序员不用再管理对象的创建。系统的耦合性大大降低,可以更加专注在业务的实现。
IOC本质
控制反转IoC(Inversion of Control),是一种设计思想,DI(依赖注入)是实现IoC的一种方法,也有人认为DI只是引oC的另一种说法。没有引oC的程序中,我们使用面向对象编程,对象的创建与对象间的依赖关系完全硬编码在程序中,对象的创建由程序自己控制,控制反转后将对象的创建转移给第三方,所谓控制反转就是:获得依赖对象的方式反转了。
采用XML方式配置Bean的时候,Bean的定义信息是和实现分离的,而采用注解的方式可以把两者合为一体Bean的定义信息直接以注解的形式定义在实现类中,从而达到了零配置的目的。
控制反转是一种通过描述(XML或注解)并通过第三方去生产或获取特定对象的方式。在Spig中实现控制反转的是IoC容器,其实现方法是依赖注入(Dependency Injection,Dl)
3、HelloSpring
-
实体类
public class Hello { private String str; public String getStr() { return str; } public void setStr(String str) { this.str = str; } @Override public String toString() { return "Hello{" + "str='" + str + ''' + '}'; } } -
beans.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 https://www.springframework.org/schema/beans/spring-beans.xsd"> <!--使用Spring创建对象,在Spring这些都称为Bean--> <bean id="hello" class="com.rop.pojo.Hello"> <property name="str" value="HelloSpring"/> </bean> </beans> -
测试
public class Test { public static void main(String[] args) { //获取Spring的上下文对象 ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml"); //在context中取出对象 Hello hello = (Hello) context.getBean("hello"); System.out.println(hello.toString()); } }
这个过程就叫控制反转:
控制:谁来控制对象的创建,传统应用程序的对象是由程序本身控制创建的,使用Spring后,对象是由Spring来创建的。
反转:程序本身不创建对象,而变成被动的接收对象。
依赖注入:就是利用set方法来进行注入的。
IOC是一种编程思想,由主动的编程变成被动的接收。
可以通过newClassPathXmlApplicationContext去浏览一下底层源码。
IOC:对象由Spring来创建,管理,装配!
4、IOC创建对象的方式
-
使用无参构造创建对象,默认
-
假如要使用有参构造创建对象
-
下标赋值
<bean id="User" class="com.rop.pojo.User"> <!--有参构造,方式一 通过下标赋值--> <constructor-arg index="0" value="SpringStudy"/> </bean> -
通过类型赋值
<bean id="User" class="com.rop.pojo.User"> <!--方式二 通过类型创建, 不建议使用,假如有两个string参数,方法无效--> <constructor-arg type="java.lang.String" value="rop"/> </bean> -
参数名
<bean id="User" class="com.rop.pojo.User"> <!--方式三,直接通过参数名来设置--> <constructor-arg name="name" value="rao"/> </bean>
-
总结:在配置文件加载的时候,容器中管理的对象就已经初始化了。
5、Spring配置
5.1、别名
<!--别名,如果添加了别名,我们也可以使用别名来获取这个对象-->
<alias name="User" alias="rop"/>
5.2、Bean的配置
<!--
id:bean的唯一标识符。相当于我们学的对象名
class:bean对象所对应的全限定名:包名+类名
name:也是一个别名,而且name更高级,可以同时取多个别名
-->
<bean id="userT" class="com.rop.pojo.UserT" name="user2 u2,u3;u4">
<property name="name" value="Spring"/>
</bean>
5.3、import
一般用于团队开发,可以将多个配置文件导入合并为一个。
选择第一个Beans。
再在applicationContext.xml导入其他的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
https://www.springframework.org/schema/beans/spring-beans.xsd">
<import resource="beans.xml"/>
<import resource="beans2.xml"/>
<import resource="beans3.xml"/>
</beans>
6、DI(依赖注入)
6.1、构造器注入
上面已经叙述了
6.2、Set方式注入【重点】
-
依赖注入:Set注入!
-
依赖
- bean对象的创建依赖于容器
-
注入
- bean对象中的所有属性由容器注入
环境搭建
- 复杂类型
- 真实测试对象
环境搭建代码
public class Address {
private String address;
public String getAddress() {
return address;
}
public void setAddress(String address) {
this.address = address;
}
}
public class Student {
private String name;
private Address address;
private String[] books;
private List<String> hobbies;
private Map<String,String> cards;
private Set<String> games;
private String wife;
private Properties info;
}
<?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
https://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="student" class="com.rop.pojo.Student">
<!--第一种:普通值注入,value-->
<property name="name" value="Mr.rop"/>
</bean>
</beans>
public class DiTest {
public static void main(String[] args) {
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
Student student = (Student) context.getBean("student");
System.out.println(student.getName());
}
}
注入信息
<?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
https://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="address" class="com.rop.pojo.Address">
<property name="address" value="123123@qq.com"/>
</bean>
<bean id="student" class="com.rop.pojo.Student">
<!--第一种:普通值注入,value-->
<property name="name" value="Mr.rop"/>
<!--第二种:bean注入,ref-->
<property name="address" ref="address"/>
<!--第三种:数组注入,-->
<property name="books">
<array>
<value>三国演义</value>
<value>红楼梦</value>
<value>水浒传</value>
<value>西游记</value>
</array>
</property>
<!--第四种:List-->
<property name="hobbies">
<list>
<value>听歌</value>
<value>篮球</value>
<value>游戏</value>
<value>跑步</value>
</list>
</property>
<!--第五种:map-->
<property name="cards">
<map>
<entry key="手机号" value="12345678901"/>
<entry key="身份证" value="111111222222223333"/>
</map>
</property>
<!--第六种:set-->
<property name="games">
<set>
<value>LOL</value>
<value>BOB</value>
<value>COC</value>
</set>
</property>
<!--第七种:同第一种-->
<property name="wife">
<null/>
</property>
<!--第八种:properties-->
<property name="info">
<props>
<prop key="学号">123123</prop>
</props>
</property>
</bean>
</beans>
6.3、拓展方式注入
可以使用p命名空间和c命名空间进行注入
p命名空间
xmlns:p="http://www.springframework.org/schema/p"
c命名空间
xmlns:c="http://www.springframework.org/schema/c"
测试
public class User {
private int age;
private String name;
public User() {
}
public User(int age, String name) {
this.age = age;
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "User{" +
"age=" + age +
", name='" + name + ''' +
'}';
}
}
<!--p命名空间注入,可以直接注入属性的值:property,本质:set注入-->
<bean id="user" class="com.rop.pojo.User" p:age="18" p:name="rop"/>
<!--c命名空间注入,通过构造器注入-->
<bean id="user2" class="com.rop.pojo.User" c:age="20" c:name="Mr.rop"/>
@Test
public void getUser(){
ApplicationContext context = new ClassPathXmlApplicationContext("userbeans.xml");
User user = context.getBean("user",User.class); //p命名空间注入
User user2 = context.getBean("user2",User.class); //c命名空间注入
System.out.println(user.toString());
System.out.println();
System.out.println(user2.toString());
}
p命名空间和c命名空间不能直接使用,需要使用上面的约束。
6.4、bean的作用域
-
单例模式(Spring默认机制)
<bean id="user2" class="com.rop.pojo.User" c:age="20" c:name="Mr.rop" scope="singleton"/> -
原型模式:每次从容器中get的时候,都会产生一个新对象
<bean id="user2" class="com.rop.pojo.User" c:age="20" c:name="Mr.rop" scope="prototype"/> -
其余的request、session、application这些个在web开发中使用到
7、Bean的自动装配
- 自动装配,是Spring满足bean依赖的一种方式
- Spring会在上下文中自动寻找,并自动给bean装配属性
在Spring中有三种装配的方式
- 在xml中显示的配置
- 在java中显示配置
- 隐式的自动装配【重要】
7.1、测试
环境搭建:一个人有两个宠物
7.2、ByName自动装配
<!--
byName:会自动在容器的上下文查找,与自己对象set方法后面的值对应的beanId
-->
<bean id="people" class="com.rop.pojo.People" autowire="byName">
<property name="name" value="rop"/>
<!--<property name="cat" ref="cat"/>-->
<!--<property name="dog" ref="dog"/>-->
</bean>
7.3、ByType自动装配
<bean class="com.rop.pojo.Cat"/>
<bean class="com.rop.pojo.Dog"/>
<!--
byType:会自动在容器的上下文查找,与自己对象属性类型相同的bean
-->
<bean id="people" class="com.rop.pojo.People" autowire="byType">
<property name="name" value="rop"/>
<!--<property name="cat" ref="cat"/>-->
<!--<property name="dog" ref="dog"/>-->
</bean>
小结:
- byName的时候,需要保证所有bean 的id唯一且这个bean需要和自动注入的属性的set方法的值一致
- byType的时候,需要保证所有bean 的class唯一且这个bean需要和自动注入的属性的类型一致
7.4、使用注解实现自动装配
使用注解须知:
-
导入约束:context约束
-
配置注解的支持:context:annotation-config/
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans https://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd"> <context:annotation-config/> </beans>
@Autowired
直接在属性上使用即可!也可以在set方法上使用。
使用autowired后,可以选择不编写set方法,前提是这个自动装配的属性在IOC(Spring)容器中存在且优先符合byType。
@Autowired
private Cat cat;
@Autowired
private Dog dog;
private String name;
如果@Autowired自动装配的环境比较复杂,自动装配无法通过一个注解【@Autowired】完成的时候,那就使用@Qualifier(value = "id名")去配合@Autowired使用,指定唯一的bean对象注入。
@Resource注解
// @Autowired
@Resource(name = "cat2")
private Cat cat;
// @Autowired
@Resource
private Dog dog;
private String name;
小结:
@Resource与@Autowired的区别:
- 都是用来自动装配的,都可以放在属性字段上
- @Autowired默认通过byType的方式,当遇到多个同类型时,使用@Qualifier配合byName进行装配
- @Resource默认通过byName方式,如果找不到名字,则通过byType实现。如果两个都找不到,就报错