Groovy中那些神奇注解之Memoized
临近年关手头比较闲,去看了一下Groovy的官方文档,才发现原来Groovy中带了那么多的注解,很多注解带来的效果,有时候让人感觉“这不是在变魔法吧”。
个人很喜欢Groovy,写不成Ruby,Groovy一样让我很愉悦。可惜在国内Groovy实在没什么人气,连个像样的官方站都没有,或许是因为喜欢脚本的都在写ruby,python,写Java的又对他不屑一顾?
接下来的一系列文章,就只介绍一些我个人觉得很好用,在开发中用到机会会比较多的注解。
PS:强烈推荐Groovyr官方文档,相当的详细,地址:http://www.groovy-lang.org/documentation.html,不用担心是英文,有点代码经验的,一看示例代码就知道是什么意思了。
好了,闲话少说,第一个注解:Memoized
Memoized的全称是:groovy.transform.Memoized,在groovy.transform包下,有很多相关的注解,可以好好了解一下
至于Memoized的作用,看名字就很明白了,就是“记住”,没错,就是把方法的执行结果缓存起来,下次调用时,如果参数一样,那方法就不再计算而是直接返回结果了,该注解对于那些参数范围比较少,并且只要参数一定,那么结果肯定也是唯一的方法,有着很好的加速效果。
举个最简单,但是个人觉得最能体现Memoized效果的例子,大家应该都实现过用递归方式实现斐波那契(fibnacci)数列吧,但是纯粹的不加优化的递归方式是相当相当的没效率的,在我的电脑上,计算到40左右,电脑就有点跑不动了,看看代码:
class Fibnacci{
long total = 0 //用来统计一共计算了多少次
int n //计算多少的数列
public Fibnacci(int n){
this.n = n
}
def fibnacci_1(n){
total ++
n < 2 ? 1 : fibnacci_1(n-1) + fibnacci_1(n-2)
}
@groovy.transform.Memoized
def fibnacci_2(n){
total ++
n < 2 ? 1 : fibnacci_2(n-1) + fibnacci_2(n-2)
}
//测试不使用Memoized
def runWithoutMem(){
long start = System.currentTimeMillis()
total = 0
def val = fibnacci_1(n)
long time = System.currentTimeMillis() - start
println "未使用Memoized,${n}的Fibnacci值是${val},共用时${time}毫秒,函数调用共${total}次"
}
//测试使用Memoized
def runWithMem(){
long start = System.currentTimeMillis()
total = 0
def val = fibnacci_2(n)
long time = System.currentTimeMillis() - start
println "使用Memoized,${n}的Fibnacci值是${val},共用时${time}毫秒,函数调用共${total}次"
}
}
def fib = new Fibnacci(40)
fib.runWithoutMem()
fib.runWithMem()
fibnacci_1和fibnacci_2方法实现是完全一样的,唯一的不一致就是一个加上了注解,同时我在类里面加上了一个变量total,来统计方法一共被调用了多少次,把上面的代码复制到Groovy自带的GroovyConsole程序中(神器呀),直接就可以运行,看看结果:

结果很惊人吧,因为Memoized把计算的结果缓存下来了,如果参数一样就不用再重复计算了,所以效率自然大大提高,这就好像我们用数组方法来实现fibnacci从而提高效率原理是差不多的,不过这样小小的一个注解,可以少写好多代码省不少事了:)
好了,就到这里了,休息,休息一下

Groovy中那些神奇注解之Memoized的更多相关文章
- Groovy中那些神奇注解之ToString
继续上一篇:Groovy中那些神奇注解之Memoized 这篇就讲讲@groovy.transform.ToString这个注解,这注解太熟悉了,熟悉到让人一看就知道是干吗的,不就是把Bean转在St ...
- Groovy中那些神奇注解之InheritConstructors
上一篇:Groovy中那些神奇注解之ToString 写完ToString,本来想今天就写到这了,突然觉得InheritConstructors注解实在也是个神器,写起来也没多少字,还是写了吧. In ...
- Mybatis中的@SelectKey注解
一.创建Maven项目 在pom.xml中,添加mybatis依赖,mysql-jdbc依赖,把编译版本改为1.8 你问,为啥mybatis不会自动依赖mysql-jdbc,需要手动写明?答:因为my ...
- Groovy中的面向对象
Groovy中的面向对象 前面说到groovy支持脚本和类,前面一节已将简单说了脚本和类之间的关系,这一节主要介绍一下groovy中类的相关知识,即面向对象相关知识. 1.类型 1.1 原始类型 gr ...
- 深入理解spring中的各种注解
Spring中的注解大概可以分为两大类: 1)spring的bean容器相关的注解,或者说bean工厂相关的注解: 2)springmvc相关的注解. spring的bean容器相关的注解,先后有:@ ...
- Java中万恶的注解
本文由码农网 – 孙腾浩原创翻译,转载请看清文末的转载要求,欢迎参与我们的付费投稿计划! 当Java 1.5引入注解,企业开发者对简化EJB和其他企业产品开发抱有很大期望.可以看一看同一时期的一篇文章 ...
- Junit中常用的注解说明
Java注解((Annotation)的使用方法是@注解名 ,能通过简单的词语来实现一些功能.在junit中常用的注解有@Test.@Ignore.@BeforeClass.@AfterClass.@ ...
- ASP.NET MVC5中的数据注解
ASP.NET MVC5中Model层开发,使用的数据注解有三个作用: 数据映射(把Model层的类用EntityFramework映射成对应的表) 数据验证(在服务器端和客户端验证数据的有效性) 数 ...
- 深入理解spring中的各种注解(转)
Spring中的注解大概可以分为两大类: 1)spring的bean容器相关的注解,或者说bean工厂相关的注解: 2)springmvc相关的注解. spring的bean容器相关的注解,先后有:@ ...
随机推荐
- hadoop笔记之MapReduce原理
MapReduce原理 MapReduce原理 简单来说就是,一个大任务分成多个小的子任务(map),并行执行后,合并结果(reduce). 例子: 100GB的网站访问日志文件,找出访问次数最多的I ...
- codeforces 383C Propagating tree 线段树
http://codeforces.com/problemset/problem/383/C 题目就是说, 给一棵树,将一个节点的值+val, 那么它的子节点都会-val, 子节点的子节点+val. ...
- 让vs2010的html编辑器验证html5语法
或者在Tools -> option -> Text Editor -> Html -> Validation
- poj 3259 (Bellman_Ford判断负环)
题意:John的农场里n块地,m条路连接两块地,k个虫洞,虫洞是一条单向路,不但会把你传送到目的地,而且时间会倒退Ts.我们的任务是知道会不会在从某块地出发后又回来,看到了离开之前的自己. 思路:虫洞 ...
- 点击按钮改变标签内容(采用lambda函数方式)
from Tkinter import* window=Tk() counter=IntVar() counter.set(0) def click(variable,value): variable ...
- 柯南君:看大数据时代下的IT架构(2)消息队列之RabbitMQ-基础概念详细介绍
一.基础概念详细介绍 1.引言 你是否遇到过两个(多个)系统间需要通过定时任务来同步某些数据?你是否在为异构系统的不同进程间相互调用.通讯的问题而苦恼.挣扎?如果是,那么恭喜你,消息服务让你可以很轻松 ...
- OA项目之打印
打印 若此页有一个打印按钮: <input type="button" id="btnPrint" class="button_sm7" ...
- iOS点滴- ViewController详解
一.生命周期 当一个视图控制器被创建,并在屏幕上显示的时候. 代码的执行顺序 1. alloc 创建对象,分配空间 2.init (initW ...
- 推荐两个不错的CAD二次开发(.Net)手册
推荐两个不错的CAD二次开发(.Net)手册 http://www.mjtd.com/helpcenter/netguide/index.html http://www.ceesky.com/book ...
- 如何利用Github Pages展示自己写的项目
接触github很久了,自己搭建过hexo博客,但是对于web项目托管github pages感觉很懵,所以在此总结分享给有需要的亲们. 教程开始: 1.创建一个新库 2.给库命名 3.创建新库后点击 ...