Java--注解
一.注解的基本
1.注解的通俗理解
注解(Annotation) 为我们在代码中添加信息提供了一种形式化的方法,是我们可以在稍后 某个时刻方便地使用这些数据(通过解析注解来使用这些数据)。
2.注解的作用
- 生成文档
- 跟踪代码依赖性,实现替代配置文件功能,减少配置。如Spring中的一些注解
- 在编译时进行格式检查,如@Override等
- 每当你创建描述符性质的类或者接口时,一旦其中包含重复性的工作,就可以考虑使用注解来简化与自动化该过程。
3.注解的状态
源码注解:只在源码中存在的注解。
编译时注解:注解在源码和.class文件中都存在。
运行时注解:运行阶段还起作用,甚至会影响运行逻辑的注解,如Spring, Mybatic中常用的注解。
4.注解的分类
a.jdk内置注解:
- @Override:表示当前的方法定义将覆盖超类中的方法,如果出现错误,编译器就会报错。
- @Deprecated:如果使用此注解,编译器会出现警告信息。
- @SuppressWarnings:忽略编译器的警告信息。
如下Override的例子,其他方法自测,child继承Person,并且覆盖了Person中的方法:
public class child extends Person { @Override//覆盖 public int age() { // TODO Auto-generated method stub return 0; } @Override public String name() { // TODO Auto-generated method stub return null; } @Override public void sing() { // TODO Auto-generated method stub } }
b.元注解
注解的注解,常见的有4个,在自定义注解中会用到,分别是
@Target,@Retention,@Documented,@Inherited
c.自定义注解
自己定义的注解。自定义注解由元注解和自己定义的注解(annotation)构成,格式,要求及实例如下:
//自定义注解 //下面4行是元注解 @Target({ElementType.METHOD,ElementType.TYPE}) @Retention(RetentionPolicy.SOURCE)//元注解的作用域,生命周期 @Inherited //允许子继承 @Documented//生成javadoc文档时会包含次注解 //使用@interface关键字定义注解 public @interface Description{ /* * 成员类型是受限的,所有基本类型、String、Class、enum、Annotation、以上类型的数组形式。 * 如果注解只有一个成员时,则成员名必须取名为value(), * 在使用是可以忽略成员名和赋值名 * 注解类可以没有成员,没有成员的注解称为标识注解 */ String value();//成员以无参无异常方式声明 //String author(); //int age() default 18;//可以给成员指定默认的值 }
5.注解解析
注解解析其实是用了Java中的反射机制,不明白的可以看看我的上一篇博客,Java--反射的逐步理解
核心代码如下:
//1.使用类加载器加载类 try { Class class1 =Class.forName("cn.ann.test.child");
//2.找到类上面的注解 boolean isExist=class1.isAnnotationPresent(Description.class);//判断是否存在这个注解 Method[] ms = class1.getMethods(); for (Method method : ms) { boolean exist=method.isAnnotationPresent(Description.class); if(exist){ Description b=(Description)method.getAnnotation(Description.class); System.out.println(b.value()); } } if(isExist){ //3.得拿到注解实例 Description a=(Description)class1.getAnnotation(Description.class); System.out.println(a.value()); } //4.找到方法上的注解 for (Method method : ms) { Annotation[] s =method.getAnnotations(); for (Annotation annotation : s) { if(annotation instanceof Description){ System.out.println(((Description) annotation).value()); } } } } catch (ClassNotFoundException e) { // TODO Auto-generated catch block e.printStackTrace(); } }
二.注解应用的一个小场景
1.场景要求
a.有一张用户表,字段包括用户ID,用户名,昵称,年龄,所在城市
b.方便的对每个字段或着字段的组合条件进行检索,并打印出SQL
2.实现方法
具体代码如下(其中注释了代码的思路):
a.用户表Users.java
@Table("user") public class Users { @Column("id") private int id; @Column("user_name") private String userName; @Column("nick_name") private String nickName; @Column("age") private int age; @Column("city") private String city; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getUserName() { return userName; } public void setUserName(String userName) { this.userName = userName; } public String getNickName() { return nickName; } public void setNickName(String nickName) { this.nickName = nickName; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } public String getCity() { return city; } public void setCity(String city) { this.city = city; } }
b.自定义注解Table.java
@Target({ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Inherited @Documented public @interface Table { String value(); }
c.自定义注解Column.java
@Target({ElementType.FIELD}) @Retention(RetentionPolicy.RUNTIME) @Inherited @Documented public @interface Column { String value(); }
d.测试类test.java
public class Test { public static void main(String[] args) { Users f1 = new Users(); f1.setId(68);//表示查询id为68的用户 f1.setAge(35);
Users f2 = new Users(); f2.setUserName("huhu"); f2.setCity("lanzhou");
Users f3 = new Users(); f3.setAge(20);
String sql1 = query(f1); String sql2 = query(f2); String sql3 = query(f3);
System.out.println(sql1); System.out.println(sql2); System.out.println(sql3); } @SuppressWarnings("unchecked") public static String query(Object obj) { StringBuilder s=new StringBuilder(); //1.获取到class Class class1 = obj.getClass(); //2.获取到table的名字 boolean isExist = class1.isAnnotationPresent(Table.class); if(isExist){ Table hTable=(Table)class1.getAnnotation(Table.class); s.append("select * from ").append(hTable.value()).append(" where 1=1"); } //3.遍历所有的字段 Field[] fields =class1.getDeclaredFields(); for (Field field : fields) { boolean exist = field.isAnnotationPresent(Column.class); if(exist){ Column f =(Column)field.getAnnotation(Column.class); String columnName = f.value(); //System.out.println(columnName); String fieldName = field.getName(); Object fieldString=null; String getMethodName="get"+fieldName.substring(0,1).toUpperCase()+fieldName.substring(1); try { /* * 4.处理每个字段对应的sql * 4.1拿到字段名 * 4.2拿到字段的值 * 4.3拼装sql */ Method aMethod = class1.getMethod(getMethodName); fieldString=(Object)aMethod.invoke(obj,null); //System.out.println(fieldString); if(fieldString==null||(fieldString instanceof Integer &&(Integer)fieldString==0)){ continue; } s.append(" and "+columnName); if(fieldString instanceof String){ s.append(" = "+"'"+fieldString+"'"); } else{ s.append(" = "+fieldString); } } catch (SecurityException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (NoSuchMethodException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IllegalArgumentException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IllegalAccessException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (InvocationTargetException e) { // TODO Auto-generated catch block e.printStackTrace(); } } } //Method[] msMethods = class1.getMethods(); //for (Method method : msMethods) { //method.invoke(class1, Object); //} return s.toString(); } }
三.一张图描述注解
谢谢大家阅读。