第五章 编译节点对象
在SCons内部,所有的文件和路径都被看作是节点(Nodes),我们可以采用这种方式让您的SConscript脚本更加便于迁移和阅读。
5.1 返回目标节点列表的编译方法
所有的构建方法返回一个节点对象列表,这些节点可以被用作其他构建方法的参数。
举例而言,如果我们想要构建两个中间文件,他们采用不同的编译参数:
Object('hello.c', CCFLAGS='-DHELLO')
Object('goodbye.c', CCFLAGS='-DGOODBYE')
将上述两个中间文件连接在一起,构建最终的输出文件,一种方式是将上述输出中间文件放入一个列表中,作为Program的参数:
Object('hello.c', CCFLAGS='-DHELLO')
Object('goodbye.c', CCFLAGS='-DGOODBYE')
Program(['hello.o', 'goodbye.o'])
这样做在跨平台方面会存在问题,因为在Windows平台上,生成的中间文件是hello.obj和goodbye.obj,而非hello.o和goodby.o。
较好的处理方式是将Object的构建输出存入变量中,这样我们可以不断在列表后追加新的内容,将其作为Program的输入:
hello_list = Object('hello.c', CCFLAGS='-DHELLO')
goodbye_list = Object('goodbye.c', CCFLAGS='-DGOODBYE')
Program(hello_list + goodbye_list)
这样SConstruct脚本的跨平台性可以得到保证,其在Linux平台输出如下:
> scons -Q
cc -o goodbye.o -c -DGOODBYE goodbye.c
cc -o hello.o -c -DHELLO hello.c
cc -o hello hello.o goodbye.o
Windows平台输出如下:
C:\>scons -Q
cl /Fogoodbye.obj /c goodbye.c -DGOODBYE
cl /Fohello.obj /c hello.c -DHELLO
link /nologo /OUT:hello.exe hello.obj goodbye.obj
embedManifestExeCheck(target, source, env)
后续将会介绍更多的编译节点的使用方法。
5.2 明确创建文件和路径节点
值得一提的是,SCons明确了文件节点和路径节点的不同,SCons支持File 和Dir两个函数,用于返回文件和路径节点:
hello_c = File('hello.c')
Program(hello_c)
classes = Dir('classes')
Java(classes, 'src')
通常情况下,您不需要手动调用File或Dir,因为调用构建方法时,会自动将输入作为文件或目录的名称,并将其转换为一个Node对象。除非您需要明确SCons的节点类型,比如在调用构建方法或者需要调用特殊文件或路径树的时候。
有时候,您可能在无法明确编译系统环境的前提下,需要一个输入,这可能是文件也可能是一个路径,此时SCons提供了一个Entry函数,用于返回一个文件节点或者路径节点。
xyzzy = Entry('xyzzy')
这里返回的xyzzy节点,将会在其第一次被调用的时候,转化为文件节点或路径节点。
5.3 打印节点文件名称
大多数情况下我们需要调用节点去打印其内部的文件名称,但是请注意此时的对象是节点列表,而非文件对象,因此打印的时候需要增加下标访问:
object_list = Object('hello.c')
program_list = Program(object_list)
print("The object file is: %s"%object_list[0])
print("The program file is: %s"%program_list[0])
在POSIX系统输出如下:
> scons -Q
The object files is: hello.o
The program file is: hello
cc - o hello.o -c hello.c
cc -o hello hello.o
在Windows系统输出如下:
C:\>scons -Q
The object file is: hello.obj
The program file is: hello.exe
cl /Fohello.obj /c hello.c /nologo
link /nologo /OUT:hello.exe hello.obj
embedManifestExeCheck(target, source, env)
请注意,在上面的示例中,object_list[0]从列表中提取了一个实际的Node对象,Python的print语句将该对象转换为要打印的字符串。
5.4 将节点对象转换为字符串
如上节介绍所示,我们可以直接打印节点的文件信息,但是如果您想得到一个节点字符串,而非列表,则可以通过python内置的str函数实现。举例而言,如果您希望使用Python的os.path.exists来确认文件是否存在,您可以采用如下方式:
import os.path
program_list = Program('hello.c')
program_name = str(program_list[0])
if not os.path.exists(program_name):
print("%s doses not exist!"%program_name)
此时,在POSIX系统中将会得到如下输出:
> scons -Q
hello does not exist!
cc -o hello.o -c hello.c
cc - o hello hello.o
5.5 GetBuildPath: 通过节点或字符串获取路径
env.GetBuildPath(file_or_list)返回节点或者字符串(描述路径)的路径,当然输入参数也可以是列表,这将会返回路径的列表信息。如果输入是单个节点的话,则相当于调用了str(node)。字符串参数可以被初始化在Environment函数参数列表中,此时将会被传递到后续的调用函数中:
env = Environment(VAR="value")
n = File("foo.c")
print(env.GetBuildPath([n, "sub/dir/$VAR"]))
打印信息如下:
> scons -Q
['foo.c', 'sub/dir/value']
scons: '.' is up to date
GetBuildPath函数也可以被默认的Environment环境调用,而非特殊指定。
网友评论