很多码农在写代码的时候不太爱写注释,结果任务一多,时间一长,需求一改,就完全不知道当初自己都干了些啥了。好在现在大多数编程语言都有注释功能,能够在代码里面做一些备注,不至于时间长了忘掉。但这些注释只是给人看的,机器并不会处理这些信息,而是把这些注释当作垃圾一样无视。

反过来,如果有些编程语言因为升级更新,替换掉了某些功能特性而导致开发受阻甚至不能使用,该怎么办呢?——这也难不倒科学家。他们想:既然码农可以写注释提醒自己不忘记代码是干什么的,那是不是也可以通过某种方法来提醒他们代码会出问题呢?

还真被他们找到了,这就是注解!

比如,像刚才说的场景:如果某个Java类被废弃了,怎么让码农们知道呢?——使用@Deprecated注解的就可以办到,就像这样:

如果类的方法被废弃,也是一样:

注解是JDK1.5中新增加的特性,它的作用说白了就是Java语言层面的「注释」,它主要是用来向JVM(Java虚拟机)解释说明类、对象、方法、属性、接口、抽象类等元素的信息。它是对数据的描述,可以说是关于数据的数据。

Java中提供了一些预定义的注解,例如@Override、@SuppressWarnings、@FunctionalInterface等,而Java开发框架的顶流Spring又在在此之上,提供了更多的注解,例如@Autowired、@Service、@RestController。

因为有了这些注解,码农的开发效率大大提高。举个最常见的栗子来说:

没有使用@Autowired注解:

/**
* 用户接口控制器
*
* @author 湘王
*/
public class UserController {
private UserService userService; // 用户登录接口
public void login(User user) {
// 先实例化UserService的实现类UserServiceImpl
userService = new UserServiceImpl();
// 继续实例化UserServiceImpl类中需要用到的各种其他类和对象
// TODO
} // 用户登出接口
public void logout(String userid) {
// 同样的过程可能要再来一次
userService = new UserServiceImpl();
// 继续实例化UserServiceImpl类中需要用到的各种其他类和对象
// TODO
}
}

使用了@Autowired注解:

/**
* 用户接口控制器
*
* @author 湘王
*/
public class UserController {
@Autowired
private UserService userService; // 用户登录接口
public void login(User user) {
// 直接调用UserServiceImpl实现类的各种对象和方法
// TODO
} // 用户登出接口
public void logout(String userid) {
// 直接调用UserServiceImpl实现类的各种对象和方法,而且不用重复实例化
// TODO
}
}

用了注解之后可以说是「舒服得不能再舒服了」。

Java目前提供了五种标准的注解(元注解,就是可以用来创造其他注解的注解)

@Target:表示注解可以用于哪些地方

@Retention:表示注解的适用范围

@Documented:将注解保存在javadoc中

@Inherited:允许子类继承父类的注解

@Repeatable:允许一个注解可被使用一次或多次

/**
* 定义注解可以应用在哪里
* PACKAGE:包
* TYPE:类、接口(包括注解类型)或者enum
* CONSTRUCTOR:构造器
* METHOD:方法
* FIELD:字段(包括enum实例)
* LOCAL_VARIABLE:局部变量
* PARAMETER:参数
*
* 如果省去@Target注解,那么注解可以应用于所有的ElementType
*
* @author xiangwang
*/
@Target(ElementType.METHOD)
/**
* 定义了注解在哪里可用
* SOURCE:源代码,将被编译器丢弃
* CLASS:class文件,会被JVM丢弃
* RUNTIME:运行时,一直保留
*/
@Retention(RetentionPolicy.RUNTIME)
// 将此注解保存在Javadoc中
@Documented
// 允许子类继承父类的注解
@Inherited
// 允许一个注解可以被使用一次或者多次
// @Repeatable(value = Object.class)
// 标记注解,不包含任何元素
public @interface Test {}

尝试着举一个小栗子,比如最常见的,验证密码有效性:

/**
* 定义注解
*
* @author xiangwang
*/
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface UseCase {
int id() default -1;
String description() default "";
}

可以用两种方式来看看效果:

1、直接运行注解:

/**
* 使用注解
*
* @author xiangwang
*/
public class PasswordUtils {
// 使用@UserCase注解
@UseCase(id = 1, description = "密码必须至少包含一位数字")
public static boolean validate(String password) {
return (password.matches("\\w*\\d\\w*"));
} public static void main(String[] args) {
System.out.println(PasswordUtils.validate("w1errd"));
}
}

2、或者使用昨天介绍的反射来解析注解:

/**
* 解析注解
*
* @author xiangwang
*/
public class UseCaseTracker {
public static void trackUseCases(List<Integer> useCases, Class<?> cl) {
for (Method m : cl.getDeclaredMethods()) {
UseCase uc = m.getAnnotation(UseCase.class);
if (uc != null) {
System.out.println("发现注解id:" + uc.id() + " - 注解description:" + uc.description());
// 剔除已有注解
useCases.remove(Integer.valueOf(uc.id()));
}
}
useCases.forEach(i -> System.out.println("缺少用例id:" + i));
} public static void main(String[] args) {
List<Integer> useCases = IntStream.range(1, 5).boxed().collect(Collectors.toList());
trackUseCases(useCases, PasswordUtils.class);
}
}

这是最最简单的注解形式。注解真正的用武之地是各类开发框架,尤其是ORM框架,明天就自己尝试着实现一个:)

Java注解(1):码农的小秘的更多相关文章

  1. 微信小程序——智能小秘“遥知之”源码分享(语义理解基于olami)

    微信小程序智能生活小秘书开发详解 >>>>>>>>>>>>>>>>>>>>> ...

  2. Java自定义注解源码+原理解释(使用Java自定义注解校验bean传入参数合法性)

    Java自定义注解源码+原理解释(使用Java自定义注解校验bean传入参数合法性) 前言:由于前段时间忙于写接口,在接口中需要做很多的参数校验,本着简洁.高效的原则,便写了这个小工具供自己使用(内容 ...

  3. 支持语音识别、自然语言理解的微信小程序(“遥知之”智能小秘)完整源码分享

    记录自己搭建https的silk录音文件语音识别服务的调用过程,所有代码可在文中找链接打包下载 >>>>>>>>>>>>> ...

  4. 一名Java架构师分享自己的从业心得,从码农到架构师我用了八年

    工作了挺久,发现有个挺有意思的现象,从程序员.高级程序员,到现在挂着架构师.专家之类的头衔,伴随着技术和能力的提高,想不明白的事情反而越来越多了. 这些疑问有些来自于跟小伙伴的交流,有些是我的自问自答 ...

  5. JAVA小项目实例源码—学习娱乐小助手

    代码地址如下:http://www.demodashi.com/demo/11456.html 一.程序实现 项目目录: MyJFrame:实现项目界面样式: AppProcess:实现调用api或爬 ...

  6. 小师妹学JVM之:java的字节码byte code简介

    目录 简介 Byte Code的作用 查看Byte Code字节码 java Byte Code是怎么工作的 总结 简介 Byte Code也叫做字节码,是连接java源代码和JVM的桥梁,源代码编译 ...

  7. 码农飞升记-Java是什么?

    1.Java概述 Java 原名 Oak 是 Sun Microsystems 公司的 James Gosling 及其团队于 1995 年 5 月推出的 Java 程序设计语言 和 Java 平台 ...

  8. 老码农冒死揭开行业黑幕:如何编写无法维护的代码[ZZ]

    下面是一篇有意思的"代码大全",可谓 逆软件工程. 老码农冒死揭开行业黑幕:如何编写无法维护的代码 原文如下 让自己稳拿铁饭碗 ;-) – Roedy Green(翻译版略有删节) ...

  9. Java注解-元数据、注解分类、内置注解和自定义注解|乐字节

    大家好,我是乐字节的小乐,上次说过了Java多态的6大特性|乐字节,接下来我们来看看Java编程里的注解. Java注解有以下几个知识点: 元数据 注解的分类 内置注解 自定义注解 注解处理器 Ser ...

随机推荐

  1. 聚是一团火散作满天星,前端Vue.js+elementUI结合后端FastAPI实现大文件分片上传

    原文转载自「刘悦的技术博客」https://v3u.cn/a_id_175 分片上传并不是什么新概念,尤其是大文件传输的处理中经常会被使用,在之前的一篇文章里:python花式读取大文件(10g/50 ...

  2. Luogu3919 【模板】可持久化数组(主席树)

    主席树模板题,注意空间\((n+m) \log(n)\) #include <iostream> #include <cstdio> #include <cstring& ...

  3. Vue3 组合式 API 中获取 DOM 节点的问题

    模板引用 Vue 提供了许多指令让我们可以直接操作组件的模板.但是在某些情况下,我们仍然需要访问底层 DOM 元素.在模板中添加一个特殊的属性ref就可以得到该元素. 访问模板引用 <scrip ...

  4. Web 布局设计(一):固定侧边栏

    前言 闲着无事,做一些实战练习,今天实现一个如标题所示的布局设计.通过此次布局设计,我希望掌握position属性值 fixed.absolute.relative.width和height属性值 i ...

  5. JS判断两个数组的元素是否完全相等

    1.使用ES6 新增的扩展运算符和Set新数据类型判断两个数组是否包含有相同的元素 var arr1 = ['green' , 'yellow' ,'blue' ,'red']; var arr2 = ...

  6. ubantu安装中文失败的改正方法

    情况说明 一开始我也是像博主们那样安装,但是一直在这个界面安装失败,太悲伤了!! 报错内容 c Failed to fetch http://us.archive.ubuntu.com/ubuntu/ ...

  7. 【JDBC】学习路径6-SQL插入、修改、删除数据

    第一章:插入使用.executeUpdate(); 返回的是受到影响的数据条数. public static boolean insert(String username,String passwor ...

  8. .Net下的Http请求调用(Post与Get)

    http请求调用是开发中经常会用到的功能.在内,调用自有项目的Web Api等形式接口时会用到:在外,调用一些第三方功能接口时,也会用到,因为,这些第三方功能往往是通过http地址的形式提供的,比如: ...

  9. KingbaseESV8R6如何针对表单独设置vacuum策略

    背景 书接上文 KingbaseES应对表年龄增长过快导致事务回卷 ,有些特殊业务场景不能靠全局的autovacuum方法,例如大型数据库系统,频繁做update的系统,还有上文提到的做实时数据同步的 ...

  10. git merge和git rebase总结

    dev分支 * da349ef (dev) e * 75350bc d * 63cbbb8 c * c6509a5 b * 13405af a 文件可能会发生冲突,需要解决一下 aaaaaaaaa b ...