首页 关于 友链

函数式编程学习笔记

swwind#函数式#haskell

前言

近年来,函数式编程正在越来越受到欢迎。

不仅最古老的函数式语言 Lisp 重获青春,而且新的函数式语言层出不穷,比如 Erlang、clojure、Scala、F# 等等。目前最当红的 Python、Ruby、Javascript,对函数式编程的支持都很强,就连老牌的面向对象的 Java、面向过程的 PHP,都忙不迭地加入对匿名函数的支持。越来越多的迹象表明,函数式编程已经不再是学术界的最爱,开始大踏步地在业界投入实用。

——阮一峰培训班

本人建议面向过程/面向对象写惯的同学从 JavaScript 开始理解函数式编程会比较好。

{"看得懂语法至少不会使你马上就弃坑"}

预备知识

我们假设所有人都会 JavaScript。

柯里化(Currying)

const add = (a, b) => a + b; // 普通的一个 add 函数
const add = (a) => (b) => a + b; // 这样就是柯里化

这样有什么用呢?

打个比方,如果我们要将一个数组里面所有元素加 1,你就可以这样写

let arr = [1, 2, 3];

arr.map(add(1)); // [2, 3, 4]

add(1) 将会返回一个函数 b => 1 + b,配合 map 使用即可。

如何自动将一个函数柯里化?

// 模板
const curry = (fn) => {
  const _c = (restNum, argsList) =>
    restNum === 0
      ? fn(...argsList)
      : (...args) => _c(restNum - args.length, [...argsList, ...args]);

  return _c(fn.length, []);
};

// 使用方法

const add = curry((a, b) => a + b);
add(1)(2); // 3
add(1, 2); // 3

合并函数

// 合并两个函数,从左到右执行
const pipe = (f, g) => (x) => g(f(x));
// 合并多个函数,从左到右执行
const compose = (...args) => args.reduce(pipe, args.shift());

函数式与非函数式的比较

筛选出一个数组中所有的数字成员。

普通写法

const filter_list = (arr) => arr.filter((x) => typeof x === "number");

函数式

const filter = (f) => (a) => a.filter(f);
const is = (t) => (x) => typeof x === t;

const filter_list = filter(is("number"));

可见函数式代码更加优美并且更加接近自然语言。

Ramda

Ramda 是一个提供 JavaScript 函数式编程的函数库。 想要了解的请自行 google,这里不做展开。

Haskell 入门

Haskell 第一眼给人的印象就是:懵逼。

他和普通的非函数式语言完全不同。

学过的人知道他在写什么。

没学过的人连他写的东西怎么运行都不知道(特别是几个奇怪的运算符)。

现在我们尝试学习 Haskell。

输出

输出字符串

-- 定义主函数(可省略,因为有自动类型推断)
main :: IO ()
-- 实现主函数
-- 输出 hello world
main = putStrLn "hello world"

putStrLn 函数用来输出字符串。

调用函数不需要括号,所有函数全部支持柯里化。

输出其他任何东西

main :: IO ()
main = do
  print "hello world"
  print [1 .. 3]
-- 输出:
-- hello world
-- [1,2,3]

do 关键字的作用是执行多条语句(多个函数?),使用缩进来判断是否在同一个块内。

函数的声明

你只需要把每个参数的类型和返回值的类型用 -> 连接起来就行了。

如果没有返回值,使用 ()(Unit /ˈjuːnɪt/) 表示 void

函数将自动返回最后一次运行的表达式的计算结果。

add :: Int -> Int -> Int
add a b = a + b

调用函数

你不需要使用括号和逗号,直接在函数名后面输入参数即可。

由于函数是从左到右执行的,所以有时候需要括号来调整优先级。

main :: IO ()
main = print (add 1 2)

如果你觉得用括号太愚蠢了,就用 $ 运算符。

main = print $ add 1 2
-- 输出 3

$ 的定义:

infixr 0 $ -- 优先级最低,从右至左执行
($) :: (a -> b) -> a -> b
f $ x = f x

自带柯里化

-- 其实运算符也是个函数
add = (+)

main :: IO ()
main = print $ map (add 5) [1, 2, 3]
-- 输出:[6,7,8]

where 关键字

全部定义在外面

isOdd :: Int -> Bool
isOdd x = x `mod` 2 == 1

main :: IO ()
main = print $ isOdd 4 -- False

使用 where 关键字

main :: IO ()
main = print $ isOdd 4
  where
    isOdd x = x `mod` 2 == 1

例题和题解

这一部分请自行观察代码理解

List 相关。

main = do
  print $ 1 : [4, 5] -- [1,4,5] :: [Int]
  print $ 'f' : "oobar" -- "foobar" :: String (= [Char])
  print $ "foo" ++ "bar" -- "foobar"

给定三个数,判断是否为三角形的三条边。

isTriangle :: Int -> Int -> Int -> Bool
isTriangle a b c
| a + b <= c = False
| a + c <= b = False
| b + c <= a = False
| otherwise  = True

计算阶乘。

Integer 是高精度整数。

frac :: Integer -> Integer
frac 0 = 1
frac n = n * frac (n - 1)

main :: IO ()
main = print $ frac 100

计算斐波那契数列。

先来一个 \( O(F_n) \) 暴力。

fib 0 = 0
fib 1 = 1
fib n = fib (n - 1) + fib (n - 2)

然后学一个优美一点的写法。

fib a b = a : fib b (a + b)
main = print $ take 20 $ fib 1 1

我们来分析一下这种写法。

首先 Haskell 进行自动类型推断,根据 : 和下文给出的参数可以推出,fib 应该返回 [Integer]

由于 Haskell 有惰性求值,所以不用担心调用的时候会无限递归。

take 20 用于取 list 的前 20 项。

takeWhile (<= 20) 用于取 list 前面所有满足 \( \le 20 \) 的元素,遇到第一个不满足的即返回。

其实都是字面意思

结尾

作者的忠告:

永远不要想着用 Haskell 写 OI 题,这是在浪费时间。

想要训练和深入理解,可以去 codewars 上做题练习。

Copyright © 2017-2025 swwind. All rights reserved
Except where otherwise noted, content on this blog is licensed under CC-BY 2.0