一、Spring概述
Spring是一款主流的Java EE輕量級開源框架。
Spring的核心模塊:IoC(控制反轉,指把創建對象過程交給Spring管理 )、AOP(面向切面編程,在不修改源代碼的基礎上增強代碼功能)
二、Spring入門
2.1 入門案例開發步驟
- 引入spring相關依賴
- 創建類,定義屬性和方法
- 按照spring要求創建配置文件
- 在spring配置文件中配置相關信息
- 進行測試
2.2 案例開發實例
創建一個spring maven項目名為Spring6,再在Spring6下面創建一個名為Spring-first的模塊,在此模塊下的pom.xml里增加依賴。
<dependencies><!-- 第2.1個spring context依賴(這是spring的基礎依賴) --><dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>5.3.7</version><!--如果是JDK17及以上的,就寫6.0.0版本+--></dependency><!--第2.2個spring junit依賴 --><dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.13.2</version>
<!-- <scope>test</scope>--></dependency></dependencies>
復制之后,點擊m小圖標進行聯網下載。
創建一個User類,里面創建一個add方法。
public class User {public void add(){System.out.println("添加。。。");}
}
創建一個bean.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"><!-- 完成user對象創建-->
<!-- id屬性:唯一標識-->
<!-- class屬性:要創建的對象所在類的全路徑--><bean id="user" class="com.hbnu.spring6.User"/>
</beans>
再寫一個測試類,測試方法調用
public class TestUser {
//這里的注解要導入 import org.junit.Test;@Testpublic void testUser(){//加載spring配置文件,創建對象ApplicationContext applicationContext=new ClassPathXmlApplicationContext("bean.xml");//獲取創建的對象User user = (User) applicationContext.getBean("user");//getBean --> xml文件里的id值System.out.println(user);//使用對象調用方法進行測試user.add();}
}
如果是利用反射創建對象呢?
//利用反射創建對象public void testUserObject1() throws Exception {Class clazz=Class.forName("com.hbnu.spring6.User");//調用方法創建對象User user = (User) clazz.getDeclaredConstructor().newInstance();System.out.println(user);user.add();}
三、容器:IoC
IoC是Inversion of Control的簡寫,譯為”控制反轉”,是一種設計思想(不是一種技術),是一個重要的面向對象編程法則,是Spring框架中最重要的核心組件之一。
Spring通過IoC容器來管理所有的Java對象的實例化和初始化,控制對象與對象之間的依賴關系。我們將由IoC容器管理的Java對象稱為Spring Bean,它與new一個對象沒有區別。
控制反轉,反轉的是什么?
- 將對象的創建權力交出去,交給第三方容器負責
- 將對象和對象之間關系的維護權交出去,交給第三方容器負責
控制反轉這種思想如何實現?
- DI :依賴注入
3.1 依賴注入
指Spring創建對象的過程中,將對象依賴屬性通過配置進行注入
常見方式包括兩種:
- set注入
- 構造注入
3.2 IoC容器在Spring的實現
IoC容器中管理的組件也叫做bean,在創建bean之前,首先要創建IoC容器。Spring提供了兩種實現方式:
BeanFactory 是IoC容器的基本實現,是Spring內部使用的接口,面向Spring本身,不提供給開發人員
ApplicationContext 是BeanFactory的子接口,提供了更多高級特性,面向Spring的使用者
ApplicationContext的主要實現類:
3.3 基于xml管理bean
1.獲取bean
xml文件里配置id值
<bean id="user" class="com.hbnu.spring6.User"/>
①根據id獲取
public void test1(){ApplicationContext ac=new ClassPathXmlApplicationContext("bean.xml");User user = (User) applicationContext.getBean("user");
}
②根據id和類型獲取
public void test2(){ApplicationContext ac=new ClassPathXmlApplicationContext("bean.xml");User user = applicationContext.getBean("user",User.class);
}
③根據類型獲取
public void test3(){ApplicationContext ac=new ClassPathXmlApplicationContext("bean.xml");User user = applicationContext.getBean(User.class);
}
且當根據類型獲取bean的時候,要求ioc容器里指定類型(class)的bean只允許有一個,配置了兩個會報錯。即下面兩行是錯的。
<bean id="user" class="com.hbnu.spring6.User"/>
<bean id="user1" class="com.hbnu.spring6.User"/>
注意:①如果組件類實現了接口,根據接口類型可以獲取bean嗎?可以,前提是bean唯一 ②如果一個接口有多個實現類,這些實現類都配置了bean,根據接口類型可以獲取bean嗎?不行,因為bean不唯一
以下是錯誤的:
<!-- 一個接口實現類獲取過程--><bean id="UserDaoImpl" class="com.hbnu.spring6.bean.UserDaoImpl"/><bean id="PersonDaoImpl" class="com.hbnu.spring6.bean.PersonDaoImpl"/>
ApplicationContext applicationContext=new ClassPathXmlApplicationContext("bean.xml");UserDao userDao = applicationContext.getBean(UserDao.class);System.out.println(userDao);userDao.run();
2.依賴注入
如果是原生Java,set方法和構造器方法怎么注入?
//set方法注入Book book=new Book();book.setAuthor("牛頓");//通過構造器注入Book book1=new Book("1","鶴");
接著看用配置文件如何注入。。。
①根據set方法注入
創建一個Book類,定義屬性,生成屬性set方法之后,在spring配置文件里配置。。。
<!-- 1.set方法完成注入--><bean id="book" class="com.hbnu.spring6.DI.Book"><property name="bname" value="spring"/><property name="author" value="小唐"/></bean><!--name對應Book類里面的屬性-->
測試一下set方法注入:
@Testpublic void testSet(){ApplicationContext applicationContext=new ClassPathXmlApplicationContext("bean-DI.xml");Book book = (Book)applicationContext.getBean("book");System.out.println(book);//Book{bname='spring', author='小唐'}}
輸出Book類里重寫的toString
②根據構造器注入
創建一個Book類,定義屬性,生成有參數的構造方法,在spring配置文件里配置。。。
<!-- 2.有參構造注入--><bean id="bookCons" class="com.hbnu.pojo.Book"><constructor-arg name="author" value="小李"/><constructor-arg name="name" value="Spring學習之有參構造注入"/></bean>
測試跟上面一樣,只是配置文件里寫的內容不一樣了
3.特殊值處理
①字面量賦值
②null值 用null標簽表示空值
③xml實體 用<>
表示大小尖括號
④CDATA節 <![CDATA[a<b]]>
4.為對象類型屬性賦值
創建兩個類對象:dept和emp
在emp類里寫一個work方法
public class Emp {//對象類型屬性:員工屬于某個部門private Dept dept;private String ename;private String age;public void work(){System.out.println(ename+" is working...");dept.info();}...}
①引用外部bean
1.引用外部bean注入<bean id="dept" class="com.hbnu.DITest.Dept"><property name="dname" value="安保部"/></bean><bean id="emp" class="com.hbnu.DITest.Emp">
<!-- 普通類型屬性注入--><property name="ename" value="lucy"/><property name="age" value="23"/>
<!-- 對象類型屬性注入-->
<!-- 表示引入外部bean,這里的ref是上面id為dept的值--><property name="dept" ref="dept"/></bean>
②使用內部bean
<!-- 2.使用內部bean注入--><bean id="emp" class="com.hbnu.DITest.Emp"><property name="age" value="25"/><property name="ename" value="joey"/><property name="dept"><bean id="dept" class="com.hbnu.DITest.Dept"><property name="dname" value="財務部"/></bean></property></bean>
③ 使用級聯賦值
<!-- 3.級聯賦值--><bean id="emp" class="com.hbnu.DITest.Emp"><property name="ename" value="mary"/><property name="age" value="25"/><property name="dept" ref="dept"/><property name="dept.dname" value="測試部"/></bean><bean id="dept" class="com.hbnu.DITest.Dept"><!--這一行可要可不要,僅作參考,上面已經給部門名稱賦值為了測試部--><property name="dname" value="技術部"/></bean>
測試
@Testpublic void testemp(){ApplicationContext applicationContext=new ClassPathXmlApplicationContext("bean-ditest.xml");Emp emp=applicationContext.getBean("emp",Emp.class);emp.work();}
5.復雜類型注入
①數組類型注入
在 Emp 類里面加一個數組屬性private String[] hobbies;
后生成get、set方法,編寫xml文件
<!-- 數組類型賦值--><bean id="dept" class="com.hbnu.DITest.Dept"/><bean id="emp" class="com.hbnu.DITest.Emp"><property name="ename" value="tom"/><property name="age" value="34"/><property name="dept" ref="dept"/><property name="hobbies"><array><value>抽煙</value><value>喝酒</value><value>燙頭</value></array></property></bean>
②集合類型注入
在Dept類里加一個員工listprivate List<Emp> empList;
,一個部門里可以有多個員工
<!-- 為list集合賦值--><bean id="emp1" class="com.hbnu.DITest.Emp"><property name="ename" value="tom1"/><property name="age" value="31"/></bean><bean id="emp2" class="com.hbnu.DITest.Emp"><property name="ename" value="tom2"/><property name="age" value="32"/></bean><bean id="dept" class="com.hbnu.DITest.Dept"><property name="dname" value="技術部"/><property name="empList"><list><ref bean="emp1"></ref><ref bean="emp2"></ref></list></property></bean>
輸出技術部里的員工:
Dept{dname='技術部', empList=[Emp{dept=null, ename='tom1', age='31', hobbies=null}, Emp{dept=null, ename='tom2', age='32', hobbies=null}]}
③map類型注入
創建Student和Teacher類
對于學生來說,一個學生對應多個講師
<!-- map類型注入--><bean id="teacherone" class="com.hbnu.DITest.Teacher"><property name="tid" value="121"/><property name="tname" value="莉莉"/></bean><bean id="teachertwo" class="com.hbnu.DITest.Teacher"><property name="tid" value="122"/><property name="tname" value="小月"/></bean><bean id="student" class="com.hbnu.DITest.Student"><property name="sid" value="111"/><property name="sname" value="張三"/><property name="teacherMap"><map><entry><key><value>10010</value></key><ref bean="teacherone"></ref></entry><entry><key><value>10011</value></key><ref bean="teachertwo"></ref></entry></map></property></bean>
輸出{10010=Teacher{Tname='莉莉', Tid='121'}, 10011=Teacher{Tname='小月', Tid='122'}}
④引用集合類型的bean
創建一個Lesson類(getter、setter、重寫toString),再在Student類里面加一個 public List<Lesson> lessonList;
,一個學生有多個老師,選多門課
<bean id="lesson1" class="com.hbnu.DITest.Lesson"><property name="lessonname" value="前端開發"/></bean><bean id="lesson2" class="com.hbnu.DITest.Lesson"><property name="lessonname" value="java開發"/></bean><bean id="teacher1" class="com.hbnu.DITest.Teacher"><property name="tname" value="張老師"/><property name="tid" value="001"/></bean><bean id="teacher2" class="com.hbnu.DITest.Teacher"><property name="tname" value="王老師"/><property name="tid" value="002"/></bean><bean id="student" class="com.hbnu.DITest.Student"><property name="sid" value="1000"/><property name="sname" value="lucy"/><!-- 注入list、map類型屬性--><property name="lessonList" ref="lessonlist"/><property name="teacherMap" ref="teachermap"/></bean><util:list id="lessonlist"><ref bean="lesson1"/><ref bean="lesson2"/></util:list><util:map id="teachermap"><entry><key><value>10010</value></key><ref bean="teacher1"/></entry><entry><key><value>10011</value></key><ref bean="teacher2"/></entry></util:map>
</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"xmlns:util="http://www.springframework.org/schema/util"xsi:schemaLocation="http://www.springframework.org/schema/utilhttp://www.springframework.org/schema/util/spring-util.xsdhttp://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsd">
輸出:
{10010=Teacher{Tname='張老師', Tid='001'}, 10011=Teacher{Tname='王老師', Tid='002'}}
[Lesson{lessonname='前端開發'}, Lesson{lessonname='java開發'}]
脫離數據庫思想注入復雜類型,創建一個dataType類,里面擁有private String[] arr;private List<String> lists;private Map<String,String> maps;private Properties properties;
屬性,編寫xml文件:
<!-- 注入復雜類型--><bean id="dataType" class="com.hbnu.pojo.DataType">
<!-- 1.注入數組類型--><property name="arr"><list><value>斯蒂芬金</value><value>毛姆</value><value>陀思妥耶夫斯基</value><value>茨威格</value></list></property><!-- 2.注入list集合類型--><property name="lists"><list><value>馬克吐溫</value><value>馬克吐溫</value><value>馬克吐溫</value><value>馬克吐溫</value></list></property><!-- 3.注入map集合類型--><property name="maps"><map><entry key="username" value="雨果"/><entry key="age" value="1984"/></map></property><!-- 4.注入properties--><property name="properties"><props><prop key="driverClass">com.mysql.cj.jdbc.Driver</prop><prop key="url">jdbc:mysql://localhost:3306/class2110?serverTimezone=GMT&useSSL=false</prop><prop key="username">root</prop><prop key="password">123456</prop></props></property></bean>
6.p命名空間
<!-- p命名空間注入--><bean id="studentp" class="com.hbnu.DITest.Student"p:sid="100" p:sname="mary" p:lessonList-ref="lessonlist" p:teacherMap-ref="teachermap">
頭部加一行
xmlns:p="http://www.springframework.org/schema/p"
7.bean的作用域
概念
在Spring中可以通過配置bean標簽的scope屬性來指定bean的作用域范圍
取值 | 含義 | 創建對象的時機 |
---|---|---|
singleton | 在IOC容器中,這個bean的對象始終為單實例 | IOC容器初始化時 |
prototype | 這個bean在IOC容器中有多個實例 | 獲取bean時 |
@Testpublic void testOrders(){ApplicationContext applicationContext=new ClassPathXmlApplicationContext("bean-ditest.xml");Orders orders=applicationContext.getBean("orders",Orders.class);Orders orders2=applicationContext.getBean("orders",Orders.class);System.out.println(orders);System.out.println(orders2);}
8.bean的生命周期
1.bean對象創建(調用無參數構造)
2.給bean對象設置相關屬性
3.bean后置處理器(初始化之前)
4.bean對象初始化(調用指定初始化方法)
5.bean后置處理器(初始化之后)
6.bean對象創建完成
7.bean對象銷毀(配置指定銷毀的方法)
8.IoC容器關閉
3.4 基于注解管理bean
步驟:1.引入依賴 2.開啟組件掃描 3.使用注解定義bean 4.依賴注入
1.開啟組件掃描
開啟組件掃描(開啟此功能后,spring就會自動從掃描指定的包,及其子包下的所有類。如果類上使用了@Component注解,就將該類裝配到容器中)
<context:component-scan base-package="com.hbnu"></context:component-scan>
2.使用注解定義bean
注解 | 說明 |
---|---|
@Component | 該注解用于描述Spring中的bean,僅僅表示容器中的一個組件,并且可以作用在應用的任何層次,例如Service層,Dao層。 |
@Repository | 該注解用于將數據訪問層(Dao層)的類標識為S加粗樣式pring中的bean,其功能與@Component相同 |
@Service | 該注解通常作用在業務層(Service層),用于將業務層的類標識為Spring中的bean,其功能與@Component相同 |
@Controller | 該注解通常作用在控制層(如SpringMVC中的Controller),用于將控制層的類標識為Spring中的bean,其功能與@Component相同 |
3.@Autowired注入
注入可以理解為導包,注入后就可以調用方法
在controller里注入service,在service里注入dao
寫個測試感受一下@Autowired注入…項目結構如下:
首先在bean.xml里編寫代碼進行包掃描,然后創建UserDao以及其實現類,
接著創建UserService以及其實現類,并將UserDao注入到UserService中。
最后創建UserController,將UserService注入后進行測試。
public class Test {public static void main(String[] args) {ApplicationContext applicationContext=new ClassPathXmlApplicationContext("bean.xml");UserController userController=applicationContext.getBean(UserController.class);userController.add();}
}