#lang在模块文件的开始处是一种module形式的简写。就像'
是quote的简写。但是#lang不能在repl里使用,一部分原因是它必须使用文件end-of-file接受,也因为简写表达是#lang依赖闭合的文件。
6.2.1module 形式
module定义,在repl和文件里都有效
(module name-id initial-module-path
decl ...)
name-id是模块名,initial-module-path是默认导入,而且每一个decl都市一个导入,导出,定义,或者表达式。当在文件里的时候,name-id一般和包含它的文件同名,但是不包括路径或者文件后缀。当通过文件路径引用模块时,name-id会被忽略。
initial-module-path是必要的,因为即使是require形式也必须导入才能使用。换句话说,initial-modult-path导入了方法体里的可用语法。最常用的模块是racket,它提供了包括require,define,provide和文档里的大多数绑定。另外一个常用模块时racket/base,也提供了大多数需要的函数和语法。
在repl里面可以用模块语法直接定义一个模块,而且只要使用模块名就可以引用该模块。
定义一个模块不会马上致谢模块的第一个和表达式。当模块被引用的时候将触发执行。但是它只会在第一次引用时(require),才会执行。
6.2.2#lang简写
lang简写没有特别的语法,因为语法被#lang的语句定义。比如这样:
#lang racket
decl ...
它和下面等价
(module name racket
decl...)
上面形式的方法名可以通过文件名来推断。
6.2.3子模块
模块能内嵌在模块里面,内嵌的模块定义了一个子模块。子模块在模块里面可以用名字直接引用。
运行一个模块不会直接运行子模块,它们会独立运行。
子模块也可以内嵌子模块。其它模块可以通过子模块路径来引用自模块。
(module* name-id inital-module-path-or-#f
decl ...)
它和module不同的地方在于
-
module无法访问包含它的模块的绑定和环境。
-
module可以访问包含它的模块的绑定,通过require,但是包含环境无法导入子模块。
此外,module形式可以定义#f代替initial-modult-path,那么子模块可以使用封闭环境的任何东西,包括没通过provide导出的。
一个应用module* #f的应用是在模块里默认不导出子模块,除非你指定导出子模块。代码类似于(require (submod "rake.rkt" extras))
6.2.4主要和测试子模块
运行一个模块并不会运行内部的module*定义的子模块。但是命名为main的子模块会运行。如果main模块不需要使用闭包模块,它可以使用module定义。或者使用module+
(module+ name-id
decl ...)
一个使用module+的模块就像使用module*,并且用#f来作为initial-module-path参数。而且,module+的name-id可以一样,所有同名的module+方法体联合起来构建一个子模块。这种合并行为特别适合用来构建测试。只要在命令行里欲行 raco test ...就可以运行其中的测试。否则加载模块不会欲行测试代码。
使用module+可以使代码分布在不同的地方。而且使用module+更具有可读性。
网友评论