python中的装饰器和java中的注解_浅谈java中注解和python中装饰器的区别

写在前面

找结论,直接看 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)

2020082818034591.png

2、@Test标记该方法为测试方法

20200828183344594.png

3.2、装饰器

1、日志打印器

20200828184011170.png

2、计算运行时间

2020082820100511.png

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

20200828211244133.png

5、闲谈一下

在写这篇文章时查阅了部分资料和网上的文章,

有人说注解和装饰器是两个完全扯不上关系的语法。

要从原理上讲,也可以说他们完全不一样,也不属于同一个体系。

但是这里我们简单揣摩一下,

java作者和python作者开发这两个语法想要实现功能是什么,

以及大多数使用者(一些框架开发者),使用注解或装饰器去实现的功能是什么。

拿mybatis中的sql注解来说,利用和反射的结合,实现了代码的简化、直观。

如果python用装饰器来实现这个功能,无疑同样完美,源代码还会更少更简洁。

还有很多的例子,如lombok、springboot等里面都有很多的注解,

他们所实现的功能不正是python中期望装饰器实现的功能?

所以殊途同归,

打开一个注解源码,里面可能仅记录了一些字段及默认值,功能由其他反射和类实现。

打开一个装饰器,里面是一个闭包函数,功能由它自己实现。

最终他们都为完成同样的一些事和效果。

使用注解(Annotation)的语言:AtScript、Java、C#(叫 Attribute)。

使用装饰器(Decorator)的语言:Python、JavaScript/ECMAScript。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值