import com.wm.javayu.model.Student;
import com.wm.javayu.model.Teacher;
import org.springframework.util.CollectionUtils;
import org.springframework.util.ObjectUtils;
import java.lang.reflect.*;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;
* @class_name: ReFlectUtil
* @description:
* @author: wm_yu
* @create: 2019/07/19
public class ReFlectUtil {
*反射获取List中的数据
* @param args
public static void main(String[] args) {
Teacher teacher = new Teacher();
List
list = new ArrayList();
list.add(new Student("张三", "男"));
list.add(new Student("李四", "女"));
list.add(new Student("王五", "女"));
teacher.setList(list);
getList(teacher,Student.class);
* 反射获取对象中的list数据
* @param object
* @param dateClass
* @param
public static void getList(Object object,T dateClass){
List resultList = new ArrayList<>();
if(!ObjectUtils.isEmpty(object)){
Field[] fields = getAllFields(object.getClass());
Field[] filterList= filterField(fields);
Arrays.stream(filterList).forEach(var -> {
//List集合
if(List.class.isAssignableFrom(var.getType())){
Type type = var.getGenericType();
if(type instanceof ParameterizedType){
if(!var.isAccessible()){
var.setAccessible(true);
//获取到属性值的字节码
try {
Class> clzz = var.get(object).getClass();
//反射调用获取到list的size方法来获取到集合的大小
Method sizeMethod = clzz.getDeclaredMethod("size");
if(!sizeMethod.isAccessible()){
sizeMethod.setAccessible(true);
//集合长度
int size = (int) sizeMethod.invoke(var.get(object));
//循环遍历获取到数据
for (int i = 0; i < size; i++) {
//反射获取到list的get方法
Method getMethod = clzz.getDeclaredMethod("get", int.class);
//调用get方法获取数据
if(!getMethod.isAccessible()){
getMethod.setAccessible(true);
T var1 = (T) getMethod.invoke(var.get(object), i);
resultList.add(var1);
} catch (Exception e) {
e.printStackTrace();
resultList.stream().forEach(var -> {
System.out.println("反射获取到的数据是什么:" + var);
* 反射获取所有的字段
* @param c
* @return
public static Field[] getAllFields(Class c){
List fieldList = new ArrayList<>();
while (c!= null){
fieldList.addAll(new ArrayList<>(Arrays.asList(c.getDeclaredFields())));
c= c.getSuperclass();
Field[] fields = new Field[fieldList.size()];
fieldList.toArray(fields);
return fields;
public static Field[] filterField(Field[] fields){
List tempList = Arrays.stream(fields).filter(field -> null != field
&& !Modifier.isFinal(field.getModifiers())
&& !Modifier.isStatic(field.getModifiers())
&& !Modifier.isAbstract(field.getModifiers())).collect(Collectors.toList());
int arrLength = CollectionUtils.isEmpty(tempList) ? 1:tempList.size();
Field[] resultArr = new Field[arrLength];
if(!CollectionUtils.isEmpty(tempList)){
tempList.toArray(resultArr);
return resultArr;
package com.wm.javayu.model;
* @class_name: Student
* @description:
* @author: wm_yu
* @create: 2019/07/19
public class Student {
public Student(String name, String gender) {
this.name = name;
this.gender = gender;
private String name;
private String gender;
public String getGender() {
return gender;
public void setGender(String gender) {
this.gender = gender;
public String getName() {
return name;
public void setName(String name) {
this.name = name;
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", gender='" + gender + '\'' +
package com.wm.javayu.model;
import java.util.List;
* @class_name: Teacher
* @description:
* @author: wm_yu
* @create: 2019/07/19
public class Teacher {
List<Student> list;
public List<Student> getList() {
return list;
public void setList(List<Student> list) {
this.list = list;
工具类:package com.wm.javayu.utils;import com.wm.javayu.model.Student;import com.wm.javayu.model.Teacher;import org.springframework.util.CollectionUtils;import org.springframework.util.ObjectUtil...
Class 类的实例表示正在运行的 Java 应用程序中的类和接口;
枚举是一种类,注解(指的是注解Annotation)是一种接口;
每个数组都是 Class字节码类中的一个具体 对象
基本的 Java 类型(boolean、byte、char、short、int、long、float 和 double)和关键字 void 也表示为 Class 对象;
1、 Class类 和它的实例的产生: Class的对象是已经存在的类型,
所以不能够直接new一个Class对象出来,是通过Class类中的一个方法获取到的。
例如:通过全限定路径类名
2、同一种类型不管通过什么方式得到Class的实例都是相等的;一个类型的字节码对象只有一份,在元空间。
3、Class的实例就看成是Java中我们学过的所有的数据类型在JVM中存在的一种状态(字节码对象)
String.class int.class List.class int[].class
1.概念:通过一个全限定类名,获取字节码文件
2.作用:
1. 提高开发灵活度,提高程序的扩展性
2. 框架(提高开发效率的别人封装好的代码)底层都是使用反射技术。例如:Tomcat、Spring...
3. 缺点:破坏封装性,性能低下(以后,能不用反射技术就不用)
3. 使用:(重点)
1. 获取字节码文件
1.1 Class clazz = Class.forName(全限定路径名) (最多使用)默认就是调用下面的方法
1.2 static 类<?> forName(String name, boolean initialize, ClassLoader loader)
name:全限定路径名
initialize:表示是否初始化,默认是false
loader:可以指定一个类加载器加载字节码文件
2. 全限定类名.class
3. 对象名.getClass()
Class类中方法newInstance():创建当前字节码对象(只能调用无参且是public修饰的构造方法)
2. 根据字节码文件获取构造方法、普通方法、字段等
Constructor[] constructors = clazz.getConstructors() 获取public修饰的构造方法数组
Constructor[] constructors = clazz.getDeclaredConstructors() 获取任意权限的所有造方法数组
Constructor constructor = clazz.getConstructor(Class 参数字节码)根据参数类型获取public修饰的指定的的构造方法
Constructor constructor = clazz.getDeclearConstructor(Class 参数字节码) 获取任意访问权限指定的构造方法
//通过构造方法对象去用构造方法创建对象 => 相当于new 一个对象
Object instance = constructor.newInstance(Object 实参);//可以创建任意访问权限的有参或者无参构造
Method[] methods = clazz.getMethods() 获取public修饰的构造方法数组,有父类中的方法
Method[] methods = clazz.getDeclaredMethods() 获取任意访问权限所有造方法数组,并且都是自己的方法
Method method = clazz.getMethod(String methodName,Class... 参数字节码)根据方法名和参数类型获取指定的的方法
methodName:方法名
Class:形参类型。如果方法没有形参,则Class可变参数不用写
Method method = clazz.getDeclaredMethod(String methodName,Class... 参数字节码)根据方法名和参数类型获取指定的的方法
methodName:方法名
Class:形参类型。如果方法没有形参,则Class可变参数不用写
//通过普通方法对象调用执行方法
method.invoke(Object obj,Object... args);
obj:对象。如果是对象的方法,就传入一个当前字节码创建的对象,如果是static方法,则写null
args:就是具体实参
Field[] fields = clazz.getFields() 获取public修饰的字段
Field[] fields = clazz.getDeclaredFields() 获取任意权限所有字段
Field field = clazz.getDeclaredField(String fieldName) 根据字段名获取任意访问权限的指定字段
Field field = clazz.Field(String fieldName)根据字段名获取public的指定字段
//通过当前的字段对象,给某一个字段赋值取值
field.get(Object obj);//如果是属于非static,就传入一个对象,如果是静态的,就传入null
obj:对象
field.set(Object obj, Object value);//如果是属于非static,就传入一个对象,如果是静态的,就传入null
obj:对象
value:值
String getName() 获取全限定类名(全限定,包含包名) Class类中方法
String getSimpleName() 获取类名简称 Class类中方法
Package getPackage() 获取包名 Class类中方法
T newInstance() 根据当前字节码创建对应的对象 Class类中方法
1. Class类中方法newInstance():创建当前字节码对象(只能调用无参且是public修饰的构造方法)
2. Constructor类中方法newInstance(Object 实参);//可以创建任意访问权限的有参或者无参构造
3. 私有化方法、字段、构造方法都必须破坏封装才能使用:
public void setAccessible(boolean flag) true表示破坏封装,false是不破坏
是哪个private修饰的方法、字段、构造方法需要执行,就需要用这个对象破坏哪一个的封装
//获取cn.itsource.reflect.User字节码文件
Class<?> clazz = Class.forName("cn.itsource.reflect.User");
//获取User的有String参构造
Constructor<?> constructor = clazz.getConstructor(String.class);
//调用有参String构造,创建一个User对象
Object newInstance = constructor.newInstance("某文");
//获取private修饰的方法:testPrivate
Method method2 = clazz.getDeclaredMethod("testPrivate");
method2.setAccessible(true);//破坏普通方法Method封装
//破坏封装后就可以执行了
Object invoke2 = method2.invoke(newInstance);//没有形参就不用写
System.out.println(invoke2);
4. 调用static方法、字段:
Field field = clazz.getDeclaredField("country");//获取任意访问权限静态变量country
field.set(null, "中国");//因为字段country是static修饰,所以不用对象调用,就传入null。字段country赋值:中文
Object object = field.get(null);//字段country取值
System.out.println(object);
2. 注解:
1.概念: 就是一个标签,有标签后,就具有某一些标签的特性。
本质就是跟类、接口、枚举平级的新结构
2.作用:
1. 帮助程序员校验代码
2. 可以提高开发效率和程序的扩展性
@Test
@Before
@After
3. 可以生成文档说明 api
操作步骤:
1.选中项目/代码,右键选中Export
2.输入Javadoc
3.第一个next(可以设置生成文档注释的目录),第二个next,设置字符集
如果是UTF-8编码,且有中文,请输入-encoding UTF-8 -charset UTF-8
4. 勾选一个生成完毕后,直接通过浏览器打开的选项
5. finish
3.使用:(重点)
1. 使用jdk或者别人定义好的标签
@ + 注解的名称 -- 比如@Override ->注解
2. 使用自定义的标签
1.JDK的元注解:就是专门用来声明其他注解的注解
作用:通过元注解了解其他注解的使用特点,还可以自定义注解
2.元注解:
@Target
@Target 作用
用来限制被修饰注解的使用范围,即注解可以在类的哪些成员上使用
@Target 取值使用ElementType.()
1. CONSTRUCTOR:可以在构造器上使用注解
2. FIELD:可以在字段上使用注解
3. LOCAL_VARIABLE:可以在局部变量上使用注解
4. METHOD:可以在普通方法上使用注解
5. PACKAGE:可以在包上使用注解
6. PARAMETER:可以在参数列表上使用注解
7. TYPE:可以在类、接口(包括注解类型) 或enum上使用注解
例如:@Target(ElementType.METHOD)//意味着@Override只能在普通方法上使用
public @interface Override {
@Retention
11.1.1 Java基本类型的Hibernate映射类型
11.1.2 Java时间和日期类型的Hibernate映射类型
11.1.3 Java大对象类型的Hibernate映射类型
11.1.4 JDK自带的个别Java类的Hibernate映射类型
11.1.5 使用Hibernate内置映射类型
11.2 客户化映射类型
11.2.1 用客户化映射类型取代Hibernate组件
11.2.2 用UserType映射枚举类型
11.2.3 实现CompositeUserType接口
11.2.4 运行本节范例程序
11.3 操纵Blob和Clob类型数据
11.4 小结
11.5 思考题
第12章 映射继承关系
12.1 继承关系树的每个具体类对应一个表
12.1.1 创建映射文件
12.1.2 操纵持久化对象
12.2 继承关系树的根类对应一个表
12.2.1 创建映射文件
12.2.2 操纵持久化对象
12.3 继承关系树的每个类对应一个表
12.3.1 创建映射文件
12.3.2 操纵持久化对象
12.4 选择继承关系的映射方式
12.5 映射多对一多态关联
12.6 小结
12.7 思考题
第13章 Java集合类
13.1 Set(集)
13.1.1 Set的一般用法
13.1.2 HashSet类
13.1.3 TreeSet类
13.1.4 向Set中加入持久化类的对象
13.2 List(列表)
13.3 Map(映射)
13.4 小结
13.5 思考题
第14章 映射值类型集合
14.1 映射Set(集)
14.2 映射Bag(包)
14.3 映射List(列表)
14.4 映射Map
14.5 对集合排序
14.5.1 在数据库中对集合排序
14.5.2 在内存中对集合排序
14.6 映射组件类型集合
14.7 小结
14.8 思考题
第15章 映射实体关联关系
15.1 映射一对一关联
15.1.1 按照外键映射
15.1.2 按照主键映射
15.2 映射单向多对多关联
15.3 映射双向多对多关联关系
15.3.1 关联两端使用元素
15.3.2 在inverse端使用元素
15.3.3 使用组件类集合
15.3.4 把多对多关联分解为两个一对多关联
15.4 小结
15.5 思考题
第16章 Hibernate的检索策略
16.1 Hibernate的检索策略简介
16.2 类级别的检索策略
16.2.1 立即检索
16.2.2 延迟检索
16.3 一对多和多对多关联的检索策略
16.3.1 立即检索(lazy属性为“false”)
16.3.2 延迟检索(lazy属性为默认值“true”)
16.3.3 增强延迟检索(lazy属性为“extra”)
16.3.4 批量延迟检索和批量立即检索(使用batch-size属性)
16.3.5 用带子查询的select语句整批量初始化orders集合(fetch属性为“subselect”)
16.3.6 迫切左外连接检索(fetch属性为“join”)
16.4 多对一和一对一关联的检索策略
16.4.1 迫切左外连接检索(fetch属性为“join”)
16.4.2 延迟检索(lazy属性为默认值“proxy”)
16.4.3 无代理延迟检索(lazy属性为“no-proxy”)
16.4.4 立即检索(lazy属性为“false”)
16.4.5 批量延迟检索和批量立即检索(使用batch-size属性)
16.5 控制迫切左外连接检索的深度
16.6 在应用程序中显式指定迫切左外连接检索策略
16.7 属性级别的检索策略
16.8 小结
16.9 思考题
第17章 Hibernate的检索方式(上)
17.1 Hibernate的检索方式简介
17.1.1 HQL检索方式
17.1.2 QBC检索方式
17.1.3 本地SQL检索方式
17.1.4 关于本章范例程序
17.1.5 使用别名
17.1.6 多态查询
17.1.7 对查询结果排序
17.1.8 分页查询
17.1.9 检索单个对象(uniqueResult()方法)
17.1.10 按主键逐个处理查询结果(iterate()方法)
17.1.11 可滚动的结果集
17.1.12 在HQL查询语句中绑定参数
17.1.13 设置查询附属事项
17.1.14 在映射文件中定义命名查询语句
17.1.15 在HQL查询语句中调用函数
17.2 设定查询条件
17.2.1 比较运算
17.2.2 范围运算
17.2.3 字符串模式匹配
17.2.4 逻辑运算
17.2.5 集合运算
17.3 小结
17.4 思考题
第18章 Hibernate的检索方式(下)
18.1 连接查询
18.1.1 默认情况下关联级别的运行时检索策略
18.1.2 迫切左外连接
18.1.3 左外连接
18.1.4 内连接
18.1.5 迫切内连接
18.1.6 隐式内连接
18.1.7 右外连接
18.1.8 使用SQL风格的交叉连接和隐式内连接
18.1.9 关联级别运行时的检索策略
18.2 投影查询
18.3 报表查询
18.3.1 使用聚集函数
18.3.2 分组查询
18.3.3 优化报表查询的性能
18.4 高级查询技巧
18.4.1 动态查询
18.4.2 集合过滤
18.4.3 子查询
18.4.4 本地SQL查询
18.4.5 查询结果转换器
18.5 查询性能优化
18.5.1 iterate()方法
18.5.2 查询缓存
18.6 小结
18.7 思考题
第19章 Hibernate高级配置
19.1 配置数据库连接池
19.1.1 使用默认的数据库连接池
19.1.2 使用配置文件指定的数据库连接池
19.1.3 从容器中获得数据源
19.1.4 由Java应用本身提供数据库连接
19.2 配置事务类型
19.3 把SessionFactory与JNDI绑定
19.4 配置日志
19.5 使用XML格式的配置文件
19.6 小结
19.7 思考题
第20章 声明数据库事务
20.1 数据库事务的概念
20.2 声明事务边界的方式
20.3 在mysql.exe程序中声明事务
20.4 Java应用通过JDBC API声明JDBC事务
20.5 Java应用通过Hibernate API声明JDBC事务
20.5.1 处理异常
20.5.2 Session与事务的关系
20.5.3 设定事务超时
20.6 Java应用通过Hibernate API声明JTA事务
20.7 Java应用通过JTA API声明JTA事务
20.8 小结
20.9 思考题
第21章 处理并发问题
21.1 多个事务并发运行时的并发问题
21.1.1 第一类丢失更新
21.1.2 脏读
21.1.3 虚读
21.1.4 不可重复读
21.1.5 第二类丢失更新
21.2 数据库系统的锁的基本原理
21.2.1 锁的多粒度性及自动锁升级
21.2.2 锁的类型和兼容性
21.2.3 死锁及其防止办法
21.3 数据库的事务隔离级别
21.3.1 在mysql.exe程序中设置隔离级别
21.3.2 在应用程序中设置隔离级别
21.4 在应用程序中采用悲观锁
21.4.1 利用数据库系统的独占锁来实现悲观锁
21.4.2 由应用程序实现悲观锁
21.5 利用Hibernate的版本控制来实现乐观锁
21.5.1 使用元素
21.5.2 使用元素
21.5.3 对游离对象进行版本检查
21.5.4 强制更新版本
21.6 实现乐观锁的其他方法
21.7 小结
21.8 思考题
第22章 管理Hibernate的缓存
22.1 缓存的基本原理
22.1.1 持久化层的缓存的范围
22.1.2 持久化层的缓存的并发访问策略
22.2 Hibernate的二级缓存结构
22.3 管理Hibernate的第一级缓存
22.4 管理Hibernate的第二级缓存
22.4.1 配置进程范围内的第二级缓存
22.4.2 配置集群范围内的第二级缓存
22.4.3 在应用程序中管理第二级缓存
22.4.4 Session与第二级缓存的交互模式
22.5 运行本章的范例程序
22.6 小结
22.7 思考题
第23章 管理Session和实现对话
23.1 管理Session对象的生命周期
23.1.1 Session对象的生命周期与本地线程绑定
23.1.2 Session对象的生命周期与JTA事务绑定
23.2 实现对话
23.2.1 使用游离对象
23.2.2 使用手工清理缓存模式下的Session
23.3 小结
23.4 思考题
第24章 Hibernate与Struts框架
24.1 实现业务数据
24.2 实现业务逻辑
24.3 netstore应用的订单业务
24.4 小结
第25章 Hibernate与EJB组件
25.1 创建EJB组件
25.1.1 编写Remote接口
25.1.2 编写Home接口
25.1.3 编写Enterprise Java Bean类
25.2 在业务代理类中访问EJB组件
25.3 发布J2EE应用
25.3.1 在JBoss上部署EJB组件
25.3.2 在JBoss上部署Web应用
25.3.3 在JBoss上部署J2EE应用
25.4 小结
附录A 标准SQL语言的用法
A.1 数据完整性
A.1.1 实体完整性
A.1.2 域完整性
A.1.3 参照完整性
A.2 DDL数据定义语言
A.3 DML数据操纵语言
A.4 DQL数据查询语言
A.4.1 简单查询
A.4.2 连接查询
A.4.3 子查询
A.4.4 联合查询
A.4.5 报表查询
附录B Java语言的反射机制
B.1 Java Reflection API简介
B.2 运用反射机制来持久化Java对象
附录C 用XDoclet工具生成映射文件
C.1 创建带有@hibernate标记的Java源文件
C.2 建立项目的目录结构
C.3 运行XDoclet工具
附录D 发布和运行netstore应用
D.1 运行netstore所需的软件
D.2 netstore应用的目录结构
D.3 安装SAMPLEDB数据库
D.4 安装和配置JBoss服务器
D.5 发布netstore应用
D.5.1 在工作模式1下发布netstore应用
D.5.2 在工作模式2下发布netstore应用
D.6 运行netstore应用
附录E Hibernate 3升级指南
E.1 Hibernate API 变化
E.1.1 包名
E.1.2 org.hibernate.classic包
E.1.3 Hibernate所依赖的第三方软件包
E.1.4 异常模型
E.1.5 Session接口
E.1.6 createSQLQuery()
E.1.7 Lifecycle 和 Validatable 接口
E.1.8 Interceptor接口
E.1.9 UserType和CompositeUserType接口
E.1.10 FetchMode类
E.1.11 PersistentEnum类
E.1.12 对Blob 和Clob的支持
E.1.13 Hibernate中供扩展的API的变化
E.2 元数据的变化
E.2.1 检索策略
E.2.2 对象标识符的映射
E.2.3 集合映射
E.2.4 DTD
E.3 查询语句的变化
E.4 把Hibernate 2应用升级到Hibernate 3应用
在使用三层架构开发一个网站时,希望把DataTable对象转换为List对象,于是在网上找资料,总结一个比较方便的方法来实现——使用反射。
初始化一个List对象 获取到T所有的属性,初始化一个T对象 遍历所有属性,如果DataTable中含有相应属性的值则为T对象赋值,如果没有对应的列则检查数据模型是否定义有误(属性名与列名比较时不区分大小写) 将T对象添加到List对象中
总体代码:
反射获取到的属性信息
DataTable中的列信息,对比上图,我们会发现,属性中首字母都是大写,而列名则是采用的Camel命名,首字母是小写,但是通过单步调试可以发现 dt.
参考:添加链接描述
获取List以及List中类的属性值的参考代码:
public static List getListFieldValueByObject (Object object , String targetFieldName) throws Exception {
List<Object> returnList = new ArrayList<>();
// 获取该对象的Class
Class objClass = object.getClas
除了int等基本类型外,Java的其他类型全部都是class(包括interface)
class(包括interface)的本质是数据类型(Type)。无继承关系的数据类型无法赋值Number n = new Double(123.456); // OK
String s = new Double(123.456); // compile error!
JVM持有的每
首先是需要引入的依赖项,如下:
implementation 'org.assertj:assertj-core:3.21.0'
implementation 'org.springframework:spring-aspects:5.3.13'
implementation 'com.github.ulisesbocchio:jasypt-spring-boot-starter:3.0.4