转自原文:https://zhuanlan.zhihu.com/p/22882167?refer=python-dev ====== Python与Tcl混合编程 - 在Python中调用Tcl ====== 本系列讨论在Python中使用Tcl脚本,以及如何使用Python对Tcl进行扩展。 ================================================================================================================================================================================================================== Python的文档对Tkinter的介绍,着重于图形界面编程,而对如何使用Tcl语言与Python代码交互,说得并不明确。本文将着重介绍如何在Python中嵌入Tcl。 ===== - Python中嵌入Tcl ===== ==== - Hello Tcl! ==== 在Python中使用Tcl,不需要安装任何第三方模块。考虑下面一个简单的程序: import tkinter interp = tkinter.Tcl() interp.eval('puts "Hello Tcl!"') 我们首先利用TKinter.Tcl()得到一个Tcl解释器对象,然后执行它的eval方法,运行一小段Tcl代码。 然而,如果打算Tcl和Python环境有所交互,eval方法就力不从心了。 ==== - 添加命令 ==== 注:如果需要更多命令支持的话,可能需要自行查看''Lib/tkinter/__init__.py''文件。 利用createcommand方法将一个Python函数作为命令添加到Tcl解释器中。 下面一个例子是创建一个将所有参数数字相加的命令。 import tkinter import operator interp = tkinter.Tcl() def SumAll(*args): #return reduce(operator.add,[int(x) for x in args]) return args[0] + args[1] interp.createcommand('sum_all',SumAll) interp.eval(""" puts [sum_all 1 2 3] """) 注:目前只支持2个参数的createcommand,对于python函数中需要多个参数支持的,需要用户自行包一下。比如: import tkinter import operator interp = tkinter.Tcl() def py_add(a, b): return int(a)+int(b) # 从tcl侧传递进来的全部是字符,需要进行适当的类型转换后再使用。 def tcl_py_add(*args): l = len(args) if l != 2: print(f'wrong # args: should be "py_add a b"') return 'None' return py_add(args[0], args[1]) interp.createcommand('py_add',tcl_py_add) interp.eval(""" puts [py_add 1 2] """) interp.eval(""" puts [py_add 1] """) === - 使用注意 === python的命令加入tcl解释器之后,python侧出现的类型check错误信息不会显示在tcl中(一般显示为一个空行),这有时候会给定位带来非常大的困扰。 一定要注意从tcl侧输入的参数传递到python之后都是字符串,如果原来python的参数是int的需要使用类型转换。 ==== - 添加和收集变量 ==== 可以在tcl代码运行前,由Python环境向Tcl解释器『注入』一些预先定义的变量。Tcl代码运行完成之后,Python环境可以获得Tcl解释器中定义的所有变量。例如: import Tkinter interp = Tkinter.Tcl() interp.setvar(name='FOO', value='2') interp.eval(""" puts $FOO set BAR 3 """) print interp.getvar('BAR') 从解释器环境中删除某个标量(即相当于unset),使用interp.unsetvar(varName) Tcl的list与Python的Tuple可相互使用。 import Tkinter interp = Tkinter.Tcl() interp.setvar('list_a',(1,2)) interp.eval(""" puts [lindex $list_a 1] #2 set v [list a b [list x y ] "c d e " " f {g h}"] """) print interp.getvar('v') #('a', 'b', ('x', 'y'), 'c d e ', ' f {g h}') ==== - 追溯变量 ==== 若想让Python能够得以知晓Tcl解释器中对某个变量的操作,可利用Tcl的trace方法结合Python引入的command。(尽管Tcl本身提供了Tcl_TraceVar方法,但Tkinter不能够使用它) import Tkinter interp = Tkinter.Tcl() def Tracer(varname,*args): print interp.getvar(varname) interp.createcommand('python_tracer',Tracer) interp.eval("trace add variable foo write python_tracer") interp.eval(""" set foo 1 set foo 2 """) 由于设置了一个write trace,当对foo进行写入的时候,Python函数Tracer会得以执行。 ==== - 获取结果 ==== eval方法的返回值是当前表达式的值。例如: import Tkinter interp = Tkinter.Tcl() interp.eval(""" array set colorcount { red 1 green 5 blue 4 white 9 } """) print interp.eval('array names colorcount') #blue white green red 不要用这种方法获得简单变量以及list的值。 ===== - python zip tcl混合 ===== **测试程序** {{linux:python:pyziptcl.zip}} #!/usr/bin/python3 import tkinter import zipfile import io fh = open('ab.zip', 'rb') fbuf = fh.read() # 读取zip二进制文件内容到buffer,注意buffer溢出。 fh.close() fio = io.BytesIO() #创建一个文件io fio.write(fbuf) #将zip二进制内容写入文件io # zfile = zipfile.ZipFile('ab.zip', 'r') zfile = zipfile.ZipFile(fio, 'r') # 直接从文件io处操作,而不是指定硬盘上的文件。 for i in zfile.filelist: bin = zfile.read(i.filename) line = str(bin, encoding = "utf-8") interp = tkinter.Tcl() interp.eval(line) **运行结果** (base) D:\pyzip>python pyzip.py abc01 abc02 opq01 opq02