当前位置: 首页 > news >正文

day24——Java高级技术深度解析:单元测试、反射、注解与动态代理

文章目录

    • 一、单元测试:JUnit框架精要
      • 1.1 单元测试核心概念
      • 1.2 JUnit快速入门实战
        • 基础步骤:
        • 断言机制验证结果
      • 1.3 JUnit核心注解解析
    • 二、反射机制:框架设计的基石
      • 2.1 反射核心概念
      • 2.2 获取Class对象的三种方式
      • 2.3 反射操作类成分
        • 获取并执行构造器
        • 操作成员变量
        • 调用成员方法
      • 2.4 反射高级应用
        • 突破泛型限制
        • 开发通用对象框架
    • 三、注解:元编程利器
      • 3.1 注解基础概念
      • 3.2 元注解:注解的注解
      • 3.3 注解解析实战
      • 3.4 反射应用场景:模拟JUnit测试框架
    • 四、动态代理:优雅的增强方案
      • 4.1 代理模式核心思想
      • 4.2 JDK动态代理实现
        • 定义接口与实现类
        • 创建代理工具类
        • 使用代理对象
      • 4.3 动态代理应用:性能监控
    • 五、总结:四大技术的关联与应用

本文全面解析Java高级技术核心内容:单元测试、反射机制、注解应用与动态代理实现,通过理论讲解与代码实践助你掌握框架底层原理

一、单元测试:JUnit框架精要

1.1 单元测试核心概念

单元测试是针对**最小功能单元(方法级别)**编写的测试代码,用于验证功能正确性。传统main方法测试存在三大痛点:

  • 无法灵活选择测试方法
  • 不能自动生成测试报告
  • 测试过程无法自动化

JUnit作为第三方开源测试框架,完美解决了这些问题:

  • ✅ 支持选择性执行测试方法或批量执行
  • ✅ 自动生成可视化测试报告(绿色成功/红色失败)
  • ✅ 高度灵活的测试代码编写

1.2 JUnit快速入门实战

基础步骤:
  1. 创建测试类
  2. 编写公共无参无返回值的测试方法
  3. 使用@Test注解标记测试方法(idea 可以直接导入,Alt+Enter)
  4. 在测试方法中调用被测代码(在@Test的方法任务地方右击,点击run 方法名)
// StringUtil工具类
public class StringUtil {public static void printNumber(String name) {System.out.println("名字长度:" + name.length());}
}// 测试类
public class StringUtilTest {@Testpublic void testPrintNumber() {StringUtil.printNumber("admin");StringUtil.printNumber(null); // 测试异常情况}
}
断言机制验证结果
@Test
public void testGetMaxIndex() {int index = StringUtil.getMaxIndex("admin");// 断言预测结果:期望值4, 实际值indexAssert.assertEquals("方法内部有Bug", 4, index);
}

注意:很多个测试类中多个测试方法,可以直接点击项目,右击run ‘All Tests’
在这里插入图片描述
在这里插入图片描述

1.3 JUnit核心注解解析

注解JUnit4JUnit5执行时机
初始化@Before@BeforeEach每个@Test方法执行
清理@After@AfterEach每个@Test方法执行
全局初始化@BeforeClass@BeforeAll所有测试方法执行(static)
全局清理@AfterClass@AfterAll所有测试方法执行(static)

资源管理实战:

public class ResourceTest {private static Socket socket;@BeforeClasspublic static void init() {socket = new Socket(); // 初始化资源}@AfterClasspublic static void cleanup() {socket.close(); // 释放资源}@Testpublic void testNetwork() {// 使用socket进行测试}
}

二、反射机制:框架设计的基石

2.1 反射核心概念

反射是在运行时获取类的字节码对象(Class对象),并动态解析类的全部成分:

  • 🏗️ 构造器(Constructor对象)
  • 📦 成员变量(Field对象)
  • ⚙️ 成员方法(Method对象)

应用场景:

  • IDE代码提示功能
  • Spring框架的IoC容器
  • MyBatis的ORM映射
  • 通用工具类开发

2.2 获取Class对象的三种方式

// 1. 类名.class
Class c1 = Student.class;// 2. Class.forName("全类名")
Class c2 = Class.forName("com.example.Student");// 3. 对象.getClass()
Student s = new Student();
Class c3 = s.getClass();System.out.println(c1 == c2); // true
System.out.println(c2 == c3); // true

在这里插入图片描述

2.3 反射操作类成分

获取并执行构造器
Class<Cat> catClass = Cat.class;// 获取私有构造器
Constructor<Cat> constructor = catClass.getDeclaredConstructor(String.class, int.class);// 暴力反射(解除私有限制)
constructor.setAccessible(true); // 执行构造器创建实例
Cat cat = constructor.newInstance("Tom", 3);

在这里插入图片描述

操作成员变量

在这里插入图片描述

Field nameField = catClass.getDeclaredField("name");
nameField.setAccessible(true);// 设置字段值
nameField.set(cat, "Jerry");// 获取字段值
String name = (String) nameField.get(cat);

在这里插入图片描述

调用成员方法

在这里插入图片描述

Method runMethod = catClass.getDeclaredMethod("run");
runMethod.setAccessible(true);// 调用无参方法
runMethod.invoke(cat);// 调用有参方法
Method eatMethod = catClass.getDeclaredMethod("eat", String.class);
eatMethod.setAccessible(true);
String result = (String) eatMethod.invoke(cat, "fish");

2.4 反射高级应用

突破泛型限制
// 编译时泛型检查
ArrayList<Integer> list = new ArrayList<>();
list.add(100);
// list.add("字符串"); // 编译报错// 运行时通过反射绕过泛型检查
Method addMethod = ArrayList.class.getDeclaredMethod("add", Object.class);
addMethod.invoke(list, "字符串"); // 成功添加
开发通用对象框架

在这里插入图片描述

public class ObjectFrame {public static void saveObject(Object obj) throws Exception {Class<?> c = obj.getClass();Field[] fields = c.getDeclaredFields();for (Field field : fields) {field.setAccessible(true);//禁止检查访问控制String fieldName = field.getName();Object value = field.get(obj);// 写入字段名和值到文件}}
}// 使用示例
Student stu = new Student("张三", 20);
ObjectFrame.saveObject(stu);

在这里插入图片描述

三、注解:元编程利器

3.1 注解基础概念

注解(Annotation)是JDK5引入的代码标记机制,用于对类、方法、字段等进行标注。核心作用:

  • 🔖 标记程序元素
  • 💡 提供元数据信息
  • ⚙️ 驱动特殊处理逻辑

自定义注解格式:
在这里插入图片描述
在这里插入图片描述

public @interface MyAnnotation {// 属性声明String value(); int count() default 1;String[] tags();
}

在这里插入图片描述

3.2 元注解:注解的注解

在这里插入图片描述

元注解用于修饰自定义注解:

元注解作用常用值
@Target指定注解使用范围TYPE, FIELD, METHOD等
@Retention指定注解生命周期SOURCE, CLASS, RUNTIME
@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface Book {String value(); // 书名double price() default 100;String[] authors(); // 作者数组
}

3.3 注解解析实战

在这里插入图片描述

@Book(value = "Java核心技术", authors = {"Cay S. Horstmann"})
public class Textbook {@Book(value = "Effective Java", price = 128.0, authors = {"Joshua Bloch"})public void recommend() {}
}// 解析类注解
Class<Textbook> clazz = Textbook.class;
if (clazz.isAnnotationPresent(Book.class)) {Book book = clazz.getAnnotation(Book.class);System.out.println("书名: " + book.value());
}// 解析方法注解
Method method = clazz.getMethod("recommend");
if (method.isAnnotationPresent(Book.class)) {Book book = method.getAnnotation(Book.class);System.out.println("价格: " + book.price());
}

3.4 反射应用场景:模拟JUnit测试框架

在这里插入图片描述

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface MyTest {}public class TestRunner {public static void main(String[] args) throws Exception {Class<?> testClass = MyTestClass.class;Object instance = testClass.newInstance();for (Method method : testClass.getDeclaredMethods()) {if (method.isAnnotationPresent(MyTest.class)) {method.invoke(instance); // 执行测试方法}}}
}

四、动态代理:优雅的增强方案

在这里插入图片描述

4.1 代理模式核心思想

当目标对象无法或不想直接完成操作时,通过代理对象控制对目标对象的访问

  • 💼 代理对象持有目标对象引用
  • 🔄 代理对象拦截方法调用
  • ➕ 在方法执行前后添加额外操作

4.2 JDK动态代理实现

定义接口与实现类
public interface Star {String sing(String song);void dance();
}public class BigStar implements Star {private String name;public BigStar(String name) {this.name = name;}@Overridepublic String sing(String song) {return name + "演唱:" + song;}@Overridepublic void dance() {System.out.println(name + "跳舞");}
}
创建代理工具类
public class ProxyUtil {public static Star createProxy(BigStar target) {return (Star) Proxy.newProxyInstance(ProxyUtil.class.getClassLoader(),new Class[]{Star.class},(proxy, method, args) -> {// 前置增强if ("sing".equals(method.getName())) {System.out.println("准备话筒,收费20万");} else if ("dance".equals(method.getName())) {System.out.println("准备场地,收费100万");}// 调用目标方法Object result = method.invoke(target, args);// 后置增强System.out.println("表演结束,结算费用");return result;});}
}
使用代理对象
public static void main(String[] args) {BigStar star = new BigStar("杨超越");Star proxy = ProxyUtil.createProxy(star);String result = proxy.sing("卡路里");System.out.println(result);proxy.dance();
}

4.3 动态代理应用:性能监控

public class PerformanceProxy {public static UserService createProxy(UserService target) {return (UserService) Proxy.newProxyInstance(PerformanceProxy.class.getClassLoader(),new Class[]{UserService.class},(proxy, method, args) -> {long start = System.currentTimeMillis();Object result = method.invoke(target, args);long end = System.currentTimeMillis();System.out.println(method.getName() + "方法执行耗时: " + (end - start) + "ms");return result;});}
}// 使用示例
UserService userService = new UserServiceImpl();
UserService proxy = PerformanceProxy.createProxy(userService);
proxy.login("admin", "123456");

五、总结:四大技术的关联与应用

技术核心能力典型应用场景
单元测试自动化验证代码逻辑保障代码质量,回归测试
反射运行时动态解析类结构框架设计,动态代码生成
注解声明式配置程序元素简化配置,驱动框架行为
动态代理无侵入增强对象功能AOP实现,性能监控,事务管理

技术联动示例:

  1. 使用注解标记测试方法(@Test)
  2. 通过反射解析测试类信息
  3. 利用动态代理增强测试方法(添加事务控制)
  4. 通过单元测试验证功能正确性

掌握这四大Java高级技术,不仅能编写更健壮高效的代码,更能深入理解主流框架的设计思想,为成为架构师奠定坚实基础。

学习建议:

  1. 先掌握单元测试保证代码质量
  2. 深入理解反射机制原理
  3. 练习自定义注解及解析
  4. 动手实现动态代理案例
  5. 研究Spring等框架的源码实现
http://www.dtcms.com/a/289111.html

相关文章:

  • [Python] -项目实战类3- 用Python制作一个记事本应用
  • CVE-2022-41128
  • Python数据处理库与语法总结
  • API获取及调用(以豆包为例实现图像分析)
  • FreeRTOS任务创建与删除
  • 掌握配置文件(三):运用Profile实现多环境配置隔离
  • 三级知识点汇总(详解)【c++】——3
  • 让不符合要求的任何电脑升级Windows11
  • 《通信原理》学习笔记——第五章
  • 开源安全大模型Foundation-Sec 8B的安全实践
  • 分享如何在保证画质的前提下缩小视频体积实用方案
  • 【记某次线上消息积压问题排查及解决方案】
  • 基于Pytorch的人脸识别程序
  • 基于FPGA实现ARINC818
  • Milvus Dify 学习笔记
  • Unity学习笔记(五)——3DRPG游戏(2)
  • DFS 迷宫问题 难度:★★★★☆
  • Python适配器模式详解:让不兼容的接口协同工作
  • CSS中Padding与Margin的区别
  • 机器学习-线性回归
  • 【数据结构】「队列」(顺序队列、链式队列、双端队列)
  • ubuntu24.04安装CUDA和VLLM
  • 企业级安全威胁检测与响应(EDR/XDR)架构设计
  • WireShark抓包分析TCP数据传输过程与内容详解
  • 多目标轨迹优化车道变换规划:自动驾驶轨迹规划新范式:基于Frenet坐标系的车道变换算法全解析
  • Node.js Express keep-alive 超时时间设置
  • spring boot2升级boot3
  • Linux简单了解历史
  • 大数据之路:阿里巴巴大数据实践——离线数据开发
  • RTC外设详解