在 Windows 上嵌入支持 tkinter 的 Python 3.5
Embed Python 3.5 with tkinter support on Windows
我的项目结构如下所示:
emb
| CMakeLists.txt
| main.c
| python35.lib
| stdlib.zip
| _tkinter.pyd
|
+---include
| |
| | abstract.h
| | accu.h
| | asdl.h
...
| | warnings.h
| | weakrefobject.h
|
+---build
| | emb.exe
stdlib.zip 包含 DLLs、Lib 和 site-packages 来自 Python 3.5.2 安装的目录,其路径附加到 sys.path
。我通过链接到 python35.lib 隐式加载 python35.dll,其中包含所有导出函数的存根动态链接库。这是CMakeLists.txt的内容:
cmake_minimum_required(VERSION 3.6)
project(embpython)
set(SOURCE_FILES main.c)
add_executable(${PROJECT_NAME} ${SOURCE_FILES})
set(PYTHON_INCLUDE_DIR include)
include_directories(${PYTHON_INCLUDE_DIR})
target_link_libraries(
${PROJECT_NAME}
${CMAKE_CURRENT_LIST_DIR}/python35.lib
${CMAKE_CURRENT_LIST_DIR}/_tkinter.pyd)
这里是main.c的内容:
#include <Python.h>
int main(int argc, char** argv)
{
wchar_t* program_name;
wchar_t* sys_path;
char* path;
program_name = Py_DecodeLocale(argv[0], NULL);
if (program_name == NULL)
{
fprintf(stderr, "Fatal error: cannot decode argv[0]\n");
exit(1);
}
Py_SetProgramName(program_name);
path = "stdlib.zip;stdlib.zip/DLLs;stdlib.zip/Lib;"
"stdlib.zip/site-packages";
sys_path = Py_DecodeLocale(path, NULL);
Py_SetPath(sys_path);
Py_Initialize();
PySys_SetArgv(argc, argv);
PyRun_SimpleString("import tkinter\n");
Py_Finalize();
PyMem_RawFree(sys_path);
PyMem_RawFree(program_name);
return 0;
}
现在,这是我遇到的错误:
Traceback (most recent call last):
File "<string>", line 1, in <module>
File " ... emb\stdlib.zip\Lib\tkinter\__init__.py", line 35, in <module>
ImportError: DLL load failed: The specified module could not be found.
我做错了什么,我该如何解决?
在 Jovito 回答了所有问题以指定如何使其成为轻型安装后才进行更新(步骤 3 到 5 这样可以节省 space):
将我的 _tkinter.pyd
文件复制到 \Pythonpath\DLLs\
到你的嵌入式安装目录 python35.dll
中。 tkinter
还需要 2 个 DLL 才能在 tcl86t.dll
和 tk86t.dll
的同一位置工作
您需要这些目录:\Pythonpath\Lib\tkinter
和\Pythonpath\tcl\tcl8.6
和\Pythonpath\tcl\tk8.6
并且必须在您的main.py
脚本中设置如下所示:
import os
os.environ['TCL_LIBRARY'] = "tcl//tcl8.6"
os.environ['TK_LIBRARY'] = "tcl//tk8.6"
这使得 Jovito 的回答尽可能轻量级。使用他的其余答案。适合我。
免责声明
这个答案并不是嵌入 Python 3.5 和 Tkinter 支持的正确或最佳方式。循序渐进的格式仅反映了这样一个事实,即这就是我设法让所有东西在我的机器上运行的方式,并且由于我无法在其他地方测试这个解决方案,所以我无法确认它在所有甚至大多数情况下都可以运行。
我是怎么做到的
- 创建 include、lib、lib\python35 和 src 中的目录
项目根目录。
- 将path\to\python35\include里面的所有文件复制到项目根目录下的include目录下
- 将 path\to\python35\Lib 中的所有文件压缩到一个名为 stdlib.zip 的文件中,并将其放在项目根目录。¹
- 将path\to\python35\DLLs里面的所有文件复制到项目根目录下的lib\python35目录下。 _tkinter.pyd库文件应该在里面。²
- 将 libpython35.a 导入库从 path\to\python35\libs 复制到 lib项目根目录下的目录。
在项目根目录的src目录下创建一个main.py文件以下内容:
import tkinter as tk
def run():
root = tk.Tk()
root.mainloop()
- 将main.py压缩成一个名为source.zip的文件,放在项目根目录下。
在项目根目录的src目录下创建一个main.c文件以下内容:
// WARNING: I did not check for errors but you definitely should!
#import <Python.h>
static const char* SYS_PATH = "source.zip;stdlib.zip;lib/python35";
int main(int argc, char** argv)
{
wchar_t* program = NULL;
wchar_t** wargv = NULL;
wchar_t* sys_path = NULL;
int i;
program = Py_DecodeLocale(argv[0], NULL);
Py_SetProgramName(program);
sys_path = Py_DecodeLocale(SYS_PATH, NULL);
Py_SetPath(sys_path);
Py_Initialize();
wargv = (wchar_t**) malloc(argc * sizeof(wchar_t*));
for (i = 0; i < argc; i++)
wargv[i] = Py_DecodeLocale(argv[i], NULL);
PySys_SetArgv(argc, wargv);
PyRun_SimpleString("import main\n"
"main.run()\n");
Py_Finalize();
PyMem_RawFree(program);
PyMem_RawFree(sys_path);
for (i = 0; i < argc; i++)
PyMem_RawFree(wargv[i]);
free(wargv);
return 0;
}
在项目根目录创建CMakeLists.txt文件,内容如下:
cmake_minimum_required(VERSION 3.6)
project(emb)
set(SOURCE_FILES src/main.c)
add_executable(emb ${SOURCE_FILES})
include_directories(include)
add_library(libpython35 STATIC IMPORTED)
set_property(
TARGET libpython35 PROPERTY IMPORTED_LOCATION
${CMAKE_CURRENT_LIST_DIR}/lib/libpython35.a)
target_link_libraries(emb libpython35)
构建和 运行。如果到目前为止您所做的一切都是正确的,您应该会看到类似这样的内容:
Traceback (most recent call last):
File "<string>", line 2, in <module>
File "C:\path\to\project\stdlib.zip\tkinter\__init__.py", line 1868, in __init__
_tkinter.TclError: Can't find a usable init.tcl in the following directories:
C:/path/to/project/lib/lib/tcl8.6
C:/path/to/project/lib/tcl8.6
C:/path/to/project/library
C:/path/to/project/tcl8.6.4/library
找不到Tcl 和Tk 目录。我们需要将它们引入并更新 TCL_LIBRARY 环境变量。
从 C:[=132] 复制 tcl8.6 和 tk8.6 目录=]到项目根目录下的lib目录下
创建 TCL_LIBRARY 环境变量并将其设置为 "lib\tcl8.6"
。
现在一切正常。
¹ 这不是绝对必要的。您也可以将 .py 文件保存在一个目录中,并将其路径附加到 sys.path
.
² 之前 python 引发 ImportError
的原因是因为 _tkinter.pyd 在一个 zip 文件中因此无法加载。
我的项目结构如下所示:
emb
| CMakeLists.txt
| main.c
| python35.lib
| stdlib.zip
| _tkinter.pyd
|
+---include
| |
| | abstract.h
| | accu.h
| | asdl.h
...
| | warnings.h
| | weakrefobject.h
|
+---build
| | emb.exe
stdlib.zip 包含 DLLs、Lib 和 site-packages 来自 Python 3.5.2 安装的目录,其路径附加到 sys.path
。我通过链接到 python35.lib 隐式加载 python35.dll,其中包含所有导出函数的存根动态链接库。这是CMakeLists.txt的内容:
cmake_minimum_required(VERSION 3.6)
project(embpython)
set(SOURCE_FILES main.c)
add_executable(${PROJECT_NAME} ${SOURCE_FILES})
set(PYTHON_INCLUDE_DIR include)
include_directories(${PYTHON_INCLUDE_DIR})
target_link_libraries(
${PROJECT_NAME}
${CMAKE_CURRENT_LIST_DIR}/python35.lib
${CMAKE_CURRENT_LIST_DIR}/_tkinter.pyd)
这里是main.c的内容:
#include <Python.h>
int main(int argc, char** argv)
{
wchar_t* program_name;
wchar_t* sys_path;
char* path;
program_name = Py_DecodeLocale(argv[0], NULL);
if (program_name == NULL)
{
fprintf(stderr, "Fatal error: cannot decode argv[0]\n");
exit(1);
}
Py_SetProgramName(program_name);
path = "stdlib.zip;stdlib.zip/DLLs;stdlib.zip/Lib;"
"stdlib.zip/site-packages";
sys_path = Py_DecodeLocale(path, NULL);
Py_SetPath(sys_path);
Py_Initialize();
PySys_SetArgv(argc, argv);
PyRun_SimpleString("import tkinter\n");
Py_Finalize();
PyMem_RawFree(sys_path);
PyMem_RawFree(program_name);
return 0;
}
现在,这是我遇到的错误:
Traceback (most recent call last):
File "<string>", line 1, in <module>
File " ... emb\stdlib.zip\Lib\tkinter\__init__.py", line 35, in <module>
ImportError: DLL load failed: The specified module could not be found.
我做错了什么,我该如何解决?
在 Jovito 回答了所有问题以指定如何使其成为轻型安装后才进行更新(步骤 3 到 5 这样可以节省 space):
将我的 _tkinter.pyd
文件复制到 \Pythonpath\DLLs\
到你的嵌入式安装目录 python35.dll
中。 tkinter
还需要 2 个 DLL 才能在 tcl86t.dll
和 tk86t.dll
您需要这些目录:\Pythonpath\Lib\tkinter
和\Pythonpath\tcl\tcl8.6
和\Pythonpath\tcl\tk8.6
并且必须在您的main.py
脚本中设置如下所示:
import os
os.environ['TCL_LIBRARY'] = "tcl//tcl8.6"
os.environ['TK_LIBRARY'] = "tcl//tk8.6"
这使得 Jovito 的回答尽可能轻量级。使用他的其余答案。适合我。
免责声明
这个答案并不是嵌入 Python 3.5 和 Tkinter 支持的正确或最佳方式。循序渐进的格式仅反映了这样一个事实,即这就是我设法让所有东西在我的机器上运行的方式,并且由于我无法在其他地方测试这个解决方案,所以我无法确认它在所有甚至大多数情况下都可以运行。
我是怎么做到的
- 创建 include、lib、lib\python35 和 src 中的目录 项目根目录。
- 将path\to\python35\include里面的所有文件复制到项目根目录下的include目录下
- 将 path\to\python35\Lib 中的所有文件压缩到一个名为 stdlib.zip 的文件中,并将其放在项目根目录。¹
- 将path\to\python35\DLLs里面的所有文件复制到项目根目录下的lib\python35目录下。 _tkinter.pyd库文件应该在里面。²
- 将 libpython35.a 导入库从 path\to\python35\libs 复制到 lib项目根目录下的目录。
在项目根目录的src目录下创建一个main.py文件以下内容:
import tkinter as tk def run(): root = tk.Tk() root.mainloop()
- 将main.py压缩成一个名为source.zip的文件,放在项目根目录下。
在项目根目录的src目录下创建一个main.c文件以下内容:
// WARNING: I did not check for errors but you definitely should! #import <Python.h> static const char* SYS_PATH = "source.zip;stdlib.zip;lib/python35"; int main(int argc, char** argv) { wchar_t* program = NULL; wchar_t** wargv = NULL; wchar_t* sys_path = NULL; int i; program = Py_DecodeLocale(argv[0], NULL); Py_SetProgramName(program); sys_path = Py_DecodeLocale(SYS_PATH, NULL); Py_SetPath(sys_path); Py_Initialize(); wargv = (wchar_t**) malloc(argc * sizeof(wchar_t*)); for (i = 0; i < argc; i++) wargv[i] = Py_DecodeLocale(argv[i], NULL); PySys_SetArgv(argc, wargv); PyRun_SimpleString("import main\n" "main.run()\n"); Py_Finalize(); PyMem_RawFree(program); PyMem_RawFree(sys_path); for (i = 0; i < argc; i++) PyMem_RawFree(wargv[i]); free(wargv); return 0; }
在项目根目录创建CMakeLists.txt文件,内容如下:
cmake_minimum_required(VERSION 3.6) project(emb) set(SOURCE_FILES src/main.c) add_executable(emb ${SOURCE_FILES}) include_directories(include) add_library(libpython35 STATIC IMPORTED) set_property( TARGET libpython35 PROPERTY IMPORTED_LOCATION ${CMAKE_CURRENT_LIST_DIR}/lib/libpython35.a) target_link_libraries(emb libpython35)
构建和 运行。如果到目前为止您所做的一切都是正确的,您应该会看到类似这样的内容:
Traceback (most recent call last): File "<string>", line 2, in <module> File "C:\path\to\project\stdlib.zip\tkinter\__init__.py", line 1868, in __init__ _tkinter.TclError: Can't find a usable init.tcl in the following directories: C:/path/to/project/lib/lib/tcl8.6 C:/path/to/project/lib/tcl8.6 C:/path/to/project/library C:/path/to/project/tcl8.6.4/library
找不到Tcl 和Tk 目录。我们需要将它们引入并更新 TCL_LIBRARY 环境变量。
从 C:[=132] 复制 tcl8.6 和 tk8.6 目录=]到项目根目录下的lib目录下
创建 TCL_LIBRARY 环境变量并将其设置为
"lib\tcl8.6"
。
现在一切正常。
¹ 这不是绝对必要的。您也可以将 .py 文件保存在一个目录中,并将其路径附加到 sys.path
.
² 之前 python 引发 ImportError
的原因是因为 _tkinter.pyd 在一个 zip 文件中因此无法加载。