枚举、注解与反射详解

1、枚举

更多用法见文章

JDK1.5引入了新的类型——枚举。
在JDK1.5 之前,我们定义常量都是:

public static fianl.... 

很难管理。
枚举,可以把相关的常量分组到一个枚举类型里,而且枚举提供了比常量更多的方法。
用于定义有限数量的一组同类常量,例如:
错误级别:
低、中、高、急
一年的四季:
春、夏、秋、冬
商品的类型:
美妆、手机、电脑、男装、女装…
在枚举类型中定义的常量是该枚举类型的实例。

定义格式

权限修饰符 enum 枚举名称 { 
	实例1,实例2,实例3,实例4; 
}
public enum Level { 
	LOW(30), MEDIUM(15), HIGH(7), URGENT(1); 
	private int levelValue; 
	private Level(int levelValue) { 
		this.levelValue = levelValue; 
	}
	public int getLevelValue() { 
		return levelValue; 
	} 
}

这里定义的LOW、MEDIUM、HIGH、URGENT其实就是一个个Level类的实例对象,所以比如LOW(30)其实就是在调用Level类的一参构造方法。

常用定义方式(不给值,通过名称来表示级别的高低):

枚举类的主要方法


只有toString方法可以重写

compareTo方法举例:
如果定义递增的四个等级a、b、c、d。
若a.compareTo(b)则返回-1,a.compareTo( c )则返回-2,a.compareTo(d)则返回-3。
d.compareTo(a)则返回3。

实现接口的枚举类

如果采用这种方式实现接口:

则每个枚举对象(LOW、MEDIUM、HIGH)所实现的show方法都是一样,那么怎么做到不同枚举对象实现不同的方法呢?

用这种方式就可以直接实现每个枚举对象的不同方法。
注意:只能通过上述用匿名内部类实现接口的方式来为每个枚举对象设置不同的方法。直接用匿名内部类无法实现

注意:这里的LOW、MEDIUM、HIGH都是枚举对象,是可以调用构造方法和普通方法的。比如:LOW()这就调用了Level3类的无参构造方法。

注意事项

  1. 一旦定义了枚举,最好不要妄图修改里面的值,除非修改是必要的。
  2. 枚举类默认继承的是java.lang.Enum类而不是Object类
  3. 枚举类不能有子类,因为其枚举类默认被final修饰
  4. 只能有private构造方法
  5. switch中使用枚举时,直接使用常量名,不用携带类名
  6. 不能定义name属性,因为自带name属性
  7. 不要为枚举类中的属性提供set方法,不符合枚举最初设计初衷。
  8. 枚举类型有一个公共的基本的父类,是 java.lang.Enum 类型,所以不能再继承别的类型,但是可以实现接口

2、注解(可以理解为给机器看的一种注释)

Java 注解(Annotation)又称 Java 标注,是 JDK5.0 引入的一种注释机制。
Java 语言中的类、方法、变量、参数和包等都可以被标注。
和注释不同,Java 标注可以通过反射获取标注内容。在编译器生成类文件时,标注可以被嵌入到字节码中。Java 虚拟机可以保留标注内容,在运行时可以获取到标注内容。当然它也支持自定义 Java 标注。
主要用于:
编译格式检查
反射中解析
生成帮助文档
跟踪代码依赖

学习的重点

理解 Annotation 的关键,是理解 Annotation 的语法和用法.
学习步骤:

  1. 概念
  2. 怎么使用内置注解
  3. 怎么自定义注解
  4. 反射中怎么获取注解内容

内置注解

@Override : 重写 * 

定义在java.lang.Override

@Deprecated:废弃 *

定义在java.lang.Deprecated

@SafeVarargs 

Java 7 开始支持,忽略任何使用参数为泛型变量的方法或构造函数调用产生的警告。

@FunctionalInterface: 函数式接口 * 

Java 8 开始支持,标识一个匿名函数或函数式接口。

@Repeatable:标识某注解可以在同一个声明上使用多次 

Java 8 开始支持,标识某注解可以在同一个声明上使用多次。

@SuppressWarnings:抑制编译时的警告信息。* 

定义在java.lang.SuppressWarnings
三种使用方式

  1. @SuppressWarnings(“unchecked”) [^ 抑制单类型的警告]
  2. @SuppressWarnings(“unchecked”,“rawtypes”) [^ 抑制多类型的警告]
  3. @SuppressWarnings(“all”) [^ 抑制所有类型的警告]

参数列表:

关键字用途
all抑制所有警告
boxing抑制装箱、拆箱操作时候的警告
cast抑制映射相关的警告
dep-ann抑制启用注释的警告
deprecation抑制过期方法警告
fallthrough抑制确在switch中缺失breaks的警告
fifinally抑制fifinally模块没有返回的警告
hiding抑制相对于隐藏变量的局部变量的警告
incomplete-switch忽略没有完整的switch语句
nls忽略非nls格式的字符
null忽略对null的操作
rawtypes使用generics时忽略没有指定相应的类型
restriction抑制禁止使用劝阻或禁止引用的警告
serial忽略在serializable类中没有声明serialVersionUID变量
static-access抑制不正确的静态访问方式警告
synthetic-access抑制子类没有按最优方法访问内部类的警告
unchecked抑制没有进行类型检查操作的警告
unqualifified-fifield-access抑制没有权限访问的域的警告
unused抑制没被使用过的代码的警告

自定义注解

元注解(用于修饰注解的注解)

作用在其他注解(自己定义的注解)的注解
有哪些?
@Retention - 标识这个注解怎么保存,是只在代码中,还是编入class文件中,或者是在运行时可以通过反射访问。它解释说明了这个注解的的存活时间。
它的取值如下:

RetentionPolicy.SOURCE 注解只在源码阶段保留,在编译器进行编译时它将被丢弃忽视。
RetentionPolicy.CLASS 注解只被保留到编译进行的时候,它并不会被加载到 JVM 中。
RetentionPolicy.RUNTIME 注解可以保留到程序运行的时候,它会被加载进入到 JVM 中,所以在程序运行时可以获取到它们。

我们可以这样的方式来加深理解,@Retention 去给一张标签解释的时候,它指定了这张标签张贴的时间。@Retention 相当于给一张标签上面盖了一张时间戳,时间戳指明了标签张贴的时间周期。

@Documented - 标记这些注解是否包含在用户文档中 javadoc。

@Target - 标记这个注解应该是哪种 Java 成员。指定了注解运用的地方。
你可以这样理解,当一个注解被 @Target 注解时,这个注解就被限定了运用的场景。

类比到标签,原本标签是你想张贴到哪个地方就到哪个地方,但是因为 @Target 的存在,它张贴的地方就非常具体了,比如只能张贴到方法上、类上、方法参数上等等。
@Target 有下面的取值:

ElementType.ANNOTATION_TYPE 可以给一个注解进行注解
ElementType.CONSTRUCTOR 可以给构造方法进行注解
ElementType.FIELD 可以给属性进行注解
ElementType.LOCAL_VARIABLE 可以给局部变量进行注解
ElementType.METHOD 可以给方法进行注解
ElementType.PACKAGE 可以给一个包进行注解
ElementType.PARAMETER 可以给一个方法内的参数进行注解
ElementType.TYPE 可以给一个类型进行注解,比如类、接口、枚举

@Inherited - 标记这个注解是自动继承的
1.子类会继承父类使用的注解中被@Inherited修饰的注解
2.接口继承关系中,子接口不会继承父接口中的任何注解,不管父接口中使用的注解有没有被@Inherited修饰
3.类实现接口时不会继承任何接口中定义的注解注解架构
Inherited 是继承的意思,但是它并不是说注解本身可以继承,而是说如果一个超类被 @Inherited 注解过的注解进行注解的话,那么如果它的子类没有被任何注解应用的话,那么这个子类就继承了超类的注解。
说的比较抽象,代码来解释:

@Inherited
@Retention(RetentionPolicy.RUNTIME)
@interface Test {}
@Test
public class A {}
public class B extends A {}

注解 Test 被 @Inherited 修饰,之后类 A 被 Test 注解,类 B 继承 A,类 B 也拥有 Test 这个注解。

@Repeatable-Repeatable 自然是可重复的意思。@Repeatable 是 Java 1.8 才加进来的,所以算是一个新的特性。
什么样的注解会多次应用呢?通常是注解的值可以同时取多个。

举个例子,一个人他既是程序员又是产品经理,同时他还是个画家:

@interface Persons {
    Person[]  value();
}
@Repeatable(Persons.class)
@interface Person{
    String role default "";
}
@Person(role="artist")
@Person(role="coder")
@Person(role="PM")
public class SuperMan{
}

注意上面的代码,@Repeatable 注解了 Person。而 @Repeatable 后面括号中的类相当于一个容器注解。

什么是容器注解呢?就是用来存放其它注解的地方。它本身也是一个注解。
我们再看看代码中的相关容器注解:

@interface Persons {
    Person[]  value();
}

按照规定,它里面必须要有一个 value 的属性,属性类型是一个被 @Repeatable 注解过的注解数组,注意它是数组。
如果不好理解的话,可以这样理解。Persons 是一张总的标签,上面贴满了 Person 这种同类型但内容不一样的标签。把 Persons 给一个 SuperMan 贴上,相当于同时给他贴了程序员、产品经理、画家的标签。
我们可能对于 @Person(role=”PM”) 括号里面的内容感兴趣,它其实就是给 Person 这个注解的 role 属性赋值为 PM 。


(01) Annotation与RetentionPolicy与ElementType 。
每1个 Annotation对象,都会有唯一的 RetentionPolicy属性;至于 ElementType 属性,则有 1~n个。

(02) ElementType(注解的用途类型)
"每1个Annotation"都与"1~n个ElementType"关联。当Annotation与某个ElementType关联时,就意味着:Annotation有了某种用途。例如,若一个 Annotation 对象是 METHOD 类型,则该Annotation 只能用来修饰方法。

package java.lang.annotation; 
public enum ElementType { 
TYPE, /* 类、接口(包括注释类型)或枚举声明 */ 
FIELD, /* 字段声明(包括枚举常量) */ 
METHOD, /* 方法声明 */ 
PARAMETER, /* 参数声明 */ 
CONSTRUCTOR, /* 构造方法声明 */ 
LOCAL_VARIABLE, /* 局部变量声明 */ 
ANNOTATION_TYPE, /* 注释类型声明 */ 
PACKAGE /* 包声明 */ 
}

(03) RetentionPolicy(注解作用域策略)。
“每 1 个 Annotation” 都与 “1 个 RetentionPolicy” 关联。
a) 若 Annotation 的类型为 SOURCE,则意味着:Annotation 仅存在于编译器处理期间,编译器处理完之后,该 Annotation 就没用了。 例如," @Override" 标志就是一个 Annotation。当它修饰一个方法的时候,就意味着该方法覆盖父类的方法;并且在编译期间会进行语法检查!编译器处理完后,“@Override” 就没有任何作用了。

b) 若 Annotation 的类型为 CLASS,则意味着:编译器将 Annotation 存储于类对应的 .class 文件中,它是 Annotation 的默认行为。

c) 若 Annotation 的类型为 RUNTIME,则意味着:编译器将 Annotation 存储于 class 文件中,并且可由JVM读入。

package java.lang.annotation; 
public enum RetentionPolicy { 
SOURCE, /* Annotation信息仅存在于编译器处理期间,编译器处理完之后就没有该 
Annotation信息了 */ 
CLASS, /* 编译器将Annotation存储于类对应的.class文件中。默认行为 */ 
RUNTIME /* 编译器将Annotation存储于class文件中,并且可由JVM读入 */ 
}

定义格式

@interface 自定义注解名{}

注解的属性

注解的属性也叫做成员变量。注解只有成员变量,没有方法。注解的成员变量在注解的定义中以“无形参的方法”形式来声明,其方法名定义了该成员变量的名字,其返回值定义了该成员变量的类型。
举例:

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface TestAnnotation {
    int id();
    String msg();
}

上面代码定义了 TestAnnotation 这个注解中拥有 id 和 msg 两个属性。在使用的时候,我们应该给它们进行赋值。
赋值的方式是在注解的括号内以 value=”” 形式,多个属性之前用 ,隔开。如下:

@TestAnnotation(id=3,msg="hello annotation")
public class Test {
}

需要注意的是,在注解中定义属性时它的类型必须是 8 种基本数据类型外加 类、接口、注解及它们的数组。

注解中属性可以有默认值,默认值需要用 default 关键值指定。比如:

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface TestAnnotation {
    public int id() default -1;
    public String msg() default "Hi";
}

TestAnnotation 中 id 属性默认值为 -1,msg 属性默认值为 Hi。

它可以这样应用:

@TestAnnotation()
public class Test {}

因为有默认值,所以无需要再在 @TestAnnotation 后面的括号里面进行赋值了,这一步可以省略。

另外,还有一种情况。如果一个注解内仅仅只有一个名字为 value 的属性时,应用这个注解时可以直接接属性值填写到括号内。

public @interface Check {
    String value();
}

上面代码中,Check 这个注解只有 value 这个属性。所以可以这样应用。

@Check("hi")
int a;

这和下面的效果是一样的

@Check(value="hi")
int a;

最后,还需要注意的一种情况是一个注解没有任何属性。比如:

public @interface Perform {}

那么在应用这个注解的时候,括号都可以省略(如@Override注解)。如下:

@Perform
public void testMethod(){}

注解的提取

注解通过反射获取。首先可以通过 Class 对象的 isAnnotationPresent() 方法判断它是否应用了某个注解

public boolean isAnnotationPresent(Class<? extends Annotation> annotationClass) {}

然后通过 getAnnotation() 方法来获取 Annotation 对象。

 public <A extends Annotation> A getAnnotation(Class<A> annotationClass) {}

或者是 getAnnotations() 方法。

public Annotation[] getAnnotations() {}

前一种方法返回指定类型的注解,后一种方法返回注解到这个元素上的所有注解。

如果获取到的 Annotation 如果不为 null,则就可以调用它们的属性方法了。比如:

@TestAnnotation()
public class Test {
    public static void main(String[] args) {
        boolean hasAnnotation = Test.class.isAnnotationPresent(TestAnnotation.class);
        if ( hasAnnotation ) {
            TestAnnotation testAnnotation = Test.class.getAnnotation(TestAnnotation.class);
            System.out.println("id:"+testAnnotation.id());
            System.out.println("msg:"+testAnnotation.msg());
        }
    }
}

程序的运行结果是:

id:-1
msg:

这个正是 TestAnnotation 中 id 和 msg 的默认值。

上面的例子中,只是检阅出了注解在类上的注解,其实属性、方法上的注解照样是可以的。同样还是要假手于反射(具体见下面的反射部分)。

注解的作用

官方解释:

注解是一系列元数据,它提供数据用来解释程序代码,但是注解并非是所解释的代码本身的一部分。
注解对于代码的运行效果没有直接影响。

注解有许多用处,主要如下:

提供信息给编译器: 编译器可以利用注解来探测错误和警告信息
编译阶段时的处理: 软件工具可以用来利用注解信息来生成代码、Html文档或者做其它相应处理。
运行时的处理: 某些注解可以在程序运行的时候接受代码的提取
值得注意的是,注解不是代码本身的一部分。

个人总结:

注解就像标签,标签只是某些人对于其他事物的评价,但是标签不会改变事物本身。所以,注解同样无法改变代码本身,注解只
是某些工具的的工具。
当开发者使用了Annotation 修饰了类、方法、Field 等成员之后,这些Annotation不会自己生效,必须由开发者提供相应
的代码来提取并处理 Annotation 信息。
这些处理提取和处理 Annotation 的代码统称为 APTAnnotation Processing Tool)。

现在,我们可以给自己答案了,注解有什么用?给谁用?给 编译器或者 APT 用的。

引用说明: 上述部分内容引用自文章https://blog.csdn.net/qq1404510094/article/details/80577555
作者:Tanyboye

注意事项

  1. 定义的注解,自动继承了java.lang,annotation.Annotation接口
  2. 注解中的每一个方法,实际是声明的注解配置参数
    方法的名称就是 配置参数的名称
    方法的返回值类型,就是配置参数的类型。只能是:基本类型/Class/String/enum
  3. 可以通过default来声明参数的默认值
  4. 如果只有一个参数成员,一般参数名为value
  5. 注解元素必须要有值,我们定义注解元素时,经常使用空字符串、0作为默认值。
  6. 如果一个注解要在运行时被成功提取,那么 @Retention(RetentionPolicy.RUNTIME) 是必须的。

案例

@Documented 
@Target(ElementType.TYPE) 
@Retention(RetentionPolicy.RUNTIME) 
public @interface MyAnnotation1 { 
	参数类型 参数名() default 默认值; 
}

上面的作用是定义一个 Annotation,我们可以在代码中通过 “@MyAnnotation1” 来使用它。
@Documented, @Target, @Retention, @interface 都是来修饰 MyAnnotation1 的。
含义:
(01) @interface
使用@interface 定义注解时,意味着它实现了 java.lang.annotation.Annotation 接口,即该注解就是 一个Annotation。
定义 Annotation 时,@interface 是必须的。
注意:它和我们通常的 implemented 实现接口的方法不同。Annotation 接口的实现细节都由编译器完成。
通过 @interface 定义注解后,该注解不能继承其他的注解或接口。

(02) @Documented
类和方法的 Annotation 在缺省情况下是不出现在 javadoc 中的。如果使用 @Documented 修饰该
Annotation,则表示它可以出现在 javadoc 中。
定义 Annotation 时,@Documented 可有可无;若没有定义,则 Annotation 不会出现在 javadoc中。

(03) @Target(ElementType.TYPE)
前面我们说过,ElementType 是 Annotation 的类型属性。而 @Target 的作用,就是来指定Annotation 的类型属性。
@Target(ElementType.TYPE) 的意思就是指定该 Annotation 的类型是 ElementType.TYPE。这就意味着,MyAnnotation1 是来修饰"类、接口(包括注释类型)或枚举声明"的注解。

定义Annotation 时,@Target 可有可无。若有 @Target,则该 Annotation 只能用于它所指定的地方;若没有 @Target,则该 Annotation 可以用于任何地方。

(04) @Retention(RetentionPolicy.RUNTIME)
前面我们说过,RetentionPolicy 是 Annotation 的策略属性,而 @Retention 的作用,就是指定Annotation 的策略属性。
@Retention(RetentionPolicy.RUNTIME) 的意思就是指定该 Annotation 的策略是RetentionPolicy.RUNTIME。这就意味着,编译器会将该 Annotation 信息保留在 .class 文件中,并且能被虚拟机读取。
定义 Annotation 时,@Retention 可有可无。若没有 @Retention,则默认是RetentionPolicy.CLASS。
推荐设置注解作用域的策略为RUNTIME,因为它的域最大包含SOURCE和CLASS。

案例2:

注解中可以定义抽象方法(注解其实就是一个接口)。在自定义的抽象方法中只有名称为value的方可以在传值时省略“name=”,这里定义了两个方法所以都需要加上“name=”。

这样写的作用(通过反射来获取注解中传递的值):
首先这是作用于类Demo2的注解,然后通过这个类的对象来找到这个注解,然后通过注解里定义的类型(value、num)来获取给的值(“张三”、100)。

可以通过default来设置默认值。否则必须传等量的同类型的值。

当要传递一组值的时候,使用大括号:

3、反射

JAVA反射机制是在运行状态中,获取任意(可以是有的也可以是没有的)一个类的结构 , 创建对象 , 得到方法,执行方法 , 属性!;
这种在运行状态动态获取信息以及动态调用对象方法的功能被称为java语言的反射机制。

类加载器

Java类加载器(Java Classloader)是Java运行时环境(Java Runtime Environment)的一部分,负责动态加载Java类到Java虚拟机的内存空间中。
java默认有三种类加载器,BootstrapClassLoader、ExtensionClassLoader、AppClassLoader。

BootstrapClassLoader(引导启动类加载器):
嵌在JVM内核中的加载器,该加载器是用C++语言写的,主要负载加载JAVA_HOME/lib下的类库,引导启动类加载器无法被应用程序直接使用。

ExtensionClassLoader(扩展类加载器):
ExtensionClassLoader是用JAVA编写,且它的父类加载器是Bootstrap。
是由sun.misc.Launcher$ExtClassLoader实现的,主要加载JAVA_HOME/lib/ext目录中的类库。它的父加载器是BootstrapClassLoader

App ClassLoader(应用类加载器):
App ClassLoader是应用程序类加载器,负责加载应用程序classpath目录下的所有jar和class文件。它的父加载器为ExtClassLoader


类通常是按需加载,即第一次使用该类时才加载。由于有了类加载器,Java运行时系统不需要知道文件与文件系统。学习类加载器时,掌握Java的委派概念很重要。

双亲委派模型:如果一个类加载器收到了一个类加载请求,它不会自己去尝试加载这个类,而是把这个请求转交给父类加载器去完成。每一个层次的类加载器都是如此。因此所有的类加载请求都应该传递到最顶层的
启动类加载器中,只有到父类加载器反馈自己无法完成这个加载请求(在它的搜索范围没有找到这个类)时,子类加载器才会尝试自己去加载。委派的好处就是避免有些类被重复加载。

加载配置文件

给项目添加resource root目录

举例(通过类加载器加载配置文件):

默认加载的是src路径下的文件,但是当项目存在resource root目录时,就变为了加载resource root下的文件了。

所有类型的Class对象

要想了解一个类,必须先要获取到该类的字节码文件对象.
在Java中,每一个字节码文件,被加载到内存后,都存在一个对应的Class类型的对象。
Class是一种类型,叫作类的类型。其对应了某个类的字节码文件。

得到Class的几种方式

  1. 如果在编写代码时, 知道类的名称, 且类已经存在, 可以通过
    包名.类名.class 得到一个类的类对象
  2. 如果拥有类的对象, 可以通过Class 对象.getClass() 得到一个类的 类对象
  3. 如果在编写代码时, 知道类的名称 , 可以通过
    Class.forName(包名+类名): 得到一个类的类对象

举例:

第一种方式的包名可以省略,第三种方式(用的比较多)可以在类不存在时使用,但它不会创建类(只是编译时不会报错),只要在将来运行的时候创建了这个类那就不会出错(动态编程)。三种方式都可以不指定泛型。

上述的三种方式, 在调用时, 如果类在内存中不存在, 则会加载到内存! 如果类已经在内存中存在, 不会重复加载,而是重复利用!
(一个class文件在内存中不会存在两个类对象)

特殊的类对象

基本数据类型的类对象: 
	基本数据类型.class 
	包装类.type
基本数据类型包装类对象: 
	包装类.class

获取Constructor(通过反射获取构造方法)

1.通过指定的参数类型, 获取指定的单个构造方法

getConstructor(参数类型的class对象数组) 

例如:
构造方法如下: Person(String name,int age)
得到这个构造方法的代码如下:

Constructor c = p.getClass().getConstructor(String.class,int.class);

注意参数数据类型的对应

2.获取构造方法数组

getConstructors();

3.获取所有权限的单个构造方法

getDeclaredConstructor(参数类型的class对象数组)

4.获取所有权限的构造方法数组

getDeclaredConstructors();

Constructor 创建对象

常用方法:

newInstance(Object... para) 

调用这个构造方法, 把对应的对象创建出来
参数: 是一个Object类型可变参数, 传递的参数顺序必须匹配构造方法中形式参数列表的顺序!

setAccessible(boolean flag) 

如果flag为true 则表示忽略访问权限检查!(可以访问任何权限的方法包括私有的)

举例(通过无参构造方法动态创建一个类的对象):

第一行首先将对应类加载到内存中。然后找到其无参构造方法(getConstructor的参数为空即可)。最后调用该无参构造方法创建对象。

举例(通过两参构造方法动态创建一个类的对象):

举例(通过反射在外部调用一个类的私有构造方法):


注意要使用getDeclaredConstructors()方法,它可以获取所有权限的单个构造方法。

获取Method(通过反射获取类中的方法)

getMethod(String methodName , class.. clss)

根据参数列表的类型和方法名, 得到一个方法(public修饰的)

getMethods(); 

得到一个类的所有方法 (public修饰的)

getDeclaredMethod(String methodName , class.. clss) 

根据参数列表的类型和方法名, 得到一个方法(除继承以外所有的:包含私有, 共有, 保护, 默认)

getDeclaredMethods(); 

得到一个类的所有方法 (除继承以外所有的:包含私有, 共有, 保护, 默认)

举例:

getMethod方法,第一个参数:方法名称,第二个参数:该方法的参数列表对应的参数类型.class。
然后用获取类的方法的对象调用invoke方法来传参(如果有参数的话)。
这里的o的类型就是你加载的类的类型(这里是Person)。
输出:

Method 执行方法

invoke(Object o,Object... para)

调用方法 ,
参数1. 要调用方法的对象
参数2. 要传递的参数列表

getName() 

获取方法的方法名称

setAccessible(boolean flag) 

如果flag为true 则表示忽略访问权限检查!(可以访问任何权限的方法)
这里要获取私有方法的步骤和上面一样,也要设置setAccessible为true并调用getDeclaredMethod方法。

获取Field(通过反射获取类中的属性)

getDeclaredField(String filedName) 

根据属性的名称, 获取一个属性对象 (所有属性)

getDeclaredFields() 

获取所有属性

getField(String filedName) 

根据属性的名称, 获取一个属性对象 (public属性)

getFields() 

获取所有属性 (public)

常用方法:

get(Object o ); 

参数: 要获取属性的对象
获取指定对象的此属性值

set(Object o , Object value);

参数1. 要设置属性值的 对象
参数2. 要设置的值
设置指定对象的属性的值

getName() 

获取属性的名称

setAccessible(boolean flag) 

如果flag为true 则表示忽略访问权限检查 !(可以访问任何权限的属性)

举例:
流程和上面大同小异,这里是先将类加载到内存中再创建对象然后创建属性对象并获取对应属性,最后调用相应方法。

获取私有属性的步骤也是大同小异:

通过反射获取注解信息

获取类/属性/方法的全部注解对象

Annotation[] annotations01 = Class/Field/Method.getAnnotations(); 
for (Annotation annotation : annotations01) { 
	System.out.println(annotation); 
}

根据注解名称获取类/属性/方法的注解对象

注解名称 对象名 = (注解名称) c.getAnnotation(注解名称.class);

举例(模拟ORM框架,ORM框架即对象、关系、映射):
ORM框架的作用:对对象进行关系映射。将JAVA中的对象映射为数据库中的表格。

具体代码:
1、自定义注解TableAnnotation

package com.lc.demo;

import jdk.jfr.Threshold;

import java.lang.annotation.*;

/**
 * @description:表格注解
 * @author: ManolinCoder
 * @time: 2021/7/30
 */
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Threshold
public @interface TableAnnotation {
    /**
    * @Description:用于标注类对应的表格名称
    * @Param:
    * @return:
    * @Author: ManolinCoder
    * @Date:
    */
    String value();
}

该注解可作用于类
2、自定义注解ColumnAnnotation

package com.lc.demo;

import java.lang.annotation.*;

/**
 * @description:字段注解
 * @author: ManolinCoder
 * @time: 2021/7/30
 */
@Target(ElementType.FIELD) //标注该注解只能作用于类的属性
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface ColumnAnnotation {
    /**
    * @Description:描述字段名
    * @Param:
    * @return:
    * @Author: ManolinCoder
    * @Date:
    */
    String columnName();
    /**
     * @Description:描述类型
     * @Param:
     * @return:
     * @Author: ManolinCoder
     * @Date:
     */
    String type();
    /**
    * @Description:描述数据的长度
    * @Param:
    * @return:
    * @Author: ManolinCoder
    * @Date:
    */
    String length();
}

该注解只能作用于类的属性
3、Bean类(Book)

package com.lc.demo;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import java.util.Objects;

/**
 * @description:这是个典型的Bean类,名称为Book
 * @author: ManolinCoder
 * @time: 2021/7/30
 */
@TableAnnotation("test_Book") //通过注解将类和数据库中的表格进行绑定,这里的表格名称为test_Book
public class Book {
    @ColumnAnnotation(columnName = "name",type = "varchar",length = "50")
    private String name;
    @ColumnAnnotation(columnName = "info",type = "varchar",length = "1000")
    private String info;
    @ColumnAnnotation(columnName = "id",type = "int",length = "11")
    private int id;

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        Book book = (Book) o;
        return id == book.id &&
                Objects.equals(name, book.name) &&
                Objects.equals(info, book.info);
    }

    @Override
    public int hashCode() {
        return Objects.hash(name, info, id);
    }

    public Book() {
    }

    public Book(String name, String info, int id) {
        this.name = name;
        this.info = info;
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getInfo() {
        return info;
    }

    public void setInfo(String info) {
        this.info = info;
    }

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    @Override
    public String toString() {
        return "Book{" +
                "name='" + name + '\'' +
                ", info='" + info + '\'' +
                ", id=" + id +
                '}';
    }
}

在该类中给相应注解赋了值
4、测试类

package com.lc.demo;

import java.lang.annotation.Annotation;
import java.lang.reflect.Field;

/**
 * @description:
 * @author: ManolinCoder
 * @time: 2021/7/30
 */
public class Demo {
    public static void main(String[] args) throws Exception {
        //首先获取到Book类
        Class c = Class.forName("com.lc.demo.Book");
        //获取类里的所有注解
//        Annotation[] as = c.getAnnotations();
//        for (Annotation a:as) {
//
//        }
        //获取某个注解(这里是TableAnnotation)
        TableAnnotation ta = (TableAnnotation) c.getAnnotation(TableAnnotation.class);
        System.out.println("表名:"+ta.value());

        //获取所有属性
        Field[] fs = c.getDeclaredFields();
        //遍历属性信息数组并获取对应的属性注解
        for (Field f:fs) {
            ColumnAnnotation ca = f.getAnnotation(ColumnAnnotation.class);
            System.out.println(f.getName()+"属性,对应数据库中的字段:"+ca.columnName()+",数据类型:"+ca.type()+",数据长度:"+ca.length());
        }
    }
}

输出结果:

内省

基于反射 , java所提供的一套应用到JavaBean的API

一个定义在包中的类,
拥有无参构造器, 
所有属性私有, 
所有属性提供get/set方法, 
实现了序列化接口, 

这种类, 我们称其为 bean类
Java提供了一套java.beans包的api , 对于反射的操作, 进行了封装!
内省流程介绍:

最终目的是获取某个类的某个属性的get和set方法、属性名称、属性类型等等。

首先用内省类的对象调用getBeanInfo方法并传入一个类作为参数。
然后getBeanInfo方法会产生一个传入类的信息对象。再通过这个对象调用getPropertyDescriptors方法,然后会得到一个MethodDescriptor数组。这个数组的每个下标所存储的东西,就是传入类的各个属性的信息包括get和set方法、属性名称、属性类型等等。
最后通过getReadMethod和getWriterMethod两个方法可以分别获取到get和set方法。
当然,不通过自省我们也能通过反射机制获取到set和get方法,但自省的优点就在于更加便捷。

举例:

注意点:

1、使用框架时要求bean类一定要遵守其定义规则:
一个定义在包中的类,
拥有无参构造器,
所有属性私有,
所有属性提供get/set方法,
实现了序列化接口,
否则会出现,bean类对象无法赋值等问题

2、

当获取boolean类型的数据的信息时,其set方法不是setXXX而是isXXX,注意区别。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值