这两天完善了我的jaskell语言的一个shell。

这个shell虽然是jaskell的,但是也可以作为一个交互式执行java代码的解释器。对于想快速地试试某个api比较有用。

相比于eclipse scrapebook,它的好处是更方便,而且,jaskell的一些函数式的特性让你可以写出更加简洁的代码。

下面举几个例子:

轻松玩Swing

打开shell,它显示这样:

Jaskell Shell version 
0.5
>
然后假设你要运行一下javax.swing.JOptionPane.showInputDialog()函数的话,你可以这样写:
>  javax.swing.JOptionPane.showInputDialog[ " your age? " ]
>
回车两下(第一下,就是简单折行,因为你可以接着写下一行代码,只有连续两下折行,shell才认为你是要执行)
结果就会出现一个简单的swing对话框。在对话框里面输入年龄“13",回车,
shell里面就会显示:13。


这里面,一点需要注意的,java方法调用不用圆括号,而是方括号。你可以把这个理解为一个reflection调用,传递的永远都是一个数组。

下面,假设你想重复地使用JOptionPane这个类,使用showInputDialog, showConfirmDialog这类的方法,总这么写javax.swing.JOptionPane也够麻烦的。我们可以简化它:
>  dialog  =  javax.swing.JOptionPane
>
=> class  javax.swing.JOptionPane
当你回车两次后,shell在"=>"提示符后面自动显示这个表达式的值:"class javax.swing.JOptionPane"。

下面我们可以重复使用dialog变量了。在这之前,我们可能想看看JOptionPane到底都支持什么静态方法,我们可以用"?"来让shell告诉我们
>   ?  dialog
这个"?"不是jaskell语言的一部分,而是shell的命令,所以不需要回车两次。回车,shell就会把JOptionPane的所有方法都列出来。

然后假设我们选择showMessageDialog,可以这样写:
>  dialog.showMessageDialog[ null " hello world " ]
>
然后,一个"hello world"的对话框就弹了出来(看不见?找一找。它可能被藏在你的当前窗口后面了。)

更简化一点,假设我要重复showMessageDialog若干遍,我可以这样写:
>  say msg  =  dialog.showMessageDialog[ null , msg]
>
=> say()
这个表达式的值是一个接受一个参数的,叫做say的函数。

下面你可以say很多东西啦:
>  say  " how are you? "
>

>  say  " java sucks! "
>
等等等等。



傻瓜多线程

好,看过了JOptionPane,我们来看看多线程。下面是用这个语言怎么启动一个线程:
>  Thread. new [(\_ -> System.out.println[ " hello world " ]) ` implements  Runnable].start[]
>
hello world
这里面Thread.new大概不需要解释。这里对构造函数的调用是Ruby风格的ClassName.new,而不是java的new ClassName。

构造函数也需要用一个list来传递参数。我们这里传递的是一个Runnable对象。
用来实现Runnable接口的是一个匿名函数,这个函数不管参数是什么,一旦调用,就println一下。

implements 是一个函数,它负责用一个函数来动态生成一个实现某接口的proxy出来。它前面的那个反向单引号表示把一个函数以中缀语法调用,所以 (somefunction `implements Runnable)等价于implements(somefunction, Runnable)。

"\"符号是lamda函数表示法。"->"符号前面的是函数参数,后面的是函数体。这里因为我们不使用这个参数,所以用"_"这个通配符。

最后,我们调用start[]方法来执行这个线程。


我们还可以用标准的"const"函数来让代码更简短一点。" const x"语义上完全等价于" \_->x"。另外,我们也可以用java风格的new操作符函数来写,jaskell对两者都支持的:
>   new  Thread[ const (System.out.println[ " hello world " ]) ` implements  Runnable].start[]
>
hello world


然后,考虑到重用,我们可以这样,先把System.out.println搞短一点,每次敲这么长太麻烦:
>  println msg  =  System.out.println[msg]
>
=> println()

(实际上,println函数是系统已经缺省就定义好的了。你完全没有必要自己定义println就可以直接用了。这里只是演示一下怎么自己定义函数)


然后,把那段启动线程的代码写成函数:
>  run task  =  Thread. new [const task ` implements  Runnable].start[]
>
=> run()

好了,下面我们可以任意启动线程做事情了:
>  run(println  " hello world " )
>
hello world

>  run(println  " pei! " )
>
pei
!

>  run (say  " nice! " )
>

最后一个say "nice!",如果不用单独线程的话,运行后这个对话框将阻塞当前线程。现在用run来运行它,就不会阻塞了。


今天你fp了吗?

最后再随便看看jaskell作为函数式语言的本分所能做的一些事。

foreach函数:

>  foreach [ 1 , 2 , "hello" ] println
>
这个函数把1,2,"hello"三个都打印一遍。

>  foreach(list  1   100 , println)
>
这个函数把数字1到100按顺序打印一遍。

map函数:
>  map (\x -> Integer.parseInt[x]) [ " 1 " , " 2 " , " 3 " ]
>
=> [
1 , 2 , 3 ]
这个函数把一个字符串列表转换成一个整数列表。


filter函数:


>  filter(\x -> x < 10 , list  1   100 )
>
=> [
1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 ]
这个代码把1到100中所有小于10的整数都取出来。

filter函数的第一个参数是一个函数,这个函数对列表中的每一个元素都进行判断,返回true或者false。

find函数:

>  find( 3 , list  1   100 )
>
=> 2
这个代码在列表中寻找整数2,如果找到,返回找到的位置(0为第一个)


lookup函数:

>  lookup(\x -> x * x > x + 50 , list  1   100 )
>
=> 7
这个代码在列表中寻找第一个符合x*x>x+50的元素,找到就返回位置。

@函数:

上面我们知道找到的数字是在位置7,可以用@来得到位置7的值:

>  list  1   100  @  7
>
=> 8
好,这个数是8。


sum函数:

>  sum(list  1   100 )
>
=> 5050


注意,最难的来了!
fold函数:

>  fold ( * 1  [ 1 , 2 , 3 , 4 , 5 ]
>
=> 120
这个fold函数以1为初始值,然后对每个列表元素,把它和当前值做乘法,用结果更新当前值,最后把计算结果返回。所以这段代码实际上做的就是把从1到5乘起来。
(*)是一个乘法函数。

你也可以用自己写的:
>  fold (\x y -> x * y)  1  [ 1 , 2 , 3 , 4 , 5 ]
一样的。


行了,差不多了。下载在:

http://dist.codehaus.org/yan/distributions/jaskell.zip

把jar文件都放到你的classpath里面,然后运行jfun.jaskell.shell.Shell类就行了。