反射
文章目录
反射概述
反射是什么
反射 (Reflection) 是 Java 的特征之一,它允许运行中的 Java 程序获取自身的信息,并且可以操作类或对象的内部属性。
Oracle 官方对反射的解释是:
Reflection enables Java code to discover information about the fields, methods and constructors of loaded classes, and to use reflected fields, methods, and constructors to operate on their underlying counterparts, within security restrictions.
The API accommodates applications that need access to either the public members of a target object (based on its runtime class) or the members declared by a given class. It also allows programs to suppress default reflective access control.
简而言之,通过反射,我们可以在运行时获得程序或程序中每一个类型的成员和成员的信息。程序中一般的对象的类型都是在编译期就确定下来的,而 Java 反射机制可以动态地创建对象并调用其属性,这样的对象的类型在编译期是未知的。所以我们可以通过反射机制直接创建对象,即使这个对象的类型在编译期是未知的。
反射的核心是 JVM 在运行时才动态加载类或调用方法/访问属性,它不需要事先(写代码的时候或编译期)知道运行对象是谁。
反射的作用
-
在运行时判断任意一个对象所属的类;
-
在运行时创建任意一个类的对象;
-
在运行时判断任意一个类所具有的成员变量和方法(通过反射甚至可以调用private方法);
-
在运行时调用任意一个对象的方法。
举例:
Class.forName("com.mysql.jdbc.Driver");注册驱动
反射的使用
获取Class
Java的对象模型中:
所有的类都是Class类的实例,Object是类,那么Object也是Class类的一个实例。
所有的类都最终继承自Object类,Class是类,那么Class也继承自Object。
首先创建Person类:
package entity;
public class Person {
private String name;
public int age;
public Person() {
}
public Person(String name) {
this.name = name;
}
public Person(int age) {
this.age = age;
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
private void playGame(){
System.out.println("playing……");
}
}
获取Class的方法有三种:
-
直接获取某个类的
class
Class<Person> personClass = Person.class;
-
调用某个对象的
getClass()
方法Person person = new Person(); Class<? extends Person> personClass = person.getClass();
-
调用
Class
类的forName()
静态方法Class<?> personClass = Class.forName("entity.Person");
生成实例对象
-
调用
Class
对象的newInstance()
方法Person person = personClass.newInstance();
-
通过获取类的构造方法进行实例化
-
无参构造
Constructor<Person> constructor = personClass.getConstructor(); Person person = constructor.newInstance();
-
有参构造
Constructor<Person> constructor = personClass.getConstructor(String.class, int.class); Person person = constructor.newInstance("tom", 18);
-
访问属性
-
访问公有属性
// 获取age属性 Field ageField = personClass.getField("age"); // 获取person对象的age属性值 Object age = ageField.get(person); System.out.println(age); //将person对象的age属性值设置为20 ageField.set(person,20); System.out.println(person.age);
-
访问私有属性
// 获取私有的name属性 Field nameField = personClass.getDeclaredField("name"); // 将name属性的可访问性设置为true nameField.setAccessible(true); // 获取person对象的name属性值 Object name = nameField.get(person); System.out.println(name); // 将person对象的name属性值设置为jerry nameField.set(person,"jerry"); System.out.println(person.getName());
调用方法
-
访问公有方法
// 获取getName()方法 Method getName = personClass.getMethod("getName");//getMethod(方法名,参数列表) // 通过person对象调用getName()方法,并获取执行结果 Object name = getName.invoke(person); System.out.println(name); // 获取setName()方法 Method setName = personClass.getMethod("setName", String.class); // 通过person对象调用setName()方法,并传递参数 setName.invoke(person,"trump"); System.out.println(person.getName());
-
访问私有方法
// 获取私有的playGame()方法 Method playGame = personClass.getDeclaredMethod("playGame"); // 将playGame()方法的可访问性设置为true playGame.setAccessible(true); // 通过person对象调用playGame()方法 playGame.invoke(person);
反射的应用
反射封装JDBC框架
传统写法:
/**
* 传统jdbc 写法
*
* @param args
* @throws Exception
*/
public static void main(String[] args) throws Exception {
//注册驱动
Class.forName("com.mysql.jdbc.Driver");// 这句话可以省略 驱动自动加载
Connection connection =
DriverManager.getConnection("jdbc:mysql:///db01", "root", "root1234");
// 如果换一张表 student代码就需要改变,下面的赋值 都需要改变 代码写死了 需要灵活
//虽然我不知道 未来是那个实体类,可以利用反射 在运行时拿到book表对应实体类的信息
PreparedStatement ps = connection.prepareStatement("insert into student(id,name) values(?,?)");
ps.setInt(1, 1);
ps.setString(2, "张三");
int i = ps.executeUpdate();
System.out.println(i > 0 ? "成功" : "失败");
connection.close();
//把可变的东西 改为灵活的
}
//问题:每次需要写sql语句 而且sql语句的表名 每次都要换,参数数量不确定。封装一个insert方法
封装AddOne方法
编写dao层代码
public abstract class BaseDao<T> {
public int addOne1(T t) {
Class<?> aClass = t.getClass();
String tableName = tClass.getSimpleName().toLowerCase(); //student 数据库表名就拿到了
//"insert into student(id,name) values(?,?)" //
//2.字符串拼接
StringBuilder sql = new StringBuilder("insert into ");
sql.append(tableName);
sql.append("(");
//3.获取属性
Field[] fields = aClass.getDeclaredFields();
for (Field field : fields) {
String fieldName = field.getName();
sql.append(fieldName);
sql.append(",");//最后多了一个逗号
}
sql.deleteCharAt(sql.length() - 1);//删除最后一位字符
sql.append(") values (");
for (Field field : fields) {
sql.append("?,");//最后多了一个逗号
}
sql.deleteCharAt(sql.length() - 1);//删除最后一位字符
sql.append(")");
System.out.println(sql);//insert into student(id,name) values (?,?)
//补全 ? ?
Connection connection = null;
try {
connection = DriverManager.getConnection("jdbc:mysql:///db01", "root", "root1234");
PreparedStatement ps = connection.prepareStatement(sql.toString());
//补全几个? ? 是位置的 需要循环
for (int i = 0; i < fields.length; i++) {
Field field = fields[i];
field.setAccessible(true);//允许访问私有属性
Object value = field.get(t);
ps.setObject(i + 1, value);
}
return ps.executeUpdate();//执行sql
} catch (Exception e) {
e.printStackTrace();
}finally {
if (connection != null) {
try {
connection.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
}
return 0;
}
}
/**
* 测试类
*/
@Test
public void test2() {
StudentDao studentDao = new StudentDao();
Student student = new Student();
student.setId(null);//自动增长
student.setName("王武");
studentDao.addOne(student);
//防止new BaseDao() 设置成抽象类
//BaseDao<Student> dao = new BaseDao<Student>();
}
封装find方法
封装updateById方法
封装deleteById方法
Guava
<dependency>
<groupId>com.google.guava</groupId>
<artifactId>guava</artifactId>
<version>31.1-jre</version>
</dependency>
//驼峰转下划线
String tableName = "PersonDog";
String to = CaseFormat.UPPER_CAMEL.to(CaseFormat.LOWER_UNDERSCORE, tableName);
System.out.println(to);//person_dog
//下划线转驼峰
String tableName = "person_dog";
String to = CaseFormat.LOWER_UNDERSCORE.to(CaseFormat.UPPER_CAMEL, tableName);
System.out.println(to);//PersonDog
自定义注解
package com.codingfuture.annotation;
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Id {
String value = "";
}
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface TableName {
String value();
}
-
Target
- FIELD:表示"被标注的注解"只能出现在属性上。
- TYPE:表示"被标注的注解"只能出现在类上。
- METHOD:表示"被标注的注解"只能出现在方法上。
- PARAMETER:表示"被标注的注解"只能出现在参数上。
- CONSTRUCTOR:表示"被标注的注解"只能出现在构造上。
-
Retention
- 这个Retention注解用来标注"被标注的注解"最终保存在哪里。
@Retention(RetentionPolicy.SOURCE):表示该注解只能被保留在java源文件中。
@Retention(RetentionPolicy.CLASS):表示该注解被保存在class文件中。
@Retention(RetentionPolicy.RUNTIME):表示该注解被保存在class文件中,并且可以被反射机制所读取。
- 这个Retention注解用来标注"被标注的注解"最终保存在哪里。
-
获取指定类型是否加参数
-
field.isAnnotationPresent(Id.class)
-
String tableName = this.aClass.getSimpleName(); if (aClass.isAnnotationPresent(TableName.class)) { TableName declaredAnnotation = aClass.getDeclaredAnnotation(TableName.class); tableName = declaredAnnotation.value(); }
-