注解与反射
笔记整理于@bilibili遇见狂神说
注解(Annotation)
什么是注解
可被程序读取的注释
注解的格式
@Override
@Deprecated
内置注解
//在java.lang.Override中定义,表示重写
@Override
//在java.lang.Deprecated中定义,表示已过时
@Deprecated
//镇压警告
@SuppressWarnings
元注解
解释其他注解的注解
//描述注解的使用范围
@Target
//表示需要在什么级别保存该注释信息
//SOURCE<CLASS<RUNTIME
//源码<编译<运行
@Retention
//说明该注解将被包含在javadoc中
@Document
//说明子类可以继承父类中的该注解
@Inherited
自定义注解
使用@interface自定义注解,自动继承java.lang.annotation.Annotation接口
@Target({ElementType.TYPE,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@interface MyAnnotation{
String name() default "";
String[] schools() default {"安工大","铜陵一中"}
}
String name()不是方法,是参数类型+参数名
如果只有一个值可以使用value进行命名
注解默认值的名字是value,可以直接写值
反射(Reflection)
什么是反射
正常情况:引入需要的包类名称->通过new实例化->取得实例化对象
反射:实例化对象->getClass()方法->得到完整的包类名称
允许程序在执行迄今取得任何类的内部信息
并能直接操作任意对象内部的属性和方法
Class类
Class本身也是一个类
Class只能由系统建立对象
一个加载的类在JVM中只会有一个Class实例
一个Class对象对应的是一个加载到JVM中的一个class文件
每个类的实例都会记得自己是由哪个Class实例所生成
通过Class可以完整地得到一个类中的所有被加载的结构
Class类是Reflection的根源,针对任何想动态加载运行的类,唯有先获得相应的Class对象
获取Class类的方法
已知类的全类名
Class clazz = Class.forName("demo01.Student");
已知具体的类,通过类的class属性获取,安全,可靠
Class clazz = Person.class;
已知某个类的实例
Class clazz = person.getClass();
内置基本数据类型可以直接用类名
ClassLoader..
总结:
package org.example;
//测试class类的创建方式有哪些
public class Test {
public static void main(String[] args) throws ClassNotFoundException {
Person person = new Student();
System.out.println("这个人是"+person.name);
//方式一:通过对象获得
Class<?> c1 = person.getClass();
System.out.println(c1.hashCode());
//方式二:forName获得
Class<?> c2 = Class.forName("org.example.Student");
System.out.println(c2.hashCode());
//方式三:通过类名.class获得
Class<?> c3 = Student.class;
System.out.println(c3.hashCode());
//方式四:TYPE属性
Class<?> c4 = Integer.TYPE;
System.out.println(c4);
//获取父类类型
Class<?> c5 = c1.getSuperclass();
System.out.println(c5);
}
}
//实体类
class Person{
String name;
public Person() {
}
public Person(String name) {
this.name = name;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
'}';
}
}
class Student extends Person{
public Student(){
this.name= "学生";
}
}
class Teacher extends Person{
public Teacher(){
this.name="老师";
}
}
哪些类可以有class对象
package org.example;
import java.lang.annotation.ElementType;
//所有类型的class
public class Test04 {
public static void main(String[] args) {
Class<?> c1 = Object.class;//类
Class<?> c2 = Comparable.class;//接口
Class<?> c3 = String[].class;//一维数组
Class<?> c4 = int[][].class;//二维数组
Class<?> c5 = Override.class;//注解
Class<?> c6 = ElementType.class;//枚举
Class<?> c7 = Integer.class;//基本数据类型
Class<?> c8 = void.class;//void
Class<?> c9 = Class.class;//class
System.out.println(c1);
System.out.println(c2);
System.out.println(c3);
System.out.println(c4);
System.out.println(c5);
System.out.println(c6);
System.out.println(c7);
System.out.println(c8);
System.out.println(c9);
}
}
获得类的信息
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
//获得类的信息
public class Test08 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, NoSuchMethodException {
Class c1 = Class.forName("org.example.User");
//获得类的名字
System.out.println(c1.getName());
System.out.println(c1.getSimpleName());
//获得类的属性
c1.getFields();//只能获得public属性
Field[] fields = c1.getDeclaredFields();
for (Field field : fields) {
System.out.println(field);
}
//获得指定属性值
Field name = c1.getDeclaredField("name");
System.out.println(name);
//获得类的方法
Method[] methods = c1.getMethods();//获得本类以及父类的全部public方法
for (Method method : methods) {
System.out.println(method);
}
methods = c1.getDeclaredMethods();//获得本类的全部方法,包括私有
for (Method method : methods) {
System.out.println(method);
}
//获得指定方法
Method getName = c1.getMethod("getName", null);
Method setName = c1.getMethod("setName", String.class);
System.out.println(getName);
System.out.println(setName);
//获得指定的构造器
Constructor[] constructors = c1.getConstructors();//public
for (Constructor constructor : constructors) {
System.out.println(constructor);
}
constructors = c1.getDeclaredConstructors();//全部
for (Constructor constructor : constructors) {
System.out.println(constructor);
}
//获得指定的构造器
Constructor<?> declaredConstructor = c1.getDeclaredConstructor(String.class,int.class,int.class);
System.out.println(declaredConstructor);
}
}
反射操作对象
package org.example;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
//动态创建对象,通过反射
public class Test09 {
public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {
Class<?> c1 = Class.forName("org.example.User");
//构建对象
User user = (User)c1.newInstance();//调用了无参构造器
System.out.println(user);
//通过构造器创建对象
Constructor<?> declaredConstructor = c1.getDeclaredConstructor(String.class, int.class, int.class);
User user1 = (User) declaredConstructor.newInstance("wfj",001,18);
System.out.println(user1);
//通过反射调用普通方法
User user2 = (User)c1.newInstance();
Method setName = c1.getDeclaredMethod("setName", String.class);
setName.invoke(user2,"wfj");
System.out.println(user2.getName());
//通过反射操作属性
User user3 = (User) c1.newInstance();
Field name = c1.getDeclaredField("name");
//安全检测关闭,使用私有属性
name.setAccessible(true);
name.set(user3,"wfj");
System.out.println(user3.getName());
}
}
setAccessible()方法性能比较高
具体:普通方法>关闭权限检测的反射>开启权限检测的反射方法
反射操作注解
ORM对象关系映射
类和表结构对应
属性和字段对应
对象和记录对应
利用注解和反射完成类和表结构的映射关系
package org.example;
import java.lang.annotation.*;
public class Test12 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException {
Class<?> c1 = Class.forName("org.example.Student2");
//通过反射获取注解
Annotation[] annotations = c1.getAnnotations();
for (Annotation annotation : annotations) {
System.out.println(annotation);
}
//获取注解value的值
Table annotation = (Table) c1.getAnnotation(Table.class);
String value = annotation.value();
System.out.println(value);
//获得类指定注解
java.lang.reflect.Field f = c1.getDeclaredField("name");
Field annotation1 = f.getAnnotation(Field.class);
System.out.println(annotation1.columnName());
System.out.println(annotation1.length());
System.out.println(annotation1.type());
}
}
@Table("db_student")
class Student2{
@Field(columnName = "db_id",type = "int",length = 10)
private int id;
@Field(columnName = "db_age",type = "int",length = 10)
private int age;
@Field(columnName = "db_name",type = "varchar",length = 3)
private String name;
public Student2(){
}
public Student2(int id,int age,String name){
this.id = id;
this.age = age;
this.name = name;
}
public void setName(String name) {
this.name = name;
}
public void setId(int id) {
this.id = id;
}
public void setAge(int age) {
this.age = age;
}
public String getName() {
return name;
}
public int getId() {
return id;
}
public int getAge() {
return age;
}
@Override
public String toString() {
return "Student2{" +
"id=" + id +
", age=" + age +
", name='" + name + '\'' +
'}';
}
}
//类的注解
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@interface Table{
String value();
}
//属性的注解
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@interface Field{
String columnName();
String type();
int length();
}
标签:反射,name,System,class,println,注解,Class,out
From: https://www.cnblogs.com/akaxedx/p/17779091.html