Java12 注解和反射

注解和反射

注解 Java.Annotation

  1. 注解入门
  2. 内置注解
  3. 自定义注解,元注解

什么是注解

Annotaion是从JDK5.0引入

Annotation的作用:
不是程序本身,可以对程序做出解释(这一点和注释comment没什么区别)
可以被其他程序(比如:编译器等)读取

Annotation的格式:
注解是以“@注释名”在代码中存在的,还可以田间一些参数值,例如:@SuppressWarnings(Value=“unchecked”)

Annotation在哪里使用:
可以附加在package,class,method,field等上面,相当于给他们添加了额外的辅助信息,可以通过反射机制编程实现对这些元数据的访问

内置注解

@Override:定义在java.lang.Override中,此注释只适用于修辞方式,表示一个方法声明打算重写超类中的另一个方法声明

@Deorecated:定义在java.lang.Deprecated中,此注释可以用于修辞方法,属性,类,表示不鼓励程序员使用这样的元素,通常是因为它很危险或者存在更好的选择

@SuppressWarnings:定义在java.lang.SuppressWarnings中,用来抑制编译时的警告信息
与前面两个注释不同,需要添加参数才能正确使用
@SuppressWarnings(“all”);
@SuppressWarnings(“unchecked”);
@SuppressWarnings(value={“unchecked”,“deprecation”});

元注解

元注解的作用就是负责注解其他注解,Java定义了4个标准的meta-annotation类型,他们被用来提供对其他annotation类型作说明

这些类型和它们所支持的类在java.lang.annotation包中可以找到
@Target,@Retention,@Documented,@Inherited

@Target:用于描述注解的使用范围(即:被描述的注解可以使用在什么地方)
@Retention:表示需要在什么级别保存该注释信息,用于描述注解的生命周期
(SOURCE<CLASS<RUNTIME)
@Document:说明该注解将包含在javadoc中

@Inherited:说明子类可以继承父类中的该注解

package TestAnatitoal;

import java.lang.annotation.*;
import java.time.temporal.ValueRange;

/**
 * 定义一个元注解
 */
public class Test1 extends Object{
    public void test(){

    }








}



//target表示我们的注解可以在那些地方使用
@Target(value={ElementType.METHOD,ElementType.TYPE})
//retention表示我们的注解在什么地方在存在
@Retention(value=RetentionPolicy.RUNTIME)
//Documented表示是否将我们的注解生成在javadoc中
@Documented
//Inherited子类可以继承父类的注解
@Inherited
@interface MyAnnotation{

}


自定义注解

使用@Interface自定义注解时,自动继承了java.lang.annotation.Annotation接口

分析:
@inerface用来声明一个注解,格式@interface注解名{定义内容}
其中的每一个方法实际上是声明了一个配置参数
方法的名称就是参数的名称
返回值类型就是参数的类型,返回值只能是基本类型(Class,String,enum)
可以通过default来声明参数的默认值
如果只有一个参数成员,一般参数名为value
注解元素必须要有值,我们定义注解元素时,经常使用空字符串,0作为默认值

反射机制Java.Reflection

静态 VS 动态语言
动态语言
是一类在运行时可以改变其结构的语言:例如新的函数、对象、甚至代码可以被引进,已有的函数可以被删除或是其他结构上的变化。通俗点说就是在运行时代码可以根据某些条件改变自身结构
主要动态语言:object-c,C#,JavaScript,PHP,Python

静态语言
与动态语言相对应,运行时结构不可变的语言就是静态语言。如Java、C、C++

Java不是动态语言,但Java可以称为“准动态语言”。即Java有一定的动态性,我们可以利用反射机制获得类似动态语言的特性。Java的动态性让编程的时候更加灵活!

Java Reflection

Reflection(反射)是Java被视为动态语言的关键,反射机制允许程序在执行期借助于Reflection API取得任何类的内部信息,并能直接操作任意对象的内部属性及方法
Class c=Class.forName(“java.lang.String”)

加载完类之后,在堆内存的方法区中就产生了一个Class类型的对象(一个类只有一个Class对象),这个对象就包含了挽着类的结构信息。我们可以通过这个对象看到类的结构。这个对象就像一面镜子,透过这个镜子看到类的结构,所以,我们形象的称之为:反射

正常方式:引入需要的“包类”名称→通过new实例化→取得实例化对象

反射方式:实例化对象→getClass()方法→得到完整的“包装”名称

Java反射机制研究及应用

Java反射机制提供的功能
在运行时判断任意一个对象所属的类
在运行时构造任意一个类的对象
在运行时判断任意与i个类所具有的成员变量和方法
在运行时获取泛型信息
在运行时调用任意一个对象的成员变量和方法
在运行时处理注解
生成动态代理

优点:
可以实现动态创建对象的编译,体现出很大的灵活性

缺点:
对性能有影响。使用反射基本上是一种解释操作,我么可以告诉JVM,我们希望做什么并且满足我们的要求。这类操作总是慢于直接执行相同的操作

相关的主要API:
java.lang.class:代表一个类
java.lang.reflect.Method:代表一个类的方法
java.lang.reflect.filed:代表类的成员变量
java.lang.reflect.Constructor:代表类的构造器

a.若已知具体的类,通过类的class属性获取,该方法最为安全可靠,程序性能最高
Class clazz =Person.class;
b.已知某个类的实例,调用该实例的getClass()方法获取Class对象
Class clazz = person.getClass();

c.已知一个类的全类名,且该类在类路径下,可通过Class类的静态方法forName()获取,可能抛出ClassNotFoundException
Class clazz =Class.forname(“demo1.Student”);

d.内置基本数据类型可以直接用类名.Type

e.还可以利用ClassLoader我们之后讲解

package Reflect;

public class Test3 {
    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("Reflect.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{
     public   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对象

class:外部类,成员(成员内部类,静态内部类),局部内部类,匿名内部类
interface:接口
[ ] :数组
enum:枚举
annotation:注解@interface
primitive type:基本数据类型
void

小结:
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法,这种动态获取、调用对象方法的功能称为java语言的反射机制。

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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值