Haskell 中的条件编译
背景
最近在用
Haskell
语言和
haskell-gi
包写一款
Haskell
的
IDE
,这款软件初步支持简体中文和英文两种语言环境(中文简体语言环境展示中文简体,其他环境展示英文)。实现这种国际化的需求,有多种方法,比如,在
C/C++
下可以使用
GNU
的
gettext
,在Haskell下也有其移植。我采用了另一种方法:通过判断当前语言环境,决定要使用的语言版本,所以实现都是可以通过纯
Haskell
代码实现,多了一层类型安全的保证。
由于目前在Haskell中还没有好用的方法获取语言环境(glib中提供了
g_get_language_names
函数实际上可以达到目的,但比较难用),所以需要调用平台相关的函数,在
Windows
下使用
GetSystemDefaultLangID
。这需要根据不同的操作系统执行的不一样的代码,即根据不同的平台编译不同部分的代码,即
条件编译
。
Haskell中的条件编译支持
比较令人失望的是,在条件编译这块,
Haskell
并没有自己出彩的地方,继续沿用了
C/C++
那一套,即使用预处理指令判断预定义的宏达到编译不同部分代码的目的。在语法上,Haskell的预编译指令跟C/C++的完全一样,所以有了
C/C++
的经验可以无缝迁移到
Haskell
上面。
下面是一些使用上的细节点:
Q1:怎么启用
Haskell
程序的预处理过程
我们知道,正常的
Haskell
程序处理过程是没有预处理的,我们需要一种方式明确告知构造过程要调用预处理器。这是通过
CPP
语言扩展实现的,这需要我们在源文件的顶层添加:
{-# LANGUAGE CPP #-}
Q2:Haskell提供了哪些标准宏
在
GHC
(8.2.2)用户手册
7.11.3.1. Standard CPP macros
列了一些可用的宏可以用于代码,包括GHC软件版本、操作系统、硬件架构、软件包等方面的宏,下面仅摘录两个:
os_HOST_OS=1
This define allows conditional compilation based on the Operating System, where⟨os⟩ is the name of the current Operating System (eg.
linux
,
mingw32
for Windows,
solaris
, etc.).
arch_HOST_ARCH=1
This define allows conditional compilation based on the host architecture, where⟨arch⟩ is the name of the current architecture (eg.
i386
,
x86_64
,
powerpc
,
sparc
, etc.).
除了这些预定义宏,你也可以在代码里像使用
C/C++
一样,定义自己的宏。
Q3:怎将通过命令参数定义宏
一般的
C/C++
编译器都允许通过编译器选项定义宏,而
GHC
也提供了类似的选项:
最常用的是
-D
选项,这跟gcc编译器的用法完全一致,可以在编译时通过定义不同的宏控制编译过程。
示例
最后一个简单例子演示怎么使用条件编译。
{-# LANGUAGE CPP #-}
main :: IO () main = #if defined linux_HOST_OS putStrLn "Linux" #elif defined mingw32_HOST_OS putStrLn "Windows" #else putStrLn "Other" #endif
有一点需要注意的是,
Windows
下的宏定义
mingw32_HOST_OS
代表全部
Windows
系统,不要被其中的
32
迷惑,以为仅代表是
32
位
Windows
系统。