写在前面
找结论,直接看 4、总结
写这篇文章之前,我先百度了一下网上的答案,发现大多数文章用了类似的解释,如下:
Java 注解也叫元数据,一种代码级别的说明。Python 装饰器是一种语法糖。
注解是给别人看的,功能不仅仅由注解决定;装饰器直接拦截,直接改变被装饰对象的行为!
注解(Annotation):仅提供附加元数据支持,并不能实现任何操作。需要另外的 Scanner 根据元数据执行相应操作。
装饰器(Decorator):仅提供定义劫持,能够对类及其方法的定义并没有提供任何附加元数据的功能。
看起来 注解和装饰器 完全在两个不同的方向作用,但真的是这样吗?
以下仅为个人见解,欢迎大佬批评指正
1、写法异同点
@aaa(“abc”)
相同点:都是@开头,注解、装饰器都可以自定义、都可以带参数、都可以被标注代码块之前执行。
不同点: java注解可以写在类、方法、变量头上,python装饰器只能写在方法(函数)头上。
2、在实例中对比
下面对注解和装饰器分别举例,对比其区别
2.1、java注解
@Override
public String toString() {
return "Teacher{" +
"id=" + id +
", name='" + name + '\'' +
", users=" + users +
'}';
}
@Override 的作用是标记重写方法,在编译阶段对方法进行检查。
从这个例子中我们发现注解作用在了对方法的编译、检查上,并未对方法的内容和功能发生改变。
2.2、python装饰器
class A():
@property
def bb(self):
return self.value
if __name__=="__main__":
A.bb = 9
print A.bb
@property 的作用是将函数bb,做为属性使用
从这个例子中我们发现装饰器能够直接改变函数的功能。
2.3、简单总结
如上两个简单的例子,我们简单总结出注解和装饰器不一样的地方
1、注解对元数据进行了检查、对比等工作,不会对所修饰的代码产生直接的影响。
2、装饰器可以对方法进行功能上的改变,可以对所修饰的代码产生直接的影响。
到这里,你也许会觉得 装饰器和注解 走的路线完全不同了,其实不然,在java中注解+反射能够实现和python里装饰器同样的效果。
3、更多实例对比
3.1、注解
1、在mybatis中使用注解传递sql
(这里并不能直接给getUsers传参数,而是利用了另一个类反射注解的参数,然后传值给getUsers)
2、@Test标记该方法为测试方法
3.2、装饰器
1、日志打印器
2、计算运行时间
4、总结
第一点:对代码块的影响
java注解:不会对所修饰的代码产生直接的影响。
python装饰器:可以对所修饰的代码产生直接的影响。
第二点:共通处
java中注解+反射 可以实现 python装饰器同样的功能,包括面向切面编程、参数校验等。
第三点:从用途看
从用途看注解像是注释文档一样,用于生成javadoc文档(以参数形式标注)、检查等。
装饰器像是为函数提供更多的功能,并装在不同的函数身上。
第四点:从原理看
java注解:所有注解本质是继承自接口(Annotation)的接口
我们看一个 JDK 内置注解的定义:
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.SOURCE)
public @interface Override {
}
这是注解 @Override 的定义,其实它本质上就是:
public interface Override extends Annotation{
}
python装饰器:被装饰函数的返回值 作为参数传给闭包函数执行(这个闭包函数名前面加个@,就是装饰器)
说白了装饰器就是一个闭包函数。
下面是一个闭包函数的实例
def outer(x):
def inner(): # 函数嵌套
return x # 跨域访问,引用了外部变量x
return inner # 函数作为返回值
closure = outer('外部变量') # 函数作为变量赋给closure
print(closure()) # 执行闭包
#结果
外部变量
下面是一个装饰器实例
def outer(x):
def inner():
return '戴了inner牌帽子的 ' + x()
return inner
@outer
def func():
return '函数func'
print(func())
#结果
戴了inner牌帽子的 函数func
5、闲谈一下
在写这篇文章时查阅了部分资料和网上的文章,
有人说注解和装饰器是两个完全扯不上关系的语法。
要从原理上讲,也可以说他们完全不一样,也不属于同一个体系。
但是这里我们简单揣摩一下,
java作者和python作者开发这两个语法想要实现功能是什么,
以及大多数使用者(一些框架开发者),使用注解或装饰器去实现的功能是什么。
拿mybatis中的sql注解来说,利用和反射的结合,实现了代码的简化、直观。
如果python用装饰器来实现这个功能,无疑同样完美,源代码还会更少更简洁。
还有很多的例子,如lombok、springboot等里面都有很多的注解,
他们所实现的功能不正是python中期望装饰器实现的功能?
所以殊途同归,
打开一个注解源码,里面可能仅记录了一些字段及默认值,功能由其他反射和类实现。
打开一个装饰器,里面是一个闭包函数,功能由它自己实现。
最终他们都为完成同样的一些事和效果。
使用注解(Annotation)的语言:AtScript、Java、C#(叫 Attribute)。
使用装饰器(Decorator)的语言:Python、JavaScript/ECMAScript。