sml中的双射函数

时间:2013-11-02 21:07:59

标签: sml smlnj ml

我想定义一个取整数n的函数并返回一个整数n *,使得n和n *在从1到n的同一整数集中,并且该函数必须是双射的。

我尝试了以下

fun bij(n) =
      let 
        val ls = zip(upto (1, n), List.rev(upto (1, n))) 
        val Tw_2 = fn(a, b) => b
      in Tw_2(List.last(ls, n-1)) end;

但不幸的是,它为我所有的n值返回1。我真的被困在这里了。 谁能给我一些关于如何实现这个的想法?

bij的行为必须类似于

bij(1) = 3
bij(2) = 2
bij(3) = 1

1 个答案:

答案 0 :(得分:1)

如果我理解你的问题,一个简单的解决方案就是:

fun bij(n, i) = n + 1 - i;

可以用下表表示

i         | 1    2    3  ... n-2  n-1  n
bij(n, i) | n  n-1  n-2  ...   3    2  1

对于1n之间的数字有效。直觉上,(正)数i i步骤“0”,我们将其映射到i的数字(实际为i - 1 )“步骤”n左侧。

也许您想通过列表明确构建上表?

fun upto(m, n) = if n < m then [] else m :: upto(m+1, n);

fun table n = ListPair.zip (upto(1, n), List.rev (upto(1, n)));

示例:

> table 5;
val it = [(1, 5), (2, 4), (3, 3), (4, 2), (5, 1)]: (int * int) list

然后获取i对列表xs,您可以使用

List.nth (xs, i-1)

全部放在一起

fun bij(n, i) =
      let
        val table = ListPair.zip (upto(1, n), List.rev (upto(1, n)));
        fun snd(x, y) = y;
      in snd(List.nth (table, i-1)) end;

除了更复杂的方式外,它与初始函数相同。