0

我正在学习大学的数据结构课程,我正在努力理解它。我对 C 中的递归和循环有一个大致的了解,但我无法理解如何在 erlang 中实现循环。例如,我在 c 中的一个程序中有以下循环:

int ii;
double cos_lat[8];    
for(ii = 2, cos_lat[1] = cos(lat); ii <= 7;; ii++)
{
 cos_lat[ii] = cos_lat[1] * cos_lat[ii-1]
}

而且我对如何在 erlang 中实现它一无所知。我知道我必须创建自己的循环定义,例如:

for(Max, Max, F) -> [F(Max)];
for(I, Max, F) -> [F(I)|for(I+1, Max, F)].

(取自实用的二郎)

然后按如下方式调用它:

for(2,7,fun(ii) -> // this is where i get stuck.

我认为我必须在数组(例如列表)上使用不同的数据结构,但我不确定如何访问列表。

感谢您的回复。只是一个更新(14/8)。我试图对代码进行递归,如下所示:

-module(cos1).
-export([cos_lat/1]).
cos_lat(Base, Iterator, Prev) -> [Base*Prev|cos_lat(Base,Iterator+1,Base*Prev)];
cos_lat(Base, 7, Prev) -> [].

并将其称为如下:

cos1:cost_lat(cos(lat),2,cos(lat).

但它就是不想工作!Erlang 确实是一门令人困惑的语言。

4

3 回答 3

4

让我们看一下“for”循环结构。

它包含初始化块条件块更新块。循环有一个状态(一开始实际上是在初始化块中定义的)。 无论您要处理哪种数据结构,因为循环的结构通常是相同的。循环状态封装了实际的数据结构。

此外,您可能会看到条件块更新块通常是循环状态的函数

使用这些信息,让我们创建通用循环函数:

-module( loops ).
-export( [ for/3 ] ).

for( State, ConditionFunc, LoopFunc ) ->
        case ConditionFunc( State ) of
                true ->
                        NewState = LoopFunc( State ),
                        % next iteration
                        for( NewState, ConditionFunc, LoopFunc );
                false ->
                        % terminate and return
                        State
        end.

例如,让我们使用我们的函数创建一个从 1 到 10 的数字列表(在 eralng shell 中):

1> c(loops). 
{ok,loops}
2> 
2> ConditionFunc = 
2> fun( { 0, List } ) -> false;
2> ( _ ) -> true 
2> end.
#Fun<erl_eval.6.82930912>
3> 
3> LoopFunc = 
3> fun( { N, List } ) -> { N - 1, [ N | List ] } end. 
#Fun<erl_eval.6.82930912>
4> 
4> { _, Result } = loops:for( { 10, [] }, ConditionFunc, LoopFunc ).
{0,[1,2,3,4,5,6,7,8,9,10]}
5> 
5> Result.
[1,2,3,4,5,6,7,8,9,10]
6> 

这不是创建数字序列的最佳方法,仅用于说明。总是你可能会发现更优雅的递归解决方案与循环解决方案。对于我们的示例,下一个解决方案更可取:

seq( A, B ) ->
        my_seq( A - 1, B, [] ).

my_seq( A, A, List ) ->
        List;
my_seq( A, B, List ) ->
        my_seq( A, B - 1, [ B | List ] ).

在外壳中:

1> loops:seq( 1, 10 ).
[1,2,3,4,5,6,7,8,9,10]

或者只是使用标准库列表中的函数:)

2> lists:seq( 1, 10 ).
[1,2,3,4,5,6,7,8,9,10]
于 2012-08-10T07:43:40.437 回答
0

在此刻:

for(2,7,fun(ii) -> // this is where i get stuck.

fun正在使用索引调用,但您实际上对索引并不感兴趣 - 您需要在上一次迭代中添加到列表中的值。您可以将其作为第四个参数传递给您的for函数;第一次调用它时,您需要使用 播种它cos(lat),并在每个递归步骤中使用新值。

于 2012-08-10T13:18:29.300 回答
0

以防万一有人感兴趣。我解决了如下问题:

    -module(cos_lat).
    -export([cos_lat/3]).

     cos_lat(_Base, _Iterator, _Prev) ->
     cos_lat(_Base, _Iterator, _Prev, []).

     cos_lat(_Base,0, _Prev, _Co) ->
     lists:reverse(_Co);

     cos_lat(Base, Iterator, Prev, Co) -> 
     cos_lat(Base, Iterator-1, Base*Prev, [Prev*Base|Co]).

因此,要调用它,您将输入:

     cos_lat:cos_lat(math:cos(lat),7,math:cos(lat)).
于 2012-08-15T07:27:56.923 回答