下面来模拟一下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配合