注解的概念及自定义Java注解并实现注解给字段赋值

一,注解基本概念

从JDK1.5开始,Java提供了对元数据(MetaData)的支持,也就是注解(Annotation)

注解,其实就是代码中的特殊标记,这些特殊标记可以在类加载,编译,运行时被读取,并执行相应的处理

使用注解最大的作用就是在不改变程序原逻辑的情况下添加补充信息

注解(Annotation)可以像修饰符一样来使用.可用于修饰包,类,构造器,方法.成员变量,参数,局部变量的声明,这些信息被保存在Annotation的"name=value"对中

注解(Annotation)能被用来为程序元素(类,方法,成员变量)设置元数据

二,常见的注解(Annotation)

1.@Override

@Override作用是基于继承,并且该注释只能用于方法,其作用是限定重写父类的方法

2.@Dprecated

@Dprecated作用是表示某个程序元素(类,方法等)已经过时或者即将被淘汰

3.@SuppressWarnings

@SuppressWarnings作用是抑制编译器警告.用于有些没有问题的程序出现黄色感叹号

常见注解的一些详细解释可以参考 ( 这篇博客)

三,注解的自定义

创建基本注解

注解的自定义在我看来也像是一种接口的定义.
在我们创建接口的时候 会用到interface关键字.

而恰好注解的定义跟接口的定义极度的类似注解的定义使用@interface关键字
其实特别的简单
我们只需要创建一个类,然后将其关键字改为@interface即可.如下
在这里插入图片描述

定义完成之后,我们就可以在其他的类中引用该注解了.
在这里插入图片描述

但是此时的注解还没有功能那我们就要让它拥有功能.

给注解添加属性

在这里插入图片描述

如图所示格式,这就是我们添加进去的属性.格式如下
数据类型 属性名() default "默认值"

这个属性在我们调用注解的时候给我们呈现的位置如下图所示
在这里插入图片描述
在这里插入图片描述

利用元注解详细定义注解

元注解的含义及应用

元注解是用来注解自定义注解的原始注解.主要包含四个 @Target @Documented @Retention @Inherited

@Target

@Target是用来标识注解的使用范围的,可以使用枚举类ElementType来赋值.如下图所示
在这里插入图片描述
枚举类ElementType的参数及作用如下图所示
在这里插入图片描述

@Retention

Retention:限定注解写在什么级别的位置上,也可以理解为修饰注解的生命周期(保存时间)的长度,其方法是用枚举类型的RetentionPolicy中的参数.如下图所示.
在这里插入图片描述
枚举类型RetentionPolicy中的参数及作用如下图所示
在这里插入图片描述

@Documented

@Documented的作用是将自定义的注解存放到javadoc中,代表本注解将会被javadoc工具提取成文档.在doc文档中的内容会因为此注解的内容不同而不同.相当于@See @Param
这个注解不需要其它的参数.如下图所示
在这里插入图片描述

@Inherited

@Inherited注解用于继承,被本注解修饰的父类及其子类可以继承父类中的注解
此注解不需要其它的参数,如下图所示.
在这里插入图片描述
.

自定义注解功能实现应用:利用自定义注解给实体类属性赋值

我们将自定义注解如下图所示配置好之后
在这里插入图片描述

就可以来利用注解实现功能了.
这里我们用给实体类的属性赋值来完成我们对自定义注解的使用.

1.创建一个实体类

首先我们创建一个实体类,我起名叫Entry 加入get/set方法.如下图
在这里插入图片描述

2.加入自定义注解并传入属性值

如下图
在这里插入图片描述

3.创建注解实现工具类

创建一个注解工具类,加入如下代码

public class LwinnerGImpl {
    public static Entry implInt(Entry entry) {
        Field[] declaredFields = entry.getClass().getDeclaredFields();
        for (Field declaredField : declaredFields) {
            if (declaredField.isAnnotationPresent(LwinnerG.class)){
                LwinnerG lwinnerG=(LwinnerG) declaredField.getAnnotation(LwinnerG.class);
                String value = lwinnerG.value();
                try {
                    declaredField.set(entry,value);
                } catch (IllegalAccessException e) {
                    e.printStackTrace();
                }
            }
        }
        return entry;

    }
}

代码分析如下图所示:
在这里插入图片描述

4.创建测试类,调用注解实现工具类,测试赋值结果.

在这里插入图片描述
输出结果如下:
在这里插入图片描述

证明我们的注解成功的给字段进行了赋值.

文章到此结束,如果文章中有错误的地方,还请不吝赐教!

本文概念内容参考博客:https://blog.csdn.net/HJS1453100406/article/details/107641823

  • 6
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
1. 反射: 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.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.元注解: 1. @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 { } 2. @Retention
Java中,我们可以通过自定义注解的方式为某一字段赋值。首先,我们需要定义一个注解,使用@interface关键字声明,并可以在注解内部定义一些属性。接下来,在需要赋值字段上,通过在字段前加上该注解的方式来使用自定义注解。 首先,我们创建一个自定义注解,例如@MyAnnotation: ```java import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; @Retention(RetentionPolicy.RUNTIME) // 指定注解的生命周期为运行时 @Target(ElementType.FIELD) // 指定注解的作用目标为字段 public @interface MyAnnotation { String value(); // 定义一个属性value,用于赋值 } ``` 然后,我们可以在需要赋值字段上使用@MyAnnotation注解,并通过value属性给字段赋值: ```java public class MyClass { @MyAnnotation("Hello") // 通过@MyAnnotation注解给field赋值为"Hello" private String field; public String getField() { return field; } public void setField(String field) { this.field = field; } } ``` 最后,我们可以使用反射获取字段注解信息,并获取注解中的实现字段赋值的功能: ```java import java.lang.reflect.Field; public class Main { public static void main(String[] args) throws Exception { MyClass obj = new MyClass(); Field field = obj.getClass().getDeclaredField("field"); // 获取字段 MyAnnotation annotation = field.getAnnotation(MyAnnotation.class); // 获取字段上的注解 if (annotation != null) { String value = annotation.value(); // 获取注解 field.setAccessible(true); // 设置字段可访问 field.set(obj, value); // 给字段赋值 } System.out.println(obj.getField()); // 输出 "Hello" } } ``` 通过以上方法,我们可以通过自定义注解的方式为某一字段赋值实现更加灵活和可扩展的代码编写方式。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值