函数(functions)


在 Lisp 中,函数分两种:有名函数和匿名函数(lambda函数)。


有名函数 defun

有名函数的标准定义格式为:

(defun <name> (list of arguments)
"docstring"
(function body))

在函数中,返回值是函数主体中的最后一个表达式的结果。与大部分语言不同的是,lisp 中的函数没有 "return xx" 这样的语句用来声明返回值。例如:

(defun hello-world ()
;;
(print "hello world!"))

调用 hello-world 函数:

(hello-world)
;; "hello world!" <-- output
;; "hello world!" <-- a string is returned.

参数

  • 必需参数

    与大部分语言相同,函数中需要定义一些必需参数:

    (defun hello (name)
    "Say hello to `name'."
    (format t "hello ~a !~&" name))
    ;; HELLO

    调用 hello

    (hello "me")
    ;; hello me ! <-- this is printed by `format`
    ;; NIL <-- return value: `format t` prints a string to standard output and returns nil.
  • 可选参数:&optional

    可选参数定义在 &optional 这个关键词后面,且这些参数是有序的。如:

    (defun hello (name &optional age gender) ...)

    调用时需要这样调用:

    (hello "me") ;; a value for the required argument, zero optional arguments
    (hello "me" "7") ;; a value for age
    (hello "me" 7 :h) ;; a value for age and gender
  • 关键词参数:&key

    通常情况下,要记住函数中参数的顺序很不方便,所以就引入了关键词参数 &key <name>。通过 :name <value> 这样的方式来传递参数。如果关键词 name 的值没有设置的话,默认为 nil

    (defun hello (name &key happy)
    "If `happy' is `t', print a smiley"
    (format t "hello ~a " name)
    (when happy
    (format t ":)~&"))

    所以,我们可以这样调用 hello

    (hello "me")
    (hello "me" :happy t)
    (hello "me" :happy nil) ;; useless, equivalent to (hello "me")

    但是 (hello "me" :happy) 是非法的。

    注:在关键词参数中,如果调用其他的关键词时,会报错,这是可以通过 &allow-other-keys 来修复。

    (defun hello (name &key happy)
    (format t "hello ~a~&" name)) (hello "me" :lisper t)
    ;; => Error: unknow keyword argument
    (defun hello (name &key happy &allow-other-keys)
    (format t "hello ~a~&" name)) (hello "me" :lisper t)
    ;; hello me
  • 默认参数

    有时,需要将函数的某个参数设置一个默认值,这样 hello 就可以这样定义

    (defun hello (name &key (happy t)) ...)

    这样,调用 hellohappy 的值默认就是 t 了。

    • 不定参数:&rest

    在不确定参数的个数时,可以使用 &rest <variable> 这样的方式来定义,其中 &rest 后面的参数会当作一个 list 来处理。

    (defun mean (x &rest numbers)
    (/ (apply #'+ x numbers)
    (1+ (length numbers))))
    (mean 1)
    (mean 1 2)
    (mean 1 2 3 4 5)

返回值

在 Lisp 中,函数的返回值就是函数主体中最后一个表达式执行的结果。也有非标准的 return-from <function name> <value> 这样的语句,但是大部分情况下用不到。同时,Common Lisp 支持返回多个值。

返回多个值有三个关键词:valuesmultiple-value-bindnth-value

返回多个值不是将所有的结果都放入一个元组或列表中,这是很常见的概念混淆。

  • values

    (defun foo (a b c)
    a)
    (foo :a :b :c)
    ;; :A
    (defvar *res* (foo :a :b :c))
    ;; :A
    (defun foo (a b c)
    (values a b c))
    (foo :a :b :c)
    ;; :A
    ;; :B
    ;; :C
    (setf *res* (foo :a :b :c))
    ;; :A

    从上面代码可以看出,如果 foo 返回的是一个列表的话,那么 res 的值将会是 (:a :b :c) 这样一个列表,而不是 :A 这个值。

  • multiple-value-list

    该关键词的作用是将返回的多个值组合成一个列表

    (multiple-value-list (values 1 2 3))
    ;; (1 2 3)
  • values-list

    values-listmultiple-value-list 相反,它返回的是列表中的每个元素

    (values-list '(1 2 3))
    ;; 1
    ;; 2
    ;; 3

匿名函数 lambda

匿名函数的声明如下:

(lambda (x) (print x))

匿名函数的调用:

((lambda (x) (print x)) "hello")
;; hello
  • funcallapply

    (funcall #'+ 1 2)
    (apply #'+ '(1 2))
  • 返回函数的函数

    (defun adder (n)
    (lambda (x) (+ x n)))
    ;; ADDER (adder 5)
    ;; #<CLOSURE (LAMBDA (X) :IN ADDER) {100994ACDB}> (funcall (adder 5) 3)
    ;; 8

    上面示例中,(adder 5) 返回的是一个匿名函数。但是需要使用 funcall 关键词来调用,不能想正常函数调用来调用。

    ((adder 3) 5)
    In: (ADDER 3) 5
    ((ADDER 3) 5)
    Error: Illegal

    Common Lisp 中提供了两个函数来查看变量或函数是否赋值/绑定:boundpfboundp

    ;; The symbol foo is bound to nothing:
    CL-USER> (boundp 'foo)
    NIL
    CL-USER> (fboundp 'foo)
    NIL
    ;; We create a variable:
    CL-USER> (defparameter foo 42)
    FOO
    * foo
    42
    ;; Now foo is "bound":
    CL-USER> (boundp 'foo)
    T
    ;; but still not as a function:
    CL-USER> (fboundp 'foo)
    NIL
    ;; So let's define a function:
    CL-USER> (defun foo (x) (* x x))
    FOO
    ;; Now the symbol foo is bound as a function too:
    CL-USER> (fboundp 'foo)
    T
    ;; Get the function:
    CL-USER> (function foo)
    #<FUNCTION FOO>
    ;; and the shorthand notation:
    * #'foo
    #<FUNCTION FOO>
    ;; We call it:
    (funcall (function adder) 5)
    #<CLOSURE (LAMBDA (X) :IN ADDER) {100991761B}>
    ;; and call the lambda:
    (funcall (funcall (function adder) 5) 3)
    8

    注:在 Lisp 中,变量名和函数名可以相同,因为 Common Lisp 中变量和函数并不是存储在一起的,而是分开存储的。

闭包(Closure)

闭包,就是让一个函数可以使用一个 词法绑定(lexcial bindings)On Lisp 中的定义为:函数和一组变量的绑定的组合(a combination of a function and a set of variable bindings)。 Let Over Lambda中对闭包的解读为:一个保存了词法的环境(a saved lexical environment)。 可以将闭包理解为 C 语言中的 结构体(struct)或者面向对象语言(Java/C++)中的 类(class)

(let ((limit 3)
(counter -1))
(defun my-counter ()
(if (< counter limit)
(incf counter)
(setf counter 0)))) (my-counter)
0
(my-counter)
1
(my-counter)
2
(my-counter)
3
(my-counter)
0

类似的

(defun repeater (n)
(let ((counter -1))
(lambda ()
(if (< counter n)
(incf counter)
(setf counter 0))))) (defparameter *my-repeater* (repeater 3))
;; *MY-REPEATER*
(funcall *my-repeater*)
0
(funcall *my-repeater*)
1
(funcall *my-repeater*)
2
(funcall *my-repeater*)
3
(funcall *my-repeater*)
0

Lisp-02: 函数的更多相关文章

  1. JavaScript进阶系列02,函数作为参数以及在数组中的应用

    有时候,把函数作为参数可以让代码更简洁. var calculator = { calculate: function(x, y, fn) { return fn(x, y); } }; var su ...

  2. Python函数02/函数的动态参数/函数的注释/名称空间/函数的嵌套/global以及nolocal的用法

    Python函数02/函数的动态参数/函数的注释/名称空间/函数的嵌套/global以及nolocal的用法 目录 Python函数02/函数的动态参数/函数的注释/名称空间/函数的嵌套/global ...

  3. JavaScript基础精华02(函数声明,arguments对象,匿名函数,JS面向对象基础)

    函数声明 JavaScript中声明函数的方式:(无需声明返回值类型) function add(i1, i2) {             return i1 + i2;//如果不写return返回 ...

  4. 02函数-05-generator(ES6)

    generator(生成器)是ES6标准引入的新的数据类型. generator看上去像一个函数,但可以返回多次,除了return语句,还可以用yield返回多次.定义方式如下: function* ...

  5. 02函数-04-箭头函数(ES6)

    ES6新增的函数:Arrow Function,定义方式就是一个箭头 箭头函数相当于匿名函数,并且简化了函数定义,和匿名函数最大的区别在于其内部的this不再"乱跑",而是由上下文 ...

  6. e lisp 自定义函数

    自定义函数 (defun multi-by-seven (number) "multi number by seven" (interactive "p") ( ...

  7. JavaScript语言精粹 笔记02 函数

    函数函数对象函数字面量调用参数返回异常给类型增加方法递归作用域闭包回调模块级联套用记忆   函数 1 函数对象 在JS中函数就是对象.对象是“名/值”对的集合并拥有一个连接到原型对象的隐藏连接.对象字 ...

  8. JS 02 函数

    函数 一.创建函数 1.function 函数名( 形参列表 ){ 函数体 } 2.var 函数名 = function( 形参列表 ) { 函数体 } 3.var 函数名 = new Functio ...

  9. Scala基础篇-02函数与代码块

    1.block 代码块也是表达式,其最终求得的值是最后一个表达式的值. {exp1;exp2} { exp1 exp2 } 2.function def funtionName(param:Param ...

  10. 4.Scala语法02 - 函数

随机推荐

  1. 实验一 Linux系统与应用准备

    实验一 Linux系统与应用准备 项目 内容 作业归属 班级课程 作业要求 课程作业要求 学号-姓名 17041419-刘金林 作业学习目标 1.学习博客园软件开发者学习社区使用技巧和经验:2.学习M ...

  2. ant tree 展开key的集合

    这次有个功能 ant的tree 展开 点击子节点 新增节点之后 数据能够照常展开 有几种方法 我能想到的 因为ant 有个expanded 只要设置为true就能展开了,但是这边有个陷阱,就是仅仅设置 ...

  3. 写react项目要注意的事项

    1,className一定是大写字母开头,例如:App-logo,App,App-header. 2,有关react元素的更新,唯一办法是创建新元素,然后重新将其传入ReactDOM.render() ...

  4. 基于springcloud搭建项目-Ribbon篇(三)

    这篇文章主要是介绍一下ribbon的用法,我们都知道ribbon是负载均衡,但是却不知道他是怎么样的负载均衡,怎么用,能干嘛? ● 其实,简单的说,Spring Cloud Ribbon是基于Netf ...

  5. springmvc与swagger2

    首先呢我们导入相关的jar包文件 为了方便copy我copy一份 <!-- 导入java ee jar 包 -->        <dependency>           ...

  6. 一次生产环境搭建11g RAC的记录

    一.使用惠普3par工具配置共享存储 该部分可由惠普工作人员协助配置,只需将需求告知即可.如果想自己配置,惠普厂商会发送相关的软件工具以及操作手册给用户. 用putty登陆共享存储,使用showpd ...

  7. R|生存分析 - KM曲线 ,值得拥有姓名和颜值

    本文首发于“生信补给站”:https://mp.weixin.qq.com/s/lpkWwrLNtkLH8QA75X5STw 生存分析作为分析疾病/癌症预后的出镜频率超高的分析手段,而其结果展示的KM ...

  8. Spring源码阅读笔记05:自定义xml标签解析

    在上篇文章中,提到了在Spring中存在默认标签与自定义标签两种,并且详细分析了默认标签的解析,本文就来分析自定义标签的解析,像Spring中的AOP就是通过自定义标签来进行配置的,这里也是为后面学习 ...

  9. DOTNET CORE源码分析之ServiceDescriptor

    ServiceDescriptor在.net core中的作用就是DI中注入服务元素的描述.每一个元素核心内容部分包括需要注入的服务元素的类型ServiceType,它对应的接口(如果有的话)Impl ...

  10. 支付宝小程序获取 user_id(openid) ThinkPHP版

    支付宝小程序获取 user_id(openid) ThinkPHP版 近期支付宝小程序个人公测了,就想着玩一下,没想到就获取用户唯一标识都这么麻烦,微信的openid的话Get请求一下就完事了,支付宝 ...