学习注解s

本文介绍了Java注解的基本概念,包括注解的作用、分类、属性及其使用方式。重点讲解了自定义注解的创建与应用,元注解的含义,以及如何通过注解进行配置。文中提供了一个案例,展示如何利用自定义注解测试类的方法,查找并记录潜在的bug。此外,还提及了复合注解的概念,并简要讨论了枚举在注解中的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

注解

注释:给人看,用于提醒程序员
注解:给程序看的,它拥有一定的功能
注解是JDK1.5之后才有
注解的格式:@关键词
注解的位置:类上、方法上、属性上

注解的分类

1、编译检查

如果写法错误,则编译报错

1 @Override : 表明该方法重写的父类的方法
2 @Deprecated : 标记过时
3 @SuppressWarnings("serial") :  抑制警告

2、文档

通过注解生成文档
/**
 * 
 * @author 老王 
 * @version 1.0
 * 
 *
 */
public  class Father {
	/**
	 * 
	 * @param a  加数1
	 * @param b  加数2
	 * @return   两数之和
	 */
	public int add(int a,int b){
		return a+b;
	};
}
通过CMD命令生成文档API
	javadoc xx.java
	javadoc *.java
步骤:
	1 创建文件夹,把需要生成文档的java文件放进该文件夹中
	2 通过cmd切换到当前文件夹,最后通过javadoc命令生成文档

3、自定义

//语法:新建自定义注解
public @interface 注解名{

	方法列表;
}
注解的底层是一个接口,它继承了Annotation接口
通过反编译证明:
1  javac xx.java
2  javap xx.class
public interface MyAnno extends java.lang.annotation.Annotation {
}
注解的属性
1 这里的属性实际上指的是接口中的方法的声明
2 方法必须有返回值
3 返回值类型也是有规定的:
	3.1  8大基本数据类型
	3.2  String
	3.3  枚举
	3.4  注解
	3.5  返回以上类型的数组形式
使用自定义注解

语法:在类上、方法上、属性上都能使用

如果该注解没有任何属性:@注解名
如果该注解有任何属性: @注解名(属性1=值1,属性2=值2.....)
public @interface  MyAnno {
	public  int  x1(); // 八大基本类型
	public  String  x2();//String类型
	public  Stu  x3();//枚举类型
	public  MyAnno2  x4();//注解类型
	public  int[]  x5();//以上类型的数组类型
}
//使用:
@MyAnno(x1=3,x2="abc",x3=Stu.jack,x4=@MyAnno2,x5={1,2,3})
public class User {}

我们可以把定义在注解中的属性当作形参
而使用注解时传递的值当作实参

关于属性的特殊案例
1 如果数组中的元素只有一个,则可以省去{}
2 我们可以给属性设置默认值,设置默认值后,在使用注解是可以不用管该属性
3 如果属性名是value,并且该注解只有这一个属性或者该注解的其他属性都有默认值,则在使用注解时可以省去value
//定义
public @interface  MyAnno {
	public  String[]  value();
}
//使用:
@MyAnno(value={"abc"})
public class User {}
//简写:
@MyAnno("abc")
public class User {}
元注解
用于约束注解的注解
位置:放在注解的上方
元注解是固定的:
	1、@Target:约束注解的使用位置(类、方法、属性)
@Target({ElementType.TYPE,ElementType.FIELD,ElementType.METHOD})
	public @interface  MyAnno {
如果不写,默认是三个都能够放
	2、@Retention  :注解被保留的阶段(字节码、类、对象)
推荐被保留到对象阶段
@Retention(RetentionPolicy.RUNTIME)
		public @interface  MyAnno {
	//默认是RetentionPolicy.class
3、@Documented :该注解能够被抽取到api文档中去 (了解)
4、@Inherited  :该注解能够被子类继承 (了解)
		@Inherited
		public @interface  MyAnno {
		@MyAnno
		public class Father{}
		public class Son extends Father{}
	//则son也拥有该注解
注解版配置
@MyDog(className="com.woniuxy.demo2.Dog",methodName="eat")
public class Test {		
		//获得该类的反射对象
		Class clazz= Test.class;
		//获得该类上的MyDog注解对象
		MyDog myDog =(MyDog)clazz.getDeclaredAnnotation(MyDog.class);
//		底层
//		
//		public class MyDogImpl implements MyDog{
//			@Override
//			public String className() {
//				return "com.woniuxy.demo2.Dog";
//			}
//			@Override
//			public String methodName() {
//				return "eat";
//			}
//		}
		//MyDog myDog=new MyDogImpl();
		//获取对象中的属性值
		String className = myDog.className();
		String methodName =myDog.methodName();
		//获取类的反射对象
		Class clazz2=Class.forName(className);
		//获取clazz2类的方法
		Method m= clazz2.getDeclaredMethod(methodName);
		//执行方法
		m.invoke(clazz2.newInstance());
	}
}
案例:通过自定义注解来测试某个类的N个方法是否有bug,把bug输出到文本中

被测试类:

package com.woniuxy.demo3;

import java.util.Scanner;

public class MyMath {
	
	public void haha(){
		System.out.println("haha");
	}
	
	@MyCheck
	public void add(){
		int a=1,b=2;
		System.out.println(a+b);
	}
	@MyCheck
	public void jian(){
		int a=1,b=2;
		System.out.println(a-b);
	}
	@MyCheck
	public void chu(){
		int a=1,b=0;
		System.out.println(a/b);
	}
	@MyCheck
	public void xx(){
		Scanner sca=null;
		String next = sca.next();
		System.out.println(next);
	}
	@MyCheck
	public void xx1(){
		int [] arr={1,2,3};
		arr[3]=4;
		System.out.println(arr);
	}
}

注解类

package com.woniuxy.demo3;

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

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyCheck {
	
}

解析注解的主函数

package com.woniuxy.demo3;

import java.io.File;
import java.io.FileWriter;
import java.lang.reflect.Method;

public class Test {
	public static void main(String[] args)throws Exception {
		//获得MyMath的反射对象
		MyMath math=new MyMath();
		Class clazz = math.getClass();
		//获得所有方法
		Method[] methods = clazz.getDeclaredMethods();
		//遍历
		FileWriter fw=new FileWriter(new File("log.txt"));
		for(Method m:methods){
			//判断当前方法是否拥有该注解		if(m.isAnnotationPresent(MyCheck.class)){
				try {
					m.invoke(math);
				} catch (Exception e) {
					String msg="方法名:"+m.getName()+":异常信息:"+e.getCause().getMessage()+"\n";
					fw.write(msg);
				}
			}
		}
		//关闭流
		fw.close();
	}
}

在以后,我们不会用到自定义注解,而是使用框架技术提供好的注解
注解 搭配 解析注解的程序 才能有效果

复合注解

注解上还能够放注解(不是元注解)

public @interface MyAnno1 {
	public int a();
}

@MyAnno1(a=1)
public @interface MyAnno2 {
	public String b();
	
}

MyAnno2就是复合注解
或者
MyAnno2 继承了 @MyAnno1

补充点

枚举(enum)
概念:一一列举,把所有的情况都列举出来,最终取值只能从中获取(数据已经固定了)

因此,枚举一般用于固定的数据(国家、日期、季节、省份,线路)

1、列表中的每一个字段用逗号分隔,最终用;代表结束

2、枚举就是java的语法糖(结构简单,语法很人性化)

3、枚举类默认是继承了Enum的类,并且该类默认是被final修饰

4、枚举中列举的字段默认前面是隐藏了public static final 
季节[] values = 季节.values();
		for(季节 season:values){
			System.out.println(season);
		}
5、枚举类都提供了一个values方法,能够获取列表
	1、使用字符串
public enum 季节 {,,,}
使用:
遍历字符串
		季节[] values = 季节.values();
		for(季节 season:values){
			System.out.println(season);
		}
2、使用对象:
注意:列表必须放在第一个位置,并且以分号结尾
public enum 季节 {("春天","春暖花开"),("夏天","很热"),("秋天","秋高气爽"),("东天","很冷");
	String 名字;
	String 描述;
	private 季节(String 名字,String 描述){
		this.名字=名字;
		this.描述=描述;
	}
}
遍历对象
季节[] values = 季节.values();
		for(季节 season:values){
			System.out.println(season.名字+":"+season.描述);
		}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值