pyinstaller遇到的问题
1.打包问题
ModuleNotFoundError: No module named ‘xxx’
依赖有第三方库的打包
-p 后面跟着的是第三方库的路径(这里我直接把整个site-package都集成到exe里面了)【 如果包含多个目录,则可以用分号隔开: 】
pyinstaller -F -p D:\K_Learning\pythonFiles\airtest_script\venv\Lib\site-packages devicesConnect.py
如何查看当前代码所依赖的第三方库的路径:
最近利用tkinter+python+pyinstaller实现了小工具的项目,在此记录下pyinstaller相关参数以及爬过的坑。 一、pyinstaller相关参数 -F, –onefile 打包一个单个文件,如果你的代码都写在一个.py文件的话,可以用这个,如果是多个.py文件就别用 (这个方便) -D, –onedir 打包多个文件,在dist中生成很多依赖文件,适合以框架形式编写工具代码,我个人比较推荐这样,代码易于维护 -K, –tk 在部署时包含 TCL/TK -a, –ascii 不包含编码.在支持Unicode的python版本上默认包含所有的编码. -d, –debug 产生debug版本的可执行文件 -w,–windowed,–noconsole 使用Windows子系统执行.当程序启动的时候不会打开命令行(只对Windows有效) -c,–nowindowed,–console 使用控制台子系统执行(默认)(只对Windows有效) pyinstaller -c xxxx.py pyinstaller xxxx.py --console -s,–strip 可执行文件和共享库将run through strip.注意Cygwin的strip往往使普通的win32 Dll无法使用. -X, –upx 如果有UPX安装(执行Configure.py时检测),会压缩执行文件(Windows系统中的DLL也会)(参见note) -o DIR, –out=DIR 指定spec文件的生成目录,如果没有指定,而且当前目录是PyInstaller的根目录,会自动创建一个用于输出(spec和生成的可执行文件)的目录.如果没有指定,而当前目录不是PyInstaller的根目录,则会输出到当前的目录下. -p DIR, –path=DIR 设置导入路径(和使用PYTHONPATH效果相似).可以用路径分割符(Windows使用分号,Linux使用冒号)分割,指定多个目录.也可以使用多个-p参数来设置多个导入路径,让pyinstaller自己去找程序需要的资源 –icon=<FILE.ICO> 将file.ico添加为可执行文件的资源(只对Windows系统有效),改变程序的图标 pyinstaller -i ico路径 xxxxx.py –icon=<FILE.EXE,N> 将file.exe的第n个图标添加为可执行文件的资源(只对Windows系统有效) -v FILE, –version=FILE 将verfile作为可执行文件的版本资源(只对Windows系统有效) -n NAME, –name=NAME 可选的项目(产生的spec的)名字.如果省略,第一个脚本的主文件名将作为spec的名字 二、pyinstaller爬坑总结 第一次玩pyinstaller,遇到了不少坑,在这里总结一下,以免日后再次掉坑里。 1、先从这次写的工具再说吧,写这个工具的初衷是由于自己手工造数据文件花了很长时间,然后就想着写个工具来解决这个问题(也是想锻炼下自己的代码能力,哈哈哈),利用下班晚上以及周末写了一周大概写出来了。 这个工具是个扫码支付文件生成工具,主要是解决手工造文件数据,有两个功能,一是基础扫码文件生成功能、二是大批量扫码文件生成功能,同时加上了用户auth控制。 在这里提醒大家,在代码里面尽量不要用import,能from.....import....就尽量用这个,因为如果是import的话,在打包的时候,会将整个包都打包到exe里面,没有意义的增大了工具的大小! 大概框架如下: 2、Gui开发完毕后,开始研究pyinstaller,也翻阅了论坛关于pyinstaller的相关用法,这里总结下我的工具的打包步骤 2.1、先下载pyinstaller,我比较懒,就直接用pip install pyinstaller,等待自动安装 2.2、在代码的路径下进行cmd,就直接跳转到该路径的cmd界面,切记路径中不要有中文 2.3、先用后台模式生成工具exe,命令为pyinstaller xxxx.py文件,主要目的是为了看报错信息,解决了报错后,再生成完整版的工具 tip:由于我的代码是存在依赖的,即多文件的,而非所有代码都在一个文件中的,所以我在这里没有使用-F,刚开始玩的时候不知道,老是用了-F生成了单文件的,没有生成依赖,老是不成功,各位朋友切记! 下图为命令生成的日志,红框部分则为exe工具生成的路径,如果在日志中只有info信息的话,说明打包过程没有问题 2.4、找到生成的exe文件,运行exe(生成的exe文件都会在dist目录下) 运行时,我遇到了以下的报错信息,是在dist找到相关的图片资源,所以需要把所有关于工具所需的资源(包含图片以及依赖的.py文件都放进dist下的项目目录中) 2.5、再次运行exe,此时运行成功,说明工具打包成功 2.6、重新生成exe工具,去掉后台模式以及更换掉图标 给大家推荐一个图标获取地址,里面ico蛮多的:easyicon 下载了自己心仪的ico后,用-i+ico路径来替换默认ico,这次重新生成exe工具,命令为pyinstaller -i ico路径 -w xxxx.py 再将需要的图片资源和代码文件放到dist下的项目文件夹中 2.7、运行exe,能够打开则表示pyinstaller打包exe成功
(这是-F生成的)
pyinstaller -F -p D:\K_Learning\pythonFiles\airtest_script\venv\Lib\site-packages devicesConnect.py
Excel相关操作
import xlrd import xlsxwriter class rwExcel: def __init__(self): pass def readexcel(self): # todo 打开excle xl = xlrd.open_workbook(r'Test.xlsx') #print(xl.read()) # todo 通过索引获取工作表 table = xl.sheets()[0] print(table) # 获取一共多少行 rows = table.nrows print(rows) # todo 获取第一行的内容,索引从0开始 row = table.row_values(0) print(row) # todo 获取第一列的整列的内容 col = table.col_values(0) print(col) # todo 获取单元格值,第几行第几个,索引从0开始 data = table.cell(2, 2).value print(data) def writeexcel(self): # todo 创建excel文件 xl = xlsxwriter.Workbook(r'../Excel_Report/test.xlsx') # todo 添加sheet sheet = xl.add_worksheet('sheet1') # todo 往单元格cell添加数据,索引写入 sheet.write_string(0, 0, 'username') # todo 位置写入 sheet.write_string('B1', 'password') # todo 设置单元格宽度大小 sheet.set_column('A:B', 30) # todo 关闭文件 xl.close() if __name__ == '__main__': w= rwExcel() read = w.readexcel()
import xlrd class RWExcel: def __init__(self): # todo 打开excle self.xl = xlrd.open_workbook(r'./ExcelFiles/Test.xlsx') # todo 通过索引获取工作表 self.table = self.xl.sheet_by_index(0) # print("工作表的名字为:{}".format(self.table.name)) # 获取一共多少行 self.rows = self.table.nrows # 获取一共有多少列 self.cols = self.table.ncols # print("当前表一共有"+str(self.rows) +"行,"+ str(self.cols) + "列") # for i in range(rows): # print(table.cell(i,0)) #打印第一列 def getSelect(self): """ :return: 获取表中select为1的行数(下标从0开始) """ #读取表格第一列除去第一行的值 self.fristcol = self.table.col_values(0) fristcol_list = [] fristcol_list.append(self.table.cell(0,0).value) #把第一行第一列的值加进去 for i in range(1,len(self.fristcol)) : # 把除去第一行的第一列的0和1强转为int类型的数值,并且加入到列表里面(第一行的值最后也在,目的是为了后面通过下标找对应的行) fristcol_list.append(int(self.fristcol[i])) #print("当前表第一列的值为:{}".format(fristcol_list)) #获取fristcol_list中为1的下标,同时用一个列表存储这些下标 indexlist = [] for i in range(len(fristcol_list)) : if fristcol_list[i] == 1 : #如果为1,那就是需要运行这一行的代码,此时获取下标,就相当于是获取第几行 indexlist.append(i) # i就是下标 #print("当前表中select是1的行数为:{}".format(indexlist)) return indexlist def selectLanguage(self): """ :return: 获取表格中select为1所对应的语言列表 """ indexlist = self.getSelect() #获取select为1的列 print("第二个方法里面的Select为1的行数{}".format(indexlist)) print("---------------------------------") languagelist = [] for i in indexlist: #获取对应select为1的平台切换语言列表(第4列) language = self.table.cell(i,3).value languagelist.append(language) #获取select为1对应的平板切换语言 print("当前表中select为1对应的切换语言为:{}".format(languagelist)) return languagelist def list_dic(self): ''' two lists merge a dict,a list as key,other list as value 把select为1的行数作为键,把要切换的语言作为值 :return:dict ''' list1 = self.getSelect() list2 = self.selectLanguage() dic = dict(map(lambda x, y: [x, y], list1, list2)) return dic def differ(self,index): i = self.table.cell(index,2).value j = self.table.cell(index,4).value if i == j : print("一样") print(i) print("--------------------------------------") print(j) return "PASS" if i != j : print("不一样") return "FAILED"
excel的写入(在已有的表格插入数据)
from openpyxl import load_workbook from .readExcelFile import RWExcel #写入已存在的xlsx文件第一种方法 class Write_excel(object): '''修改excel数据''' def __init__(self,filename,index,msg): """ :param filename: 测试的文件名 :param index: 对应要写入的行数的下标 :param msg: 需要写入的内容 """ self.filename = filename self.index = index self.msg = msg def write(self): #写入已存在的xlsx文件 wb = load_workbook(self.filename)#生成一个已存在的wookbook对象 wb1 = wb.active#激活sheet wb1.cell(self.index+1,5,self.msg)#往sheet中的第二行第五列写入msg的数据,注意这里的下标是从1开始 wb.save(self.filename) # 保存 def write_result(self): result = RWExcel().differ(self.index) # 写入已存在的xlsx文件 wb = load_workbook(self.filename) # 生成一个已存在的wookbook对象 wb1 = wb.active # 激活sheet wb1.cell(self.index + 1, 6, result) # 把对比的结果写入到表格中 wb.save(self.filename) # 保存 def saveExcel(self): pass # if __name__ == '__main__': # run = Write_excel().write("./Test.xlsx",1,"你好,摩西摩西")

