Ocaml程序的结构

现在我们将从整体上观察一些OCaml程序。我将传授以下内容:局部和全局定义,何时使用;;而何时用;,模块,嵌套函数,以及引用。为此我们会见到很多现在还不理解意义的目前还未接触过的OCaml概念。不用担心这些细节,只要专注于程序的整体形状以及我指出的那些特性。

局部"变量"(实际是局部表达式)

我们来看C中的average函数并且加一个局部变量。

double average (double a, double b)
{
  double sum = a + b;
  return sum / 2;
}

现在来对OCaml版本做同样的事:

let average a b =
  let sum = a +. b in
  sum /. 2.0;;

标准短语let name = expression in用来定义一个命名的局部表达式。以后name在这个函数中就可以用来代替expression,直到一个;;结束本代码块。注意我们在in之后并不缩进.只要把let ... in看作是一个整体就可以。

现在我们比较一下C局部变量和这些命名的局部表达式的区别。它们是不一样的。C变量sum在栈上有已经分配的位置,如果你需要你可以稍后在函数中给它赋值,甚至可以直接得到它的地址值。而在OCaml中却不能这样。OCaml中sum只是表达式a +. b的简短别名。我们无法给sum赋值或改值。(稍后我们会看到对真正的变量我们可以这样做。)

译注:这并不是说每引用一次suma +. b就会被求值一次。sum是一个局部绑定(binding),其对于值是一个绑定关系,绑定是不可变的,因而不是变量,只能被隐藏,不能被改写。

下面的例子更清楚地说明这个区别。下面的两段代码返回同样的值((a+b) + (a+b)^2^):

let f a b =
  (a +. b) +. (a +. b) ** 2.
  ;;
let f a b = let x = a +. b in x +. x ** 2. ;;

后者可能快一些(但大多数编译器应该可以做到“消除公共子表达式”。),而且可读性更强。后者中的x只是a +. b的缩略名称.

全局“变量”(实际是全局表达式)

我们也可以在top level中像上面定义局部“变量”那样定义全局名,但实际上这些都不是真正的变量,而只是缩略名。下面是一个缩减后的实际代码:

let html =
  let content = read_whole_file file in
  GHtml.html_from_string content
  ;;
let menu_bold () = match bold_button#active with | true -> html#set_font_style ~enable:[`BOLD] () | false -> html#set_font_style ~disable:[`BOLD] () ;;
let main () = (* code omitted *) factory#add_item "Cut" ~key:_X ~callback: html#cut ;;

在这段实际代码中,html是一个HTML编辑小部件(widget)(来自lablgtk库的一个对象),它在程序开始处被第一个let html =语句创造,而后在以后的几个函数中被引用。

注意上面程序段中的html名不是像C或其他命令式语言中那样的真正的全局变量。并没有空间被分配用来保存“html指针”(译注:但是由内存分配用来保存值本身),也无法给html赋值,比如,让它指向另一个小部件。以后的章节中我们会介绍真正的变量:引用。

Let-绑定

任何let ...,无论是在top level(全局)还是在一个函数中,经常被称作let-绑定(binding).

引用:真正的变量

如果我们需要真正的可赋值和改值的变量怎么办?我们需要用到引用。引用很像C/C++中的指针.在Java中,所有存放对象的变量实际都是指向对象的指针。在Perl中,引用就是引用,和OCaml一样。

下面举例说明在OCaml中怎样创造一个int的引用:

# ref 0;;
- : int ref = {contents = 0}

实际这个语句并无用处。我们创造了一个引用,但是因为没有命名它会被垃圾收集器立刻收走。(实际上在编译时就可能被扔掉。)我们来命名一个引用。

# let my_ref = ref 0;;
val my_ref : int ref = {contents = 0}

此引用现在保存一个整数0。让我们给它赋其他值:

# my_ref := 100;;
- : unit = ()

然后我们来看这个引用包含的值:

# !my_ref;;
- : int = 100

因此,:=操作符用来给引用赋值,!操作符用来取出引用的值(解引用)。 下面是一个粗略的比较:

OCaml                   C/C++

let my_ref = ref 0;;    int a = 0; int *my_ptr = &a;
my_ref := 100;;         *my_ptr = 100;
!my_ref                 *my_ptr

引用是有用的,但你会发现你不会经常用它。在函数定义中用let name = expression in来命名局部表达式将会常用得多。

嵌套函数(Nested Functions)

C中事实上没有嵌套函数的概念。GCC对C支持嵌套函数但是我发现几乎没有程序用到这个扩展。不管怎样,下面是GCC的info页给出的关于嵌套函数的说明:

“嵌套函数”是在另一个函数内部定义的函数。(GNU C++ 不支持嵌套函数)嵌套函数的名字域是它被定义的那个程序块。例如:下面我们定义一个名为‘square‘的嵌套函数,然后调用它两次:

foo (double a, double b)
{
  double square (double z) { return z * z; }

  return square (a) + square (b);
}

嵌套函数可以使用包含它的函数当前可见的所有变量。这就是"lexical scoping(词法变量域)"。例如,下面我们演示一个嵌套函数如何使用一个名为'offset'的继承变量。

bar (int *array, int offset, int size)
{
  int access (int *array, int index)
    { return array[index + offset]; }
  int i;
  /* ... */
  for (i = 0; i < size; i++)
    /* ... */ access (array, i) /* ... */
}

现在你明白了吧?嵌套函数在OCaml中是非常有用也是非常常用的。下面是真实代码中的嵌套函数的例子:

let read_whole_channel chan =
  let buf = Buffer.create 4096 in
  let rec loop () =
    let newline = input_line chan in
    Buffer.add_string buf newline;
    Buffer.add_char buf '\n';
    loop ()
  in
  try
    loop ()
  with
    End_of_file -> Buffer.contents buf;;

不用担心这段代码在干什么-它包含了很多我们还未涉及的概念。只要专注于名叫loop的只有一个 unit 参数的嵌套函数就可以了。你可以在函数read_whole_channel中来调用函数loop(),但是它在read_whole_channel函数外是没有定义的。嵌套函数可以使用主函数中定义的变量。(这里loop使用了局部变量bufchan)。

嵌套函数的格式和局部命名表达式相同。let name arguments = function-definition in.

通常我们如上面的例子中那样让函数定义缩进,如果你函数是递归的记住用let rec代替let,如上面的例子中那样。

模块和open

OCaml带有很多有趣的模块(含有用代码的库)。例如标准模块中有画图、与GUI小部件(widget)交互、处理大数、数据结构、POSIX系统调用等模块。这些库位于/usr/lib/ocaml/VERSION/ (当然是指在Unix系统下的情况)(译注:安装目录见仁见智)。 现在我们专注于一个比较简单的模块Graphics

模块Graphics安装在5个文件中(在我的系统上):

/usr/lib/ocaml/3.08/graphics.a
/usr/lib/ocaml/3.08/graphics.cma
/usr/lib/ocaml/3.08/graphics.cmi
/usr/lib/ocaml/3.08/graphics.cmxa
/usr/lib/ocaml/3.08/graphics.mli

现在我们只专注于graphics.mli文件.这是一个text文件,因而是可读的。首先注意文件名是graphics.mli而不是Graphics.mli。OCaml通常将文件名的第一个字母大写作为模块名。如果不知道这一点的话这有些容易使人混淆。

如果你想用Graphics中的函数,存在两种方法。一是在程序开头声明open Graphics;;。二是在所有函数调用前加上前缀,比如Graphics.open_graphopen 有点像Java中的import语句,不过更像Perl中的 use语句。(译注:或者是python的from blah import *

#load "graphics.cma";;

[Windows用户:必须使用一个定制的toplevel才可以使这个例子能够正常的在交互模式下使用。在命令行中键入命令 `ocamlmktop -o ocaml-graphics graphics.cma`(译注:这个问题ms在4.02.0中已经没有了)。]

下面几个例子将更好的说明。(下面两个例子画出不同的东西,你可以自己试一下)。注意第一个例子调用Aopen_graph,而第二个使用Graphics.open_graph

(* 用 ocamlc graphics.cma grtest1.ml -o grtest1 来编译这个程序*)

open Graphics;;
open_graph " 640x480";;
for i = 12 downto 1 do let radius = i * 20 in set_color (if i mod 2 = 0 then red else yellow); fill_circle 320 240 radius done;;
read_line ();;
(* 用 ocamlc graphics.cma grtest2.ml -o grtest2 来编译这个程序*) Random.self_init ();;
Graphics.open_graph " 640x480";;
let rec iterate r x_init i = if i = 1 then x_init else let x = iterate r x_init (i-1) in r *. x *. (1.0 -. x);;
for x = 0 to 639 do let r = 4.0 *. (float_of_int x) /. 640.0 in for i = 0 to 39 do let x_init = Random.float 1.0 in let x_final = iterate r x_init 500 in let y = int_of_float (x_final *. 480.) in Graphics.plot x y done done;;
read_line ();;

两个例子都用到了一些我们还没有涉及的特性:面向过程式的for循环,if-then-else和递归。我们将在以后介绍。不过我们应该观察这些程序而试着发现(1)它们的原理,(2)类型推导如何帮助我们消除bugs。

Pervasives模块

有一个模块我们无需使用"open"。这就是Pervasives模块(你可以阅读/usr/lib/ocaml/3.08/pervasives.mli)。所有Pervasives中的符号会被自动引入所有的OCaml程序中。

重命名模块

如果你想用Graphics模块中的符号,但是不想全部引入它们而又觉得每次使用前缀Graphics太麻烦,那怎么办呢? 你可以像下面这样重命名它们:

module Gr = Graphics;;
Gr.open_graph " 640x480";;
Gr.fill_circle 320 240 240;;
read_line ();;

实际上这在引入一个嵌套模块(模块可以被嵌套)而又不想每次键入完整路径名的时候非常有用。

使用;;;或两者都不用。

This section is not up to date with the English one

现在,我们将看看一个很重要的概念,什么时候你应该使用;;,什么时候你应该使用;,什么时候你都不用。这是一个很有意思的窍门,除非你能真正掌握这点。而且往往也会花费初学者很长的时间来掌握。

规则#1:你必须使用;;在代码的最顶端来分隔不同的语句。并且绝对不要在函数定义中或者其他的语句中使用。

首先看看以下的一段代码:

Random.self_init ();;
Graphics.open_graph " 640x480";;
let rec iterate r x_init i = if i = 1 then x_init else let x = iterate r x_init (i-1) in r *. x *. (1.0 -. x);;

我们有两个顶层代码和一个函数定义。(函数名为iterate),每一句都用;;分隔。

规则#2:有时候你可以省略掉;;。初学者可以不用考虑这个问题,而总是按照第一条规则使用;;。既然后面将要看到别人写的一些代码,你还是需要知道什么时候我们可以省略;;>。可以省略的地方是:

  • 关键字let之前。
  • 关键字open之前。
  • 关键字type之前。
  • 文件的最后。
  • 一些其他(非常少)Ocaml能够“猜出”是语句结尾而不是中间的地方。

下面是一些正确省略了所有;;的代码:

open Random                   (* ;; *)
open Graphics;;
self_init ();;
open_graph " 640x480" (* ;; *) let rec iterate r x_init i = if i = 1 then x_init else let x = iterate r x_init (i-1) in r *. x *. (1.0 -. x);;
for x = 0 to 639 do let r = 4.0 *. (float_of_int x) /. 640.0 in for i = 0 to 39 do let x_init = Random.float 1.0 in let x_final = iterate r x_init 500 in let y = int_of_float (x_final *. 480.) in Graphics.plot x y done done;;
read_line () (* ;; *)

规则#3和规则#4是关于单独的;。它与;;完全不同。单独的分号;被称作连接点(sequence point),具有在C,C++,Java和Perl中一样的用途。我打赌你不知道它表示“先执行这个位置之前的语句,执行完成之后继续之后的语句”。

规则#3:把let ... in看作一条语句,永远不要在它后面加上单独的;

规则#4:在所有代码块中其他的语句后面跟上一个单独的;,最后一个例外

上面示例中的for循环内部就是一个很好的例子。 请注意我们从来都不会在下面的代码中使用单独的;

for i = 0 to 39 do
  let x_init = Random.float 1.0 in
  let x_final = iterate r x_init 500 in
  let y = int_of_float (x_final *. 480.) in
  Graphics.plot x y
done

上面代码中唯一可能考虑加上;的地方是在Graphics.plot x y的后面,但是因为这是代码块中的最后一条语句,规则#4要求我们不要加在这里。

";"的注意事项

Brian Hurt更正了一些我对于";"的观点。

;是和+一样的运算符。当然只是概念上的,并不完全一样。+具有int -> int -> int类型 接受两个整型并返回一个整型(求和)。;的类型是unit -> 'b -> 'b 接受两个值并简单返回第二个。就像C中的,(逗号)运算符,你可以如同a + b + c + d一样写a ; b ; c ; d。 几乎任何东西都是表达式,这是OCaml中一个从未很好描述过的“脑筋急转弯”("mental leaps")。if/then/else是一个表达式。 a ; b是一个表达式。match foo with ...是一个表达式。下面的代码完全合法(并且都是作同样一件事):

let f x b y = if b then x+y else x+0
let f x b y = x + (if b then y else 0)
let f x b y = x + (match b with true -> y | false -> 0)
let f x b y = x + (let g z = function true -> z | false -> 0 in g y b)
let f x b y = x + (let _ = y + 3 in (); if b then y else 0)

特别注意最后一个,用了;作为运算符“联合”("join")两个语句。OCaml所有的函数可以表示成:

let name [parameters] = expression

OCaml关于表达式的定义只比C广一点。事实上,C有"statements"的概念,但是所有C的语句只是OCaml的表达式(和;运算符一起)。

;不同于+的一个地方是不能像函数一样引用+。比如,可以定义一个对一列整数求和的函数sum_list

let sum_list = List.fold_left ( + ) 0

实践出真知

在这一节中,我们将看看Lablgtk1.2中的真实代码片段(Lablgtk是一个从Ocaml到Unix本 地Gtk的图形接口库)。值得提示的是,这些片段包含了很多尚未讨论的概念,因此不要太拘泥 于细节,而只注目于整体结构,比如作者在哪里使用;;;open,他们是怎么缩进代 码的,是怎么用局部和全局命名表达式的。

自然,你还需要一些提示:

  • ?foo~foo 在OCaml中分别表示函数的可选和命名参数。这个特性在C衍生的语言 中未必有对应的概念,但是Perl,Python和Smalltalk都允许函数忽略某些参数,或者 以其他顺序传入参数。
  • foo#bar 是调用对象foobar方法。这和C++的foo->bar,Java的foo.bar, Perl的$foo->bar类似。

第一个例子:这里作者open了一些标准库(由于下一个关键字是open或者let,;;可以省略)。 他先创建了一个叫file_dialog的函数,并在里面用let sel = ... in定义了命名表达式sel。 然后他用sel调用了几个函数。

(* 第一个例子 *)
open StdLabels
open GMain

let file_dialog ~title ~callback ?filename () =
  let sel =
    GWindow.file_selection ~title ~modal:true ?filename () in
  sel#cancel_button#connect#clicked ~callback:sel#destroy;
  sel#ok_button#connect#clicked ~callback:do_ok;
  sel#show ()

第二个例子不过是一长串全局变量。注意到作者用规则#2省略了每个;;

(* 第二个例子 *)
let window = GWindow.window ~width:500 ~height:300 ~title:"editor" ()
let vbox = GPack.vbox ~packing:window#add ()

let menubar = GMenu.menu_bar ~packing:vbox#pack ()
let factory = new GMenu.factory menubar
let accel_group = factory#accel_group
let file_menu = factory#add_submenu "File"
let edit_menu = factory#add_submenu "Edit"

let hbox = GPack.hbox ~packing:vbox#add ()
let editor = new editor ~packing:hbox#add ()
let scrollbar = GRange.scrollbar `VERTICAL ~packing:hbox#pack ()

第三个例子中,作者导入GdkKeysyms模块中的所有符号。但是有个不一样的let绑定: let _ = expression。它对expression进行求值,产生相应的副作用,并且丢弃计算结果。 在这个例子中,运行的是Main.main(),它是Gtk的事件循环,它的副作用是弹出一个窗口并 运行整个应用程序。显然的是,该函数的返回结果是无关紧要的(或许是unit,但是我没在意), 再说,事件循环不会在应用程序结束前退出。

注意到在这个例子中,我们有一长串的过程指令,这是很典型的指令式编程。

(* 第三个例子 *)
open GdkKeysyms

let () =
  window#connect#destroy ~callback:Main.quit;
  let factory = new GMenu.factory file_menu ~accel_group in
  factory#add_item "Open..." ~key:_O ~callback:editor#open_file;
  factory#add_item "Save" ~key:_S ~callback:editor#save_file;
  factory#add_item "Save as..." ~callback:editor#save_dialog;
  factory#add_separator ();
  factory#add_item "Quit" ~key:_Q ~callback:window#destroy;
  let factory = new GMenu.factory edit_menu ~accel_group in
  factory#add_item "Copy" ~key:_C ~callback:editor#text#copy_clipboard;
  factory#add_item "Cut" ~key:_X ~callback:editor#text#cut_clipboard;
  factory#add_item "Paste" ~key:_V ~callback:editor#text#paste_clipboard;
  factory#add_separator ();
  factory#add_check_item "Word wrap" ~active:false
    ~callback:editor#text#set_word_wrap;
  factory#add_check_item "Read only" ~active:false
    ~callback:(fun b -> editor#text#set_editable (not b));
  window#add_accel_group accel_group;
  editor#text#event#connect#button_press
    ~callback:(fun ev ->
      let button = GdkEvent.Button.button ev in
      if button = 3 then begin
        file_menu#popup ~button ~time:(GdkEvent.Button.time ev); true
      end else false);
  editor#text#set_vadjustment scrollbar#adjustment;
  window#show ();
  Main.main ()