8

我尝试使用 2 种方法在 Oz 中打印出斐波那契数列:使用 Emac 作为编辑器的函数和过程。程序在这里:

declare 
fun {Fibo N} 
   case N of 
      1 then 1 
   [] 2 then 1
[] M then {Fibo (M-1)} + {Fibo (M-2)} 
   end 
end 
declare
proc {Loop K}
   if K ==1 then  {Browse K}
   else
      {Loop K-1}
       {Browse {Fibo K}}
   end
end
{Loop 10}

和功能:

declare 
fun {Fibo N} 
   case N of 
      1 then 1 
   [] 2 then 1
[] M then {Fibo (M-1)} + {Fibo (M-2)} 
   end 
end
declare
fun {Loo L}
   if L ==1 then  {Browse L}
   else
      {Loo L-1}
       {Browse {Fibo L}}
   end
end
{Loo 10}

问题是唯一的程序“循环”有效。结果是:

1
1
2
3
5
8
13
21
34
55

函数“Loo”没有,它会抛出一些难以理解的错误:

%********************** static analysis error *******************
%**
%** illegal arity in application
%**
%** Arity found:          1
%** Expected:             2
%** Application (names):  {Loo _}
%** Application (values): {<P/2> _<optimized>}
%** in file "Oz", line 13, column 6

%********************** static analysis error *******************
%**
%** illegal arity in application
%**
%** Arity found:          1
%** Expected:             2
%** Application (names):  {Loo _}
%** Application (values): {<P/2> 10}
%** in file "Oz", line 17, column 0
%** ------------------ rejected (2 errors)

我仍然不知道为什么。我认为函数和过程在 OZ 中具有类似的效果。

4

2 回答 2

6

必须使用函数调用语法调用函数:

_ = {Loo 10}

或者使用附加参数来接收值:

{Loo 10 _}

_发音为“don't care”,表示不需要变量的值。

此外,函数必须通过将表达式作为每个分支的最后一部分来返回值。所以你的固定Loo功能看起来像这样:

fun {Loo L}
   if L == 1 then
      {Browse L}
      unit
   else
      _ = {Loo L-1}
      {Browse {Fibo L}}
      unit
   end
end
_ = {Loo 10}

但是,如果您没有任何有趣的东西要返回,那么使用这样的循环函数没有多大意义。也许您真正想要的是构建一个列表并将其作为结果返回

于 2011-04-29T20:29:44.173 回答
3

Loo您在第 13 行的定义中有错字。

您正在调用Loop,它不存在。我想你应该打电话Loo

更新:您所看到的是由于函数和过程之间的差异;函数总是返回值,过程不会。Loo您正在为( )提供一个参数K-1,但Loo需要两个参数;一个输入变量和一个捕获返回值的变量。奥兹告诉你这一点,你应用了错误的数量Loo(当你应该应用两个参数(二进制)时,你应用了一个参数(一元))。

这意味着您还必须将返回值分配给变量。执行以下操作之一:

  1. A = {Loo K-1}
  2. {Loo K-1 A}

A是将分配返回值的变量。对于您不关心函数返回什么的情况,一个常见的约定是_用作返回变量名。

于 2011-04-29T09:29:12.217 回答