Skip to content

模式匹配和for循环的相似之处

函数式新手经常遇到的一个问题是似乎我不能写for循环了。这篇博客会带你看看不同的for循环对应到函数式是怎么样的。

遍历集合

你们应该都见过这种代码:

python
for x in xs:
    update_state(state, x)
return state;

这种经常是基于迭代器的,我们把它去糖一下,这里用Java:

java
for (Foo x: collectionOfFoo)
    do_something_with(state, x);
return state;

// into
var iter = collectionOfFoo.iterator();
while (iter.hasNext()) {
    var x = iter.peek();
    do_something_with(state, x);
    iter.next();
}
return state;

没那么好看了吧!

注意到每次循环我们都在更新状态,每次循环迭代器要么为空要么有下一个值,在循环里面我们会“解构”迭代器(把头拔下来)直到它变成空的为止。我们在Haskell里也能干一模一样的事:

haskell
-- 空的情况
traverse_collection []     state     = state

-- 执行循环逻辑并“进入下一次循环”
traverse_collection iter state       = traverse_collection (tail iter) (do_something_with state (head iter))

-- 而用模式匹配:
-- traverse_collection (x:xs) state  = traverse_collection xs (do_something_with state x)

注意到do_something_with 一直在那,但进入下一次循环的逻辑隐含在递归里。

Haskell版里用的模式匹配还包含了“把头拔下来”,因为大部分Haskell的数据结构都是递归的。在Java版里没写这个逻辑,写了会更长。

这一整套逻辑被抽象成了foldl,Java增强形for循环要求你是Iterable,同理,foldl要求Foldable

执行某逻辑X次

接下来我们再看看“执行某逻辑X次”的循环,这种一般函数式新手最怀念(可能是因为他们之前学到所有循环都基于这一种),不过接下来我们会发现命令式地写这种逻辑很啰嗦而且没有糖。

java
int fib(int n) {
    int x = 1;
    int y = 1;
    for (int i = 0; i < n - 1; i++) {
        int tmp = y;
        y += x;
        x = tmp;
    }
    return y;
}

而在函数式的版本中我们可以用尾递归实现一模一样的逻辑,而且还短很多。

haskell
fib n = fib' 0 1 n
    where
        fib' _ y 0 = y
        fib' x y n' = fib' y (x + y) (n' - 1)

带下标地遍历集合

java
for (int i = 0; i < xs.length; i++)
    do_something_with(i, xs[i], state)

在Haskell版中我们可以复用之前的traverse_collection

haskell
-- 定义新的需要下标的do_something_with

f xs state = traverse_collection (zip xs [1..]) state

注意到命令式的脱糖后需要更多的代码,但函数式的代码量没有增多(当然这里用foldl更方便)。

你可以用stream API来获得相似的体验,但他自己就是借鉴自函数式的 😃

例:根据空格分割字符串

Java:

java
List<String> split(String input) {
    var state = new ArrayList<String>();
    state.add("");
    for (char c: input.toCharArray()) {
        if (c == ' ') {
            state.add("");
        } else {
            var index = state.size() - 1;
            state.set(index, state.get(index) + c);
        }
    }
    return state;
}

// split("Hello Functional Programming");
// ==> [Hello, Functional, Programming]

Haskell:

haskell
split :: String -> [String]
split str = reverse (split' str [""])
    where
        split' :: String -> [String] -> [String]
        split' [] ss = ss
        split' (x:xs) ss@(h:t)
            | x == ' ' = split' xs ("" : ss)
            | otherwise = split' xs ((h ++ [x]):t)

-- λ ->  split "Hello Functional Programming"
-- ["Hello", "Functional", "Programming"]
不用reverse百分百对应java逻辑的版本
haskell
split :: String -> [String]
split str = split' str [""]
    where
        split' :: String -> [String] -> [String]
        split' [] ss = ss
        split' (x:xs) ss
            | x == ' ' = split' xs (ss `snoc` "")
            | otherwise = split' xs (changeLast ss (last ss `snoc` x))

        -- 这两个都是标准库里的,不过我放在这以便查看
        snoc xs x = xs ++ [x]
        changeLast xs x = init xs ++ [x]

-- λ ->  split "Hello Functional Programming"
-- ["Hello", "Functional", "Programming"]