您好,登錄后才能下訂單哦!
這篇文章主要為大家展示了pyinstaller打包找不到文件怎么辦,內容簡而易懂,希望大家可以學習一下,學習完之后肯定會有收獲的,下面讓小編帶大家一起來看看吧。
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
已被打包。然后要注意一個問題:
打包出來的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://www.jb51.net/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打包找不到文件怎么辦的內容,如果你們有學習到知識或者技能,可以把它分享出去讓更多的人看到。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。