pyinstaller打包找不到文件的问题解决


Posted in Python onApril 15, 2020

1、将python程序打包成单文件(使用 -F 参数)后,尝试运行外部文件却提示找不到的问题

当你将python程序打包成单文件(使用 -F 参数)后,运行程序,它实际上是先将exe内的资源文件解压到临时文件夹,然后再运行的,所以会导致这种问题

比如,当你在程序里面调用一个外部exe时,但却提示找不到该exe文件。

例子(这里我用win32api去隐式运行外部exe文件):

import win32api
win32api.ShellExecute(0, 'open', 'nginx.exe', '', '', 0)

首先,你需要将这个外部的exe文件添加进pyinstaller的打包里。

有两种方法:

1、直接用参数添加:

--add-data "nginx.exe;."

完整命令:pyinstaller -F main.py --add-data "nginx.exe;."

2、在spec文件添加:

每次执行pyinstaller打包命令后会生成spec文件,打开它
在里面找到data=[] 列表,添加元素,变成了:datas=[('nginx.exe', '.')]
然后用spec打包:pyinstaller main.spec
我解释一下这个点“.”是什么意思:

由于使用单文件打包出来的exe会先解压再运行,所以点“.” 其实表示你打包的这个exe文件运行解压的完整路径

如:C:\...\temp(临时文件夹)\asdqwezxc(你程序运行时自动解压到的目录)

  • 所以这个nginx.exe 被打包后,会解压到 C:\...\temp\asdqwezxc\nginx.exe
  • 如果把点“.”改为test,就会解压到 C:\...\temp\asdqwezxc\test\nginx.exe
  • 以此类推

好,现在nginx.exe已被打包。然后要注意一个问题:

打包出来的exe在运行时,它的工作路径和它解压到的路径,是不一样的!

你可以测试一下:

import os
print(os.getcwd())

可以发现,打印出来的工作路径并不是它运行时解压到的路径!
而是这个打包出来的exe,它本身所存在的路径!

问题来了:

诸如open('xxx.txt')这些操作文件的函数,一般首先都是在工作路径查找你所指定的文件的。

所以,当我们直接这样执行已打包的外部文件时,程序会报找不到文件!所以请使用它的解压路径。

下面提供一个函数,可以很方便的获取到解压路径:

import os, sys
def base_path(path):
  if getattr(sys, 'frozen', None):
    basedir = sys._MEIPASS
  else:
    basedir = os.path.dirname(__file__)
  return os.path.join(basedir, path)

print(base_path(''))
print(base_path('test\gg.exe'))

第一句打印会显示完整的解压路径:

C:\...\temp\asdqwezxc\

第二句打印是这样的:

C:\...\temp\asdqwezxc\test\gg.exe

所以当我们在调用已打包的外部文件时,应该先使用os.chdir()将工作路径改为解压路径:

再进行操作,就不会报文件找不到了

os.chdir(base_path(''))
win32api.ShellExecute(0, 'open', 'nginx.exe', '', '', 0)

不过要注意的是,如果你要写出文件到程序所在的目录(非解压目录),那么你得把工作目录改回来,否则文件会被写出到解压路径(临时文件夹)。

稍微封装一下就好了:

import os, sys

def base_path(path):
  if getattr(sys, 'frozen', None):
    basedir = sys._MEIPASS
  else:
    basedir = os.path.dirname(__file__)
  return os.path.join(basedir, path)

tmd = base_path('') # 这是解压路径
cwd = os.getcwd() # 这是程序的所在路径

# 当需要调用打包的外部文件时
os.chdir(tmd) # 先把工作路径变成解压路径
do() # 执行你要干的事情

# 当需要写出文件到程序所在目录时
os.chdir(cwd) # 把工作路径切换回来
do() # 执行你要干的事情

2、当你使用cython将py文件编译成pyd文件后使用pyinstaller打包,提示找不到模块的问题
直接使用pyinstaller打包py文件是很容易导致源码被反编译的

所以在打包的时候最好将py文件编译成pyd文件,这样可以很大程度上防止反编译。

为什么呢?因为pyd文件的来历是这样的:

py文件 → c文件 → pyd文件
所以直接反编译pyd只能得到上一步cython生成的c文件,而无法得到我们的py源文件。

如何打包pyd成文件请看这篇文章:https://3water.com/article/184725.htm
接下来回到我们的问题。

解决方法很简单,请看:

比如说,我有一个文件main.py,引入了位于同级目录下的test.py模块

# main.py:就像这样直接引入
import test

现在我将test.py 编译成pyd文件,生成了:test.cp37-win_amd64.pyd

这个pyd文件名除了我们原本的文件名test,还会带上编译环境的名称,这个环境后缀名我们可以不用管 ,因为python引入模块还是很智能的(会自动引入.pyd文件,因为它的优先级高于.py文件)。

这么智能,但是为什么我用pyinstaller打包时就提示找不到文件?
其实我们需要在打包时--hidden-import这些模块

1、直接添加

在打包时添加--hidden-import test即可
完整命令:pyinstaller -F --hidden-import test

2、使用spec文件

同样的,运行一次pyinstaller打包命令后会生成spec文件,打开它
找到hiddenimports=[],添加test模块,变成了:hiddenimports=['test']
很简单对吧?
而且除了我们自己写的一些py模块,其它模块在打包时可能也会提示找不到,都可以用这个方法解决。

3、打包成单文件时(使用-F参数),运行时要求管理员权限的参数--uac-admin无效的问题
请看我的这篇文章:pyinstaller打包单文件时?uac-admin选项不起作用怎么办

本质上大概也是因为找不到文件。

4、无控制台打包(使用-w参数),运行时弹框提示Failed to execute script的问题
请看我的这篇文章:pyinstaller打包成无控制台程序时运行出错,与popen冲突的解决方法

这个问题一般是程序内有输入导致的,这个输入可以是input(),也可以是其它的一些stdin操作(如os.popen实际上会造成输入请求)

本质上就是:使用-w参数(无控制台)打包时程序里不要请求输入

当然,实在要用输入,又不想要控制台怎么办?很简单,把控制台隐藏了就行!

下列两个方法,试试看:

import ctypes
def hideConsole():
  """
  Hides the console window in GUI mode. Necessary for frozen application, because
  this application support both, command line processing AND GUI mode and theirfor
  cannot be run via pythonw.exe.
  """

  whnd = ctypes.windll.kernel32.GetConsoleWindow()
  if whnd != 0:
    ctypes.windll.user32.ShowWindow(whnd, 0)
    # if you wanted to close the handles...
    #ctypes.windll.kernel32.CloseHandle(whnd)

def showConsole():
  """Unhides console window"""
  whnd = ctypes.windll.kernel32.GetConsoleWindow()
  if whnd != 0:
    ctypes.windll.user32.ShowWindow(whnd, 1)

暂时就这么多呃,都是本人在打包过程中实际上遇到过的问题和经验。。基本可用

到此这篇关于pyinstaller打包找不到文件的问题解决的文章就介绍到这了,更多相关pyinstaller打包找不到文件内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Python 相关文章推荐
初步探究Python程序的执行原理
Apr 11 Python
浅谈python函数之作用域(python3.5)
Oct 27 Python
Python编程scoketServer实现多线程同步实例代码
Jan 29 Python
python实现反转部分单向链表
Sep 27 Python
python pandas实现excel转为html格式的方法
Oct 23 Python
python IDLE 背景以及字体大小的修改方法
Jul 12 Python
python2和python3实现在图片上加汉字的方法
Aug 22 Python
pytorch::Dataloader中的迭代器和生成器应用详解
Jan 03 Python
pytorch 实现张量tensor,图片,CPU,GPU,数组等的转换
Jan 13 Python
python 使用递归实现打印一个数字的每一位示例
Feb 27 Python
Python jieba结巴分词原理及用法解析
Nov 05 Python
python 通过 pybind11 使用Eigen加速代码的步骤
Dec 07 Python
使用Pycharm分段执行代码
Apr 15 #Python
pyinstaller打包成无控制台程序时运行出错(与popen冲突的解决方法)
Apr 15 #Python
pyinstaller打包单文件时--uac-admin选项不起作用怎么办
Apr 15 #Python
在python中利用pycharm自定义代码块教程(三步搞定)
Apr 15 #Python
PyInstaller将Python文件打包为exe后如何反编译(破解源码)以及防止反编译
Apr 15 #Python
Python任务调度模块APScheduler使用
Apr 15 #Python
Python实现代码块儿折叠
Apr 15 #Python
You might like
《星际争霸II》全新指挥官斯台特曼现已上线
2020/03/08 星际争霸
php抓即时股票信息
2006/10/09 PHP
PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式
2012/04/09 PHP
PHP快速按行读取CSV大文件的封装类分享(也适用于其它超大文本文件)
2014/04/10 PHP
destoon供应信息title调用出公司名称的方法
2014/08/22 PHP
必须收藏的23个php实用代码片段
2016/02/02 PHP
PHP在线打包下载功能示例
2016/10/15 PHP
把JS与CSS写在同一个文件里的书写方法
2007/06/02 Javascript
JavaScript实现的内存数据库LokiJS介绍和入门实例
2014/11/17 Javascript
javascript实现无限级select联动菜单
2015/01/02 Javascript
jQuery时间插件jquery.clock.js用法实例(5个示例)
2016/01/14 Javascript
js实现div模拟模态对话框展现URL内容
2016/05/27 Javascript
浅谈js中几种实用的跨域方法原理详解
2016/12/02 Javascript
微信小程序swiper实现滑动放大缩小效果
2018/11/15 Javascript
微信小程序自定义导航栏(模板化)
2019/11/15 Javascript
快速解决Vue、element-ui的resetFields()方法重置表单无效的问题
2020/08/12 Javascript
分享8个JavaScript库可更好地处理本地存储
2020/10/12 Javascript
详解微信小程序(Taro)手动埋点和自动埋点的实现
2021/03/02 Javascript
Python中的魔法方法深入理解
2014/07/09 Python
python实现自动更换ip的方法
2015/05/05 Python
详解django三种文件下载方式
2018/04/06 Python
python 检查文件mime类型的方法
2018/12/08 Python
对python 中class与变量的使用方法详解
2019/06/26 Python
Python二元算术运算常用方法解析
2020/09/15 Python
美国领先的汽车轮胎和轮毂供应商:TireBuyer
2016/07/21 全球购物
香港优质食材和美酒专门店:FoodWise
2017/09/01 全球购物
.NET概念性的面试题
2012/02/29 面试题
管理部部长岗位职责
2013/12/05 职场文书
酒店经理职责
2014/01/30 职场文书
四年大学生活的自我评价范文
2014/02/07 职场文书
集中采购方案
2014/06/10 职场文书
小学六一儿童节活动开幕词
2016/03/04 职场文书
python实战之90行代码写个猜数字游戏
2021/04/22 Python
python 开心网和豆瓣日记爬取的小爬虫
2021/05/29 Python
frg-100简单操作(设置)说明
2022/04/05 无线电
Redis实现短信验证码登录的示例代码
2022/06/14 Redis