Java注解(1):码农的小秘
很多码农在写代码的时候不太爱写注释,结果任务一多,时间一长,需求一改,就完全不知道当初自己都干了些啥了。好在现在大多数编程语言都有注释功能,能够在代码里面做一些备注,不至于时间长了忘掉。但这些注释只是给人看的,机器并不会处理这些信息,而是把这些注释当作垃圾一样无视。
反过来,如果有些编程语言因为升级更新,替换掉了某些功能特性而导致开发受阻甚至不能使用,该怎么办呢?——这也难不倒科学家。他们想:既然码农可以写注释提醒自己不忘记代码是干什么的,那是不是也可以通过某种方法来提醒他们代码会出问题呢?
还真被他们找到了,这就是注解!
比如,像刚才说的场景:如果某个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):码农的小秘的更多相关文章
- 微信小程序——智能小秘“遥知之”源码分享(语义理解基于olami)
微信小程序智能生活小秘书开发详解 >>>>>>>>>>>>>>>>>>>>> ...
- Java自定义注解源码+原理解释(使用Java自定义注解校验bean传入参数合法性)
Java自定义注解源码+原理解释(使用Java自定义注解校验bean传入参数合法性) 前言:由于前段时间忙于写接口,在接口中需要做很多的参数校验,本着简洁.高效的原则,便写了这个小工具供自己使用(内容 ...
- 支持语音识别、自然语言理解的微信小程序(“遥知之”智能小秘)完整源码分享
记录自己搭建https的silk录音文件语音识别服务的调用过程,所有代码可在文中找链接打包下载 >>>>>>>>>>>>> ...
- 一名Java架构师分享自己的从业心得,从码农到架构师我用了八年
工作了挺久,发现有个挺有意思的现象,从程序员.高级程序员,到现在挂着架构师.专家之类的头衔,伴随着技术和能力的提高,想不明白的事情反而越来越多了. 这些疑问有些来自于跟小伙伴的交流,有些是我的自问自答 ...
- JAVA小项目实例源码—学习娱乐小助手
代码地址如下:http://www.demodashi.com/demo/11456.html 一.程序实现 项目目录: MyJFrame:实现项目界面样式: AppProcess:实现调用api或爬 ...
- 小师妹学JVM之:java的字节码byte code简介
目录 简介 Byte Code的作用 查看Byte Code字节码 java Byte Code是怎么工作的 总结 简介 Byte Code也叫做字节码,是连接java源代码和JVM的桥梁,源代码编译 ...
- 码农飞升记-Java是什么?
1.Java概述 Java 原名 Oak 是 Sun Microsystems 公司的 James Gosling 及其团队于 1995 年 5 月推出的 Java 程序设计语言 和 Java 平台 ...
- 老码农冒死揭开行业黑幕:如何编写无法维护的代码[ZZ]
下面是一篇有意思的"代码大全",可谓 逆软件工程. 老码农冒死揭开行业黑幕:如何编写无法维护的代码 原文如下 让自己稳拿铁饭碗 ;-) – Roedy Green(翻译版略有删节) ...
- Java注解-元数据、注解分类、内置注解和自定义注解|乐字节
大家好,我是乐字节的小乐,上次说过了Java多态的6大特性|乐字节,接下来我们来看看Java编程里的注解. Java注解有以下几个知识点: 元数据 注解的分类 内置注解 自定义注解 注解处理器 Ser ...
随机推荐
- 自动挂载mount
# 自动挂载mount(/etc/fstab) /dev/fd0 /media/floppy0 auto rw,user,noauto,exec,utf8 0 0 # 第一列:/dev/fd0 挂载源 ...
- 5. MGR管理维护 | 深入浅出MGR
GreatSQL社区原创内容未经授权不得随意使用,转载请联系小编并注明来源. 目录 1. 切换主节点 2. 切换单主/多主模式 3. 添加新节点 4. 删除节点 5. 异常退出的节点重新加回 6. 重 ...
- php YII2空数组插入报错问题处理 Array to string conversion
问题描述 前端传空数组 [],php接收后处理不当插入数据库时报错Array to string conversion 参数示例 { "id": 0, //ID整型 "t ...
- Apache DolphinScheduler 1.3.6 功能发布说明
参与人员 @chengshiwen.@hailin0.@wanghong1314.@ruanwenjun.@xxjingcd.@zhangguohao.@zhuangchong.@syb8535531 ...
- Luogu5020 货币系统 (完全背包)
bool型完全背包 #include <iostream> #include <cstdio> #include <cstring> #include <al ...
- Luogu3802 小魔女帕琪 (排列组合)
注意除数为0情况 #include <iostream> #include <cstdio> #include <cstring> #include <alg ...
- CF580D Kefa and Dishes (状压DP)
枚举最后食物 #include <iostream> #include <cstdio> #include <cstring> #include <algor ...
- [NOI P模拟赛] 传统艺能(子序列自动机、矩阵乘法,线段树)
(2:00)OID:"完了,蓝屏了!"(代码全消失) 众人欢呼 OID:开机,"原题测试--" (30min later)OID 开始传统艺能: " ...
- 【碳硫磷模拟赛】消失的+和* (树形DP)
好久没做过这么恶心的DP题了 题面 题面很简单,有一个计算式,由+号.*号.括号和小于10的正整数组成,现在所有的+和*(由于属于违禁词而)都被-号给和谐掉了,现在要求所有可能的原计算式的结果之和. ...
- 【读书笔记】C#高级编程 第四章 继承
(一)继承的类型 1.实现继承和接口继承 在面向对象的编程中,有两种截然不同的继承类型:实现继承和接口继承. 实现继承:表示一个类型派生于一个基类型,它拥有该基类型的所有成员字段和函数.在实现继承中, ...