下面來模擬一下Spring的注解開發,關注點就是你<mark>不需要`new`一個對象,依然可以訪問該對象的方法和屬性,那是因為該類已經被反射并`new`好了</mark>。
<br/>
本節示例代碼下載地址:https://gitee.com/flymini/spring/tree/master/mnzj
<br/>
步驟如下:
[TOC]
# 1. 創建一個Web項目
# 2. 代碼封裝過程
**1. 封裝注解**
(1)*`org.example.mnzj.anno.Bean`*
```java
package org.example.mnzj.anno;
import java.lang.annotation.*;
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface Bean {
String value();
}
```
(2)*`org.example.mnzj.anno.Property`*
```java
package org.example.mnzj.anno;
import java.lang.annotation.*;
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface Property {
String value();
}
```
**2. 將上面定義的兩個注解應用到Java類中**
(1)dao層
*`org.example.mnzj.dao.StudentDao`*
```java
package org.example.mnzj.dao;
public interface StudentDao {
Integer queryStudentCounts();
}
```
*`org.example.mnzj.dao.impl.StudentDaoImpl`*
```java
package org.example.mnzj.dao.impl;
import org.example.mnzj.anno.Bean;
import org.example.mnzj.dao.StudentDao;
@Bean("studentDao")
public class StudentDaoImpl implements StudentDao {
public Integer queryStudentCounts() {
System.out.println("StudentDaoImpl.queryStudentCounts");
return null;
}
}
```
(2)service層
*`org.example.mnzj.service.StudentService`*
```java
package org.example.mnzj.service;
public interface StudentService {
Integer queryStudentCounts();
}
```
*`org.example.mnzj.service.impl.StudentServiceImpl`*
```java
package org.example.mnzj.service.impl;
import org.example.mnzj.anno.Bean;
import org.example.mnzj.anno.Property;
import org.example.mnzj.dao.StudentDao;
import org.example.mnzj.service.StudentService;
@Bean("studentService")
public class StudentServiceImpl implements StudentService {
@Property("studentDao")
private StudentDao studentDao;
public Integer queryStudentCounts() {
System.out.println("StudentServiceImpl.queryStudentCounts");
return studentDao.queryStudentCounts();
}
}
```
**3. 對被`@Bean`和`@Property`注解標記的類進行反射**
*`org.example.mnzj.common.AnnotationFactory`*
```java
package org.example.mnzj.common;
import org.example.mnzj.anno.Bean;
import org.example.mnzj.anno.Property;
import java.io.File;
import java.lang.reflect.Field;
import java.util.HashMap;
import java.util.Map;
public class AnnotationFactory {
private static Map<String, Object> map = new HashMap<String, Object>();
public static Object getInstance(String key) {
return map.get(key);
}
static {
// 獲取.java文件編譯后的.class文件所在的目錄
String path = AnnotationFactory.class.getClassLoader().getResource("").getPath();
File file = new File(path);
initBeans(path); // 初始化被@Bean注解標記的類
initProperties(); // 初始化被@Property注解標記的類
}
/**
* 找到并初始化被@Bean注解標記的類
*
* @param path String, .class文件所在的目錄
*/
private static void initBeans(String path) {
File file = new File(path);
File[] subFiles = file.listFiles();
for (File subFile : subFiles) {
if (subFile.isDirectory()) {
initBeans(subFile.getPath());
} else {
// 實例化被@Bean注解的類
initBean(subFile.getPath());
}
}
}
/**
* 實例化被@Bean注解標記的類
*
* @param path String, .class文件所在的目錄
*/
private static void initBean(String path) {
if (!path.endsWith(".class")) {
return;
}
// 獲取被@Bean注解標記的類的完全限定名
int begin = AnnotationFactory.class.getClassLoader().getResource("").getPath().length();
int end = path.lastIndexOf(".");
String className = path.substring(begin - 1, end);
className = className.replace("\\", ".");
try {
Class cls = Class.forName(className); // 加載類
// 判斷該類是否被@Bean注解標記
if (cls.isAnnotationPresent(Bean.class)) {
Object instance = cls.newInstance(); // 實例化該類
Bean annotation = (Bean) cls.getDeclaredAnnotation(Bean.class);
map.put(annotation.value(), instance);
}
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* 實例化被@Property注解標記的屬性
*/
private static void initProperties() {
for (Object instance : map.values()) {
Field[] fields = instance.getClass().getDeclaredFields();
for (Field field : fields) {
if (field.isAnnotationPresent(Property.class)) {
Property annotation = field.getAnnotation(Property.class);
String key = annotation.value();
Object obj = map.get(key);
field.setAccessible(true);
try {
field.set(instance, obj);
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
}
}
```
**4. 封裝controller層**
(1)*`org.example.mnzj.common.BaseServlet`*
```java
package org.example.mnzj.common;
import org.example.mnzj.anno.Property;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
public class BaseServlet extends HttpServlet {
protected void doGet(HttpServletRequest request, HttpServletResponse response) {
String methodName = request.getParameter("method");
try {
Method method = this.getClass().getDeclaredMethod(methodName, HttpServletRequest.class, HttpServletResponse.class);
method.setAccessible(true);
String url = (String) method.invoke(this, request, response);
if (url == null) {
return;
}
if (url.startsWith("redirect:")) {
int index = url.indexOf(":");
url = url.substring(index + 1);
response.sendRedirect(url + ".jsp");
} else {
request.getRequestDispatcher(url + ".jsp").forward(request, response);
}
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* 對繼承該類的子類的成員屬性進行賦值
*
* @throws ServletException
*/
@Override
public void init() throws ServletException {
Field[] fields = this.getClass().getDeclaredFields();
for (Field field : fields) {
if (field.isAnnotationPresent(Property.class)) {
Property annotation = field.getAnnotation(Property.class);
String key = annotation.value();
Object obj = AnnotationFactory.getInstance(key);
try {
field.setAccessible(true);
field.set(this, obj);
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
}
}
}
}
```
(2)*`org.example.mnzj.controller.StudentController`*
```java
package org.example.mnzj.controller;
import org.example.mnzj.anno.Property;
import org.example.mnzj.common.BaseServlet;
import org.example.mnzj.service.StudentService;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
@WebServlet("/student")
public class StudentController extends BaseServlet {
@Property("studentService")
private StudentService studentService;
private String welcome(HttpServletRequest request, HttpServletResponse response) {
System.out.println("IndexController.doGet方法執行-----start");
studentService.queryStudentCounts();
System.out.println("IndexController.doGet方法執行-----end");
return null;
}
}
```
**5. 測試**
啟動項目后訪問 http://localhost:8080/mnzj_war_exploded/student?method=welcome
,控制臺打印的結果如下:
```
IndexController.doGet方法執行-----start
StudentServiceImpl.queryStudentCounts
StudentDaoImpl.queryStudentCounts
IndexController.doGet方法執行-----end
```
上面寫了那么多,其目的就想實現一個功能:我不`new`一個對象,但是我依然能夠訪問它的方法。
<br/>
在上面的代碼中,`StudentServiceImpl ` 中調用了`StudentDao.queryStudentCounts`方法;在 `StudentController` 中調用了`StudentService.queryStudentCounts`方法。但是請注意我并沒有實例化`StudentDaoImpl`和`StudentServiceImpl`類,因為這兩個類被`@Bean`注解標記,作為屬性時被`@Property`注解標記,然后在 `AnnotationFactory` 反射了被這兩個注解標記的類并實例化了這個兩個類,并在 `BaseServlet` 的`init`方法中對繼承`BaseServlet`的子類的屬性進行賦值,所以當我們調用方法時,對象已在`AnnotationFactory`幫我們自動實例化了。
- Mybatis
- mybatis是什么
- mybatis優缺點
- 環境搭建
- 使用步驟
- 傳參方式
- 無需傳參
- 一個參數
- 多個參數
- 增/刪/改
- 查詢
- 單表查詢
- 一對一查詢
- 一對多查詢
- 動態SQL
- 注解操作
- Spring
- Spring什么
- Spring優點
- Spring組成
- 第一個Spring程序
- 兩大核心技術
- IoC控制反轉
- IoC思想
- IoC容器使用步驟
- 屬性注入
- IoC注入方式
- 模擬IoC實現
- AOP
- AOP概念
- AOP原理
- AOP關鍵術語
- AOP編程過程
- 切入點規則
- 5種增強方式
- Spring注解開發
- 注解開發的優勢
- Bean注解開發
- AOP注解開發
- 完全注解開發
- 模擬Spring注解開發
- 自動裝配
- 配置文件拆分
- SpringBean
- Bean常用屬性
- Bean的作用域
- Bean的生命周期
- Spring整合MyBatis
- 整合步驟
- SqlSessionTemplate
- 業務層添加事務
- 事務的作用
- 配置文件事務
- 注解事務
- 事務參數
- SpringMVC
- SpringMVC是什么
- 環境搭建
- 請求流程
- 核心組件
- 前后端交互
- 簡單交互演示
- 常用注解
- 后端數據傳遞至前端
- ServletAPI
- 訪問靜態資源
- 異常處理
- HandlerExceptionResolver
- 局部異常
- 全局異常
- 轉發與重定向
- 轉發演示
- 重定向演示
- 轉發與重定向的區別
- 獲取表單數據
- 表單標簽
- REST風格的URL
- 異步處理
- 異步請求
- JSON數據處理
- 中文亂碼處理
- 日期處理
- 上傳文件
- 攔截器
- 視圖解析器
- 視圖類型
- 多視圖解析器
- 自定義pdf視圖
- JSR303數據驗證
- JSR303是什么
- 常用約束
- 使用步驟
- SpringMVC整合Mybatis
- 整合步驟
- Mybatis分頁插件
- SpringBoot
- SpringBoot是什么
- 環境搭建
- SpringBoot啟動分析
- SpringBoot啟動類
- 啟動過程
- SpringBoot配置文件
- 配置文件類型
- 更改配置文件
- 讀取配置文件
- 占位符
- 配置優先級
- 自定義IoC容器
- 定義方式
- 引入Spring配置文件
- @Configuration
- SpringBoot自動配置
- 自動配置原理
- 條件注解
- 自動配置報告
- 自定義自動配置
- 關閉自動配置
- 接管自動配置
- 多環境配置
- CommandLineRunner
- SpringBoot與Web開發
- 引入模板引擎
- Thymeleaf模板
- Freemarker模板
- 靜態資源訪問
- webjars
- 靜態資源位置
- ico圖標
- 指定首頁
- 更換Web服務器
- 國際化
- 攔截器
- 錯誤處理機制
- 錯誤處理機制原理
- 定制錯誤頁面
- 定制錯誤數據
- 上傳文件
- 注冊servlet三大組件
- 注冊Servlet
- 注冊過濾器
- 注冊監聽器
- 外部Tomcat與jsp模板
- 前后端交互
- 傳遞json字符串
- 傳遞js對象
- 傳遞表單
- 下載功能
- Swagger2文檔
- SpringBoot整合JDBC
- 整合步驟
- 核心API
- JdbcTemplate
- 增刪改
- 查詢
- NamedParameterJdbcTemplate
- 增刪改
- 查詢
- SpringBoot整合Mybatis
- 整合步驟
- 切換為Druid數據源
- 添加事務
- Mybatis分頁插件
- 場景啟動器
- 場景啟動器是什么
- 自定義場景啟動器
- SpringBoot與日志
- 日志框架
- slf4j日志
- slf4j日志實現
- 統一切換為slf4j
- 日志配置
- 日志文件
- 切換日志框架
- 切換日志場景啟動器
- SpringBoot與緩存
- JSR107緩存技術
- Spring緩存抽象
- 緩存注解
- SpEL表達式
- 使用緩存
- 自定義key生成器
- 緩存工作原理與流程
- SpringBoot整合Redis
- 整合步驟
- 初步使用
- 序列化機制
- 緩存管理器
- SpringBoot與任務
- 異步任務
- 實現異步任務
- 注意事項與原理
- 自定義線程池
- 定時任務
- cron表達式
- 創建定時任務
- @Scheduled參數
- 動態時間
- 郵件任務
- Quartz定時任務
- Quartz是什么
- 創建定時任務
- 觸發器與任務
- 任務的CURD
- 兩種觸發器
- 并發問題
- 持久化
- 任務持久化
- Quartz集群
- misfire策略
- 打包插件
- appassembler-maven-plugin
- appassembler與assembly配合