木兰在启动时将扫描提供的命令行参数用于获取各种设置信息和进入不同的模式。
可以对木兰按以下形式发起命令行调用:
木兰 [选项] [-树p码兰调交反执溯版助] [源码文件] [其他参数]
选项:
--语法树 -树 将木兰源码转换为 Python 语法树
--木兰变python -p 将木兰源码转换为 Python 源码
--生成字节码 -码 将木兰源码转换为 donsok 字节码 (实验性)
--python变木兰 -兰 将 Python 源码转换为木兰源码
--调试 -调 使用 Pdb 环境调试代码
--交互 -交 以交互式审查脚本
--反汇编 -反 将生成的 Python 字节码反汇编
--执行代码=<代码> -执 执行来自命令行参数的代码
--显示回溯 -溯 显示异常的栈回溯信息
--版本 -版 显示版本
--帮助 -助 显示帮助信息
下面是原始木兰的命令行帮助文本:
usage: ulang-0.2.2.exe [-apbcidsDth] input_file
Options and arguments:
--dump-ast, -a dump ast info
--dump-python, -p dump python source code
--dump-blockly, -b dump blockly xml (experimental)
--dump-bytecode, -c dump donsok bytecode (experimental)
--python-to-ulang, -s convert python to ulang
--debug, -D debug with Pdb (experimental)
--interact, -i inspect interactively after running script
--disassemble, -d disassemble the python bytecode
--exec-code=<code> -e run code from cli argument
--show-backtrace, -t show backtrace for errors
--version, -v show the version
--help, -h show this message
下面是重现项目与原始木兰的命令行选项的对照表:
重现项目 | 原始木兰 |
---|---|
语法树 | dump-ast |
木兰变python | dump-python |
生成字节码 | dump-bytecode |
python变木兰 | python-to-ulang |
调试 | debug |
交互 | interact |
反汇编 | disassemble |
执行代码=<代码> | exec-code=
|
显示回溯 | show-backtrace |
版本 | version |
帮助 | help |
【待调研】 | dump-blockly |
提供给木兰运行的代码文件。
当传递给此选项的值为 -
时,将从标准输入流读取文件,直到 EOF:
例如有如下代码 你好世界.ul
:
println('Hello world')
$ cat 你好世界.ul | 木兰 -
Hello world
其他参数是用户传递给脚本文件的参数。开发者可以通过导入 sys
模块的 argv
成员访问到其他参数。
例如下面的代码文件打印命令行参数.ul
:
using argv in sys
println(argv)
$ 木兰 打印命令行参数.ul a b c 1 2 3
[木兰, 打印命令行参数.ul, a, b, c, 1, 2, 3]
sys.argv
获取到的结果是一个全部由字符串构成的列表: [木兰执行文件名
, 脚本名
, 参数1
, 参数2
, ..., 参数n
]。需要注意的是,用户提供的命令行参数是从第三个元素开始的。
其中脚本名以后的部分,就是提供给脚本的参数,脚本程序可以根据这些参数,执行不同的功能。
通过遍历 sys.argv
成员,可以遍历所给的命令行参数列表。如下面这段代码 命令行参数相加.ul
,它将所给的命令行参数的数字相加:
using argv in sys
结果 = 0
所给参数 = argv[2:] // 命令行参数从下标 2 开始
for 参数 in 所给参数 {
结果 += int(参数)
}
println(结果)
$ 木兰 命令行参数相加.ul 1 3 5
9
$ 木兰 命令行参数相加.ul 9
9
$ 木兰 命令行参数相加.ul
0
下面的代码文件十进制转换.ul
是一个处理单个命令行参数的例子,这个程序将不断接受用户输入,然后根据提供的命令行参数输出不同进制的数字:
using argv, exit in sys
using bin, hex, oct in builtins
帮助 = '十进制转换.ul [二进制 | 八进制 | 十六进制]'
if len(argv) < 3 {
println(帮助)
exit(1)
}
所给参数 = argv[2]
模式函数 = {
'二进制': bin,
'八进制': oct,
'十六进制': hex
}.get(所给参数, 0)
if 模式函数 == 0 {
println(帮助)
exit(1)
}
while true {
数字 = input('输入数字(输入 exit 退出): ')
if 数字 == 'exit' {
break
}
if !数字.isnumeric() {
continue
}
println(模式函数(int(数字)))
}
$ 木兰 十进制转换.ul 十六进制
输入数字(输入 exit 退出): 2021727
0x1ed95f
输入数字(输入 exit 退出): 20200928
0x1343de0
输入数字(输入 exit 退出): exit
$ 木兰 十进制转换.ul 二进制
输入数字(输入 exit 退出): 20220926
0b1001101001000101111111110
输入数字(输入 exit 退出): 2047
0b11111111111
输入数字(输入 exit 退出): exit
下面的代码文件纯数字参数.ul
是一个处理多个命令行参数的例子。此程序将扫描提供的命令行参数,计算由纯数字构成的参数的数量。
using argv in sys
数量 = 0
for 参数 in argv[2:] {
if 参数.isnumeric() {
数量 += 1
}
}
println('命令行参数中由数字组成的有 %s 个' % 数量)
$ 木兰 纯数字参数.ul 1 a 2 b c d 3 e
命令行参数中由数字组成的有 3 个
$ 木兰 纯数字参数.ul mx ma mc md
命令行参数中由数字组成的有 0 个
打印给定木兰源码文件的语法树。
例如有如下代码文件code1.ul
:
x = 1
$ 木兰 --语法树 code1.ul
Module(body=[Assign(targets=[Name(id='x', ctx=Store(), lineno=1, col_offset=1)], value=Num(n=1, lineno=1, col_offset=5), lineno=1, col_offset=1)])
注:此输出是在 Python3.7 版本下运行的结果,在 Python3.8 下的运行结果见 Py3.8支持文档
更多测试用例请见: 测试/unittest/语法树.py
将用户提供的 木兰代码文件 转换成等效的 Python 代码并输出。
$ 木兰 --木兰变python 纯数字参数.ul
import sys
from math import *
ARGV = sys.argv[1:]
from sys import argv
数量 = 0
for 参数 in argv[2:]:
if 参数.isnumeric():
数量 += 1
print(__rem__('命令行参数中由数字组成的有 %s 个', 数量))
将木兰源码转换为 donsok 字节码 (实验性功能),由于 pygen未实现亦未找到第三方库,该选项暂时不可用
将用户提供的 Python代码文件 转换成等效的木兰代码并输出。
例如有如下的代码文件遍历列表.py
:
for i in range(10):
print(i * 2)
$ 木兰 --python变木兰 遍历列表.py
/* 本文件由命令 `木兰 -兰 ` 自动生成. */
for i in range(10) {
println(i * 2)
}
使用 pdb 调试指定的文件。
例如有如下文件 调试测试.ul
:
x = 1
println('x = %s' % x)
x = 2
println('x = %s' % x)
$ 木兰 --调试 调试测试.ul
> 调试测试.ul(1)<module>()
-> x = 1
(Pdb) l
1 -> x = 1
2 println('x = %s' % x)
3 x = 2
4 println('x = %s' % x)
[EOF]
(Pdb) step
> 调试测试.ul(2)<module>()
-> println('x = %s' % x)
(Pdb) step
--Call--
> D:\木兰\环境.py(117)__内置_求余()
-> def __内置_求余(a, b):
(Pdb) next
> D:\木兰\环境.py(119)__内置_求余()
-> return a % b
(Pdb) next
--Return--
> D:\木兰\环境.py(119)__内置_求余()->'x = 1'
-> return a % b
(Pdb) next
--Call--
> D:\木兰\环境.py(211)<lambda>()
-> 'println': lambda *各物件: 自定义输出(*各物件, **{'终止符': '\n'}),
(Pdb) continue
x = 1
x = 2
指定了该选项后,木兰在执行完给定的源码文件时,将携带执行完毕后的全局名称空间进入到交互环境。例如有如下代码文件预置.ul
:
println('定义 a = 1')
a = 1
println('设置完毕,程序结束。')
$ 木兰 --交互 预置.ul
定义 a = 1
设置完毕,程序结束。
木兰向您问好
更多信息请说'你好'
> a
1
> a + 2
3
若正常进入交互环境,则不会有 a
变量。
$ 木兰
木兰向您问好
更多信息请说'你好'
> a
(>﹏<)请先定义‘a’再使用,见第1行
> a + 2
(>﹏<)请先定义‘a’再使用,见第1行
通过 dis
模块反汇编由木兰源码编译生成的 Python 字节码。
例如,有如下的代码文件代码片段1.ul
:
a = 1
b = 2
print(a + b)
$ 木兰 --反汇编 代码片段1.ul
1 0 LOAD_CONST 0 (1)
2 STORE_NAME 0 (a)
2 4 LOAD_CONST 1 (2)
6 STORE_NAME 1 (b)
3 8 LOAD_NAME 2 (print)
10 LOAD_NAME 0 (a)
12 LOAD_NAME 1 (b)
14 BINARY_ADD
16 CALL_FUNCTION 1
18 POP_TOP
20 LOAD_CONST 2 (None)
22 RETURN_VALUE
执行来自命令行参数的代码。如:
$ 木兰 --执行代码="println('Hello world!')"
Hello world!
启用此选项后,当执行的脚本发生异常时,将使用 Python 原生的异常处理器处理。例如在执行下面这段代码时,由于没有定义 foo
名称,此段代码将会发生错误。
假若有如下代码文件错误测试.ul
:
foo()
下面是未指定 --显示回溯
选项时的运行结果:
$ 木兰 错误测试.ul
(>﹏<)请先定义‘foo’再使用
见第1行:foo()
下面是指定了 --显示回溯
选项的运行结果:
$ 木兰 --显示回溯 错误测试.ul
Traceback (most recent call last):
File "C:\Python37\lib\runpy.py", line 193, in _run_module_as_main
"__main__", mod_spec)
File "C:\Python37\lib\runpy.py", line 85, in _run_code
exec(code, run_globals)
File "C:\木兰\__main__.py", line 4, in <module>
中(sys.argv)
File "C:\木兰\中.py", line 154, in 中
exec(可执行码, 环境变量)
File "错误测试.ul", line 1, in <module>
foo()
NameError: name 'foo' is not defined
显示当前木兰的版本信息
显示帮助信息