打包py、文件转换、验证码识别、获取文件等问题

2024-07-28 09:03:06 浏览数 (1)

pyinstaller——用于打包py程序

在日常中我们写好的pyhton脚本每次运行时都需要安装软件,但是这样造成了一个不好的现象就是,你写好脚本后需要供别人使用的时候,别人没下载软件则无法运行脚本,很麻烦很难受。

所以我们只需要将Python程序打包为exe文件以后,发送给任何人都可以使用,我们打包成exe需要使用到的工具是pyinstaller模块。

首先我们先安装pyinstaller

Python 默认并不包含 PyInstaller 模块,因此需要自行安装 PyInstaller 模块。安装 PyInstaller 模块与安装其他 Python 模块一样,使用 pip 命令安装即可。在命令行输入如下命令:pip install pyinstaller

PyInstaller生成可执行程序

pyinstaller 选项 Python 源文件:pyinstaller -F py文件路径

也可以先在cmd中 cd 进入到文件路径中,然后输入:yinstaller -F 文件名

不管这个 Python 应用是单文件的应用,还是多文件的应用,只要在使用 pyinstaller 命令时编译作为程序入口的 Python 程序即可。

例子:

我们简单创建一个文件:price.py

代码语言:python代码运行次数:0复制
import time

def main():
    for i in range(3):
    print(“开始打印第%s次” % i)
    time.sleep(2) # 沉睡两秒后继续执行



if name == ‘main’:
    main()
    print(“执行完毕”)

接下来使用命令行工具进入到此 app 目录下,执行如下命令:

我们先在cmd进入到文件地址,执行pyinstaller -F price.py将看到详细的生成过程。当生成完成后,将会在此 app 目录下看到多了一个 dist 目录,并在该目录下看到有一个 app.exe 文件,这就是使用 PyInstaller 工具生成的 EXE 程序。

打包后,在 pyinstaller 选项 Python 源文件 中,选项处时有很多参数的:

例如(其中部分):

代码语言:bash复制
-F,-onefile产生单个的可执行文件

-D,–onedir产生一个目录(包含多个文件)作为可执行程序

-a,–ascii不包含 Unicode 字符集支持

-d,–debug产生 debug 版本的可执行文件

但是这样打包以后的exe是原生图标,对于强迫症患者来说是一件很难受的事情,不过我们可以是使用 -i 加上需要显示的图片,这样就能更改打包后exe文件的图片了。pyinstaller -F price.py -i a.ico,但要注意所需设置的图片名格式得为ico后缀的才行

注:使用pyinstaller -F flower.py打包完成的exe文件运行时会弹出黑色的框(命令行窗口),为了使运行exe文件不再出现黑框,加上-w参数,即pyinstaller -F -w flower.py,则运行exe文件时黑框便没有了。

python将csv文件转换为unix的txt文件

目的:将csv文件保存为用空格分隔的且编码方式为utf-8,unix格式的txt文件。相关代码如下:

代码语言:python代码运行次数:0复制
# -*- coding:utf-8 -*-
# @author:Ye Zhoubing
# @datetime:2024/7/8 19:31
# @software: PyCharm
import csv
import os
import stat

def convert_csv_to_txt(csv_file, txt_file):
    with open(csv_file, 'r') as file:
        csv_reader = csv.reader(file)
        with open(txt_file, 'w', encoding="utf-8") as output_file:
            for row in csv_reader:
                line = ' '.join(row)  # 使用空格连接每个字段
                output_file.write(line   'n')  # 写入TXT文件,并在行末添加换行符


def change_dos_to_unix(file):
    subpath = file
    mode = os.stat(subpath)[stat.ST_MODE]
    with open(subpath, "rb ") as file_handle:
        all_text = file_handle.read()
        all_change_text = all_text.replace(b"rn", b"n")
        file_handle.seek(0, 0)
        file_handle.truncate()
        file_handle.write(all_change_text)


# 示例用法
csv_file = 'test.csv'
txt_file = 'output.txt'
convert_csv_to_txt(csv_file, txt_file)

change_dos_to_unix(txt_file)  # txt文件转换为UNIX格式
print("over")

参考链接

ddddocr识别计算型验证码方法

ddddocr是不支持直接识别计算计算型验证码的,比如下面这个。

可以识别后再进行处理。

代码如下:

代码语言:python代码运行次数:0复制
# -*- coding:utf-8 -*-
# @author:Ye Zhoubing
# @datetime:2024/7/1 19:16
# @software: PyCharm
# -*- coding: utf-8 -*-
# @Author: b01e
# @Time: 2023/8/9 17:19
# @File: dddd_ikun.py
# @Describe: from ddddocr 修改
# coding=utf-8
"""
注意一下使用版本的pillow
这个工程中使用的是pillow-9.5.0
针对的是:https://www.houdunren.com/login后盾人网站的登录验证码
todo:目前识别有些问题
"""
import ddddocr
import requests
import re


class RegularCalculation:
    __div = re.compile(r"(d .?d*/-d .?d*)|(d .?d*/d .?d*)")  # 查找除法运算

    __mul = re.compile(r"(d .?d**-d .?d*)|(d .?d**d .?d*)")  # 查找乘法运算

    __add = re.compile(r"(-?d .?d* -d .?d*)|(-?d .?d* d .?d*)")  # 查找加法运算

    __sub = re.compile(r"(-?d .?d*--d .?d*)|(-?d .?d*-d .?d*)")  # 查找减法运算

    def division(self, value):
        """计算除法"""
        exp = re.split(r'/', self.__div.search(value).group())  # 将除法表达式拆分成列表
        return value.replace(self.__div.search(value).group(), str(float(exp[0]) / float(exp[1])))  # 计算并用结果替换列表中表达式

    def multiplication(self, value):
        """计算乘法"""
        exp = re.split(r'*', self.__mul.search(value).group())  # 将乘法表达式拆分成列表
        return value.replace(self.__mul.search(value).group(), str(float(exp[0]) * float(exp[1])))  # 计算并用结果替换列表中表达式

    def addition(self, value):
        """计算加法"""
        exp = re.split(r' ', self.__add.search(value).group())  # 将加法表达式拆分成列表
        return value.replace(self.__add.search(value).group(), str(float(exp[0])   float(exp[1])))  # 计算并用结果替换列表中表达式

    def subtraction(self, value):
        """计算减法"""
        exp = self.__sub.search(value).group()  # 去除减法表达式
        if exp.startswith('-'):  # 如果表达式形如:-2.2-1.2;需变换为:-(2.2 1.2)
            exp = exp.replace('-', ' ')  # 将-号替换为 号; 2.2 1.2
            res = self.addition(exp).replace(' ', '-')  # 调用Add运算,将返回值 3.4变为-3.4
        else:
            exp = re.split(r'-', exp)
            res = str(float(exp[0]) - float(exp[1]))
        return value.replace(self.__sub.search(value).group(), res)


class VerificationCode:

    def __init__(self):
        # 这里将验证码图片保存在本地test.png
        # head = {
        #     "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.159 Safari/537.36"
        # }
        #
        # res_img = requests.get('https://www.登录地址.com/', headers=head)
        # with open('test.png', 'wb') as f:
        #     f.write(res_img.content)
        pass

    @staticmethod
    def verification_code():
        ocr = ddddocr.DdddOcr(show_ad=False)  # show_ad=False关闭广告
        with open("test.png", 'rb') as f:
            img_bytes = f.read()
        res = ocr.classification(img_bytes)
        # print(res)
        try:
            if ' ' in res:
                result = RegularCalculation().addition(res)
            elif '-' in res:
                result = RegularCalculation().subtraction(res)
            elif 'x' in res:  # 因为测试的验证码只有加号,识别出来的乘号是x其实也是+
                # result = RegularCalculation().multiplication(res.replace('x', '*'))
                result = RegularCalculation().addition(res.replace('x', ' '))
            else:
                result = RegularCalculation().division(res)
        except Exception as e:
            result = '计算出错'
            print(e)
        print(result.split('.')[0])


if __name__ == '__main__':
    test = VerificationCode()
    test.verification_code()

注意:

目前有些问题,ddddocr识别该验证码识别率比较低。还有就是 符号稍微偏一点,可能程序就会将其识别为x。目前就是手动将识别成x的转换成

获取控制台运行的python程序及参数

有时候,需要对运行的程序进行获取(比如多个程序同时进行,但参数与程序名有些区别),要想获取控制台运行的python程序及参数,下面的代码片段便可实现:

代码语言:python代码运行次数:0复制
# -*- coding:utf-8 -*-
# @author:Ye Zhoubing
# @datetime:2024/7/2 9:01
# @software: PyCharm
import sys

i = sys.argv[1]

# 获取当前正在执行的Python脚本的名称
script_name = sys.argv[0]

# 获取传递给脚本的参数列表(如果有的话)
arguments = sys.argv[1]

print(f"程序: {script_name} {arguments},已完成")

文件拖拽进窗口获取文件路径

下面介绍两种包下的实现方式:pyQt5与tkinter

pyQt5

相关代码如下:

代码语言:python代码运行次数:0复制
# -*- coding:utf-8 -*-
# @author:Ye Zhoubing
# @datetime:2024/6/30 15:55
# @software: PyCharm
"""
拖拽文件进入窗口获取文件路径

"""
import sys
from PyQt5.QtWidgets import QApplication, QMainWindow, QTextEdit, QPushButton, QVBoxLayout, QWidget, QDesktopWidget

'''
#1、使用以下代码中的文件拖拽功能,只需将文件或文件夹拖拽到文本编辑框中即可。如果文件是本地文件,它们将以文件路径的形式显示在文本编辑框中。
#2、如果你想要进一步处理这些文件路径,比如复制、移动、读取或执行其他操作,你可以在 processFiles 方法中添加你的自定义代码,该方法在用户点击提交按钮后被调用。在该方法中,你可以访问文本编辑框的内容,将其拆分成文件路径,并执行相应的操作。
'''


# 使用子类来继承父类的方法,这里的’DragDropTextEdit‘,继承自  ’QTextEdit‘ ,并且添加了文件拖拽的支持。
# 这使得你可以将它用作拖拽文件的目标,以便在应用程序中方便地处理文件路径。
class DragDropTextEdit(QTextEdit):
    def __init__(self, parent=None):
        super(DragDropTextEdit, self).__init__(parent)
        self.setAcceptDrops(True)  # 定义的 DragDropTextEdit 类的构造函数中调用的方法,它的作用是启用该文本编辑框接受拖拽操作。

    # 当用户拖拽文件或其他可拖拽的内容进入文本编辑框时,这个方法会被触发
    def dragEnterEvent(self, event):
        if event.mimeData().hasUrls():  # 它检查拖拽事件中的 MIME 数据是否包含文件路径
            event.accept()  # 如果包含文件路径,它调用 event.accept() 来接受拖拽事件,允许文件拖拽进入文本编辑框。
        else:  # 否则,如果不包含文件路径,它调用 event.ignore() 来忽略拖拽事件,表示不允许拖拽操作。
            event.ignore()

    def dropEvent(self, event):  # 当用户释放鼠标按钮时,这个方法会被触发,用于处理拖拽事件。在这个方法中,你可以获取拖拽事件中的文件路径。
        for url in event.mimeData().urls():  # 使用 event.mimeData().urls() 来获取所有拖拽事件中的 URL 列表,每个 URL 表示一个文件或文件夹。
            if url.isLocalFile() and url.toLocalFile():  # 使用 url.toLocalFile() 来获取本地文件路径,如果是本地文件的话。
                self.append(url.toLocalFile())  # 接下来,你可以将这些文件路径添加到文本编辑框中,或者进行任何你希望执行的处理。


class MainApp(QMainWindow):  # 创建实例化类
    def __init__(self):
        super().__init__()
        self.initUI()

    def initUI(self):
        self.setWindowTitle('文件拖拽窗口')  # 设置主窗口的标题为 "文件拖拽窗口"。
        # 获取屏幕的宽度和高度
        screen = QDesktopWidget().screenGeometry()
        screenWidth = screen.width()
        screenHeight = screen.height()
        # 计算窗口居中的坐标
        x = (screenWidth - self.width()) // 2
        y = (screenHeight - self.height()) // 2
        # 设置窗口大小已经出现在屏幕的什么位置
        self.setGeometry(x, y, 600, 400)  # 设置主窗口的初始位置和大小。 (x, y)是设置窗口出现的位置。窗口的宽度为 600 像素,高度为 400 像素。

        # 初始化窗口排版模式
        central = QWidget(self)  # 创建一个名为 central 的 QWidget(窗口中央部件),用于将其他小部件添加到主窗口的中央区域。
        self.setCentralWidget(central)  # 将 central 部件设置为主窗口的中央部分。这意味着所有其他小部件将放置在 central 部件中,以确保它们在窗口中间显示。
        display = QVBoxLayout(central)  # 创建一个垂直布局管理器 display,它将用于管理 central 部件中的小部件的位置和大小。垂直布局意味着小部件将按垂直方向排列。

        # 窗口
        self.textEdit = DragDropTextEdit()  #####这里来实例化上面子类继承的内容DragDropTextEdit 的实例,并将其赋值给 self.textEdit 属性。这个文本编辑框支持文件拖拽功能。
        display.addWidget(self.textEdit)  # 将 self.textEdit 添加到垂直布局管理器 display 中
        # 按钮
        self.submit_Button = QPushButton('提交文件', self)  # 创建提交按钮的名称
        self.submit_Button.clicked.connect(self.processFiles)  # 给提交按钮绑定事件函数processFiles
        display.addWidget(self.submit_Button)  # 展示出提交按钮

    def processFiles(self):
        file_paths = self.textEdit.toPlainText().split('n')
        # 这里添加处理文件的代码
        print("文件路径如下:", file_paths)


def main():
    app = QApplication(sys.argv)
    ex = MainApp()
    ex.show()
    sys.exit(app.exec_())


if __name__ == '__main__':
    main()

界面如下:

Tkinter

注:Tkinter本身是不支持拖拽文件进入它的窗口的,所以需要借助pip install tkinterdnd2 这个包,然后from tkinterdnd2 import * 这样就可以使用拖拽文件到窗口功能了

相关代码如下:

代码语言:python代码运行次数:0复制
# -*- coding:utf-8 -*-
# @author:Ye Zhoubing
# @datetime:2024/6/30 14:15
# @software: PyCharm
"""
拖拽获取文件路径
todo:目前无法将文件拖入窗口,待解决
"""
import tkinter as tk
from tkinter import messagebox
from tkinterdnd2 import DND_FILES, TkinterDnD

root = TkinterDnD.Tk()

#获取拖拽的文件名称
def on_drop(event):
    file_path = event.data
    file_label.config(text="拖拽的文件: "   file_path)

#设置窗口标题
root.title("文件拖拽")
#设置窗口大小
root.geometry("300x300")

#设置拖拽功能
file_label = tk.Label(root, text="请把文件拖拽到这里:")
file_label.pack(pady=20)

# 允许文件拖入
root.drop_target_register(DND_FILES)


root.dnd_bind('<<Drop>>', on_drop)

root.mainloop()

参考链接

0 人点赞