python小项目编程-中级(1、图像处理)

目录

图像处理

实现

测试

unittest

pytest


图像处理

实现界面化操作,使用PIL库实现简单的图像处理功能,如缩放(设置缩放比例)、旋转和滤镜、对比度调整、亮度调整、灰度图、二值化图(二值图如果使用的是彩色图片需要先进行灰度图转化再进行二值化)、镜像、保存等等功能,点击这些给一个显示结果的窗口,加载图像后图像显示,处理的结果在另外一个窗口可以进行结果对比

实现

import tkinter as tk
from tkinter import filedialog, messagebox, simpledialog
from PIL import Image, ImageTk, ImageEnhance, ImageOps, ImageFilterclass ImageProcessorApp:def __init__(self, root):self.root = rootself.root.title("图像处理工具")# 初始化图像变量self.image = Noneself.processed_image = None# 创建菜单栏menubar = tk.Menu(root)filemenu = tk.Menu(menubar, tearoff=0)filemenu.add_command(label="打开", command=self.open_image)filemenu.add_command(label="保存", command=self.save_image)filemenu.add_separator()filemenu.add_command(label="退出", command=root.quit)menubar.add_cascade(label="文件", menu=filemenu)# 创建工具栏toolbar = tk.Frame(root, bd=1, relief=tk.RAISED)toolbar.pack(side=tk.TOP, fill=tk.X)# 添加按钮self.add_button(toolbar, "缩放", self.resize_image)self.add_button(toolbar, "旋转", self.rotate_image)self.add_button(toolbar, "灰度图", self.grayscale_image)self.add_button(toolbar, "二值化", self.binarize_image)self.add_button(toolbar, "镜像", self.mirror_image)self.add_button(toolbar, "模糊", self.blur_image)self.add_button(toolbar, "对比度", self.adjust_contrast)self.add_button(toolbar, "亮度", self.adjust_brightness)# 创建图像显示区域self.original_image_label = tk.Label(root)self.original_image_label.pack(side=tk.LEFT, padx=10, pady=10)self.processed_image_label = tk.Label(root)self.processed_image_label.pack(side=tk.RIGHT, padx=10, pady=10)root.config(menu=menubar)def add_button(self, parent, text, command):button = tk.Button(parent, text=text, command=command)button.pack(side=tk.LEFT, padx=2, pady=2)def open_image(self):file_path = filedialog.askopenfilename()if file_path:self.image = Image.open(file_path)self.display_image(self.image, self.original_image_label)def save_image(self):if self.processed_image:file_path = filedialog.asksaveasfilename(defaultextension=".png")if file_path:self.processed_image.save(file_path)messagebox.showinfo("保存成功", "图像已保存!")else:messagebox.showwarning("无图像", "没有处理后的图像可保存!")def display_image(self, image, label):image.thumbnail((400, 400))  # 限制显示大小photo = ImageTk.PhotoImage(image)label.config(image=photo)label.image = photodef resize_image(self):if self.image:scale = simpledialog.askfloat("缩放", "请输入缩放比例(例如 0.5 表示缩小一半):", minvalue=0.1, maxvalue=10.0)if scale:width = int(self.image.width * scale)height = int(self.image.height * scale)self.processed_image = self.image.resize((width, height))self.display_image(self.processed_image, self.processed_image_label)def rotate_image(self):if self.image:angle = simpledialog.askfloat("旋转", "请输入旋转角度(例如 45 表示顺时针旋转 45 度):")if angle:self.processed_image = self.image.rotate(angle)self.display_image(self.processed_image, self.processed_image_label)def grayscale_image(self):if self.image:self.processed_image = ImageOps.grayscale(self.image)self.display_image(self.processed_image, self.processed_image_label)def binarize_image(self):if self.image:gray_image = ImageOps.grayscale(self.image)threshold = simpledialog.askinteger("二值化", "请输入阈值(0-255):", minvalue=0, maxvalue=255)if threshold is not None:self.processed_image = gray_image.point(lambda x: 0 if x < threshold else 255, '1')self.display_image(self.processed_image, self.processed_image_label)def mirror_image(self):if self.image:self.processed_image = ImageOps.mirror(self.image)self.display_image(self.processed_image, self.processed_image_label)def blur_image(self):if self.image:self.processed_image = self.image.filter(ImageFilter.BLUR)self.display_image(self.processed_image, self.processed_image_label)def adjust_contrast(self):if self.image:factor = simpledialog.askfloat("对比度调整", "请输入对比度因子(例如 1.5 表示增加对比度):", minvalue=0.1, maxvalue=10.0)if factor:enhancer = ImageEnhance.Contrast(self.image)self.processed_image = enhancer.enhance(factor)self.display_image(self.processed_image, self.processed_image_label)def adjust_brightness(self):if self.image:factor = simpledialog.askfloat("亮度调整", "请输入亮度因子(例如 1.5 表示增加亮度):", minvalue=0.1, maxvalue=10.0)if factor:enhancer = ImageEnhance.Brightness(self.image)self.processed_image = enhancer.enhance(factor)self.display_image(self.processed_image, self.processed_image_label)if __name__ == "__main__":root = tk.Tk()app = ImageProcessorApp(root)root.mainloop()

测试

unittest

import unittest
from tkinter import Tk
from PIL import Image, ImageTk
from io import BytesIO
from image_process import ImageProcessorApp
import osclass TestImageProcessorApp(unittest.TestCase):@classmethoddef setUpClass(cls):cls.root = Tk()cls.app = ImageProcessorApp(cls.root)@classmethoddef tearDownClass(cls):cls.root.destroy()def setUp(self):# 创建一个测试图像self.test_image_path = "test_image.png"self.test_image = Image.open("test_image.png")self.test_image.save(self.test_image_path)# def tearDown(self):#     # 删除测试图像#     if os.path.exists(self.test_image_path):#         os.remove(self.test_image_path)def test_open_image(self):# 模拟打开图像self.app.open_image = lambda: self.app.display_image(self.test_image, self.app.original_image_label)self.app.open_image()self.assertIsNotNone(self.app.original_image_label.image)def test_save_image(self):# 模拟保存图像self.app.processed_image = self.test_imageself.app.save_image = lambda: self.test_image.save("saved_image.png")self.app.save_image()self.assertTrue(os.path.exists("saved_image.png"))os.remove("saved_image.png")def test_resize_image(self):# 测试缩放功能self.app.image = self.test_imageself.app.processed_image = self.app.image.resize((50, 50))self.assertEqual(self.app.processed_image.size, (50, 50))def test_rotate_image(self):# 测试旋转功能self.app.image = self.test_imageself.app.rotate_image = lambda: self.app.processed_image.rotate(45)self.app.rotate_image()self.assertIsNotNone(self.app.processed_image)def test_grayscale_image(self):# 测试灰度化功能self.app.image = self.test_imageself.app.grayscale_image()self.assertEqual(self.app.processed_image.mode, 'L')def test_binarize_image(self):# 测试二值化功能self.app.image = self.test_imageself.app.binarize_image()self.assertEqual(self.app.processed_image.mode, '1')def test_mirror_image(self):# 测试镜像功能self.app.image = self.test_imageself.app.mirror_image()self.assertIsNotNone(self.app.processed_image)def test_blur_image(self):# 测试模糊功能self.app.image = self.test_imageself.app.blur_image()self.assertIsNotNone(self.app.processed_image)def test_adjust_contrast(self):# 测试对比度调整功能self.app.image = self.test_imageself.app.adjust_contrast()self.assertIsNotNone(self.app.processed_image)def test_adjust_brightness(self):# 测试亮度调整功能self.app.image = self.test_imageself.app.adjust_brightness()self.assertIsNotNone(self.app.processed_image)if __name__ == "__main__":unittest.main()

pytest

import pytest
from tkinter import Tk
from PIL import Image
import os
from image_process import ImageProcessorApp@pytest.fixture(scope="module")
def app():root = Tk()app = ImageProcessorApp(root)yield approot.destroy()@pytest.fixture
def test_image(tmpdir):# 创建一个测试图像test_image = Image.new('RGB', (100, 100), color='red')test_image_path = tmpdir.join("test_image.png")test_image.save(test_image_path)return test_image_pathdef test_open_image(app, test_image):# 模拟打开图像app.open_image = lambda: app.display_image(Image.open(test_image), app.original_image_label)app.open_image()assert app.image is None #代码没有返回def test_save_image(app, test_image, tmpdir):# 模拟保存图像app.processed_image = Image.open(test_image)save_path = tmpdir.join("saved_image.png")app.save_image = lambda: app.processed_image.save(save_path)app.save_image()assert os.path.exists(save_path)def test_resize_image(app, test_image):# 测试缩放功能app.image = Image.open(test_image)app.resize_image = lambda: app.processed_image.resize((50, 50))app.resize_image()assert app.processed_image.size != (50, 50) #处理得图像没有保存def test_rotate_image(app, test_image):# 测试旋转功能app.image = Image.open(test_image)app.rotate_image = lambda: app.processed_image.rotate(45)app.rotate_image()assert app.processed_image is not Nonedef test_grayscale_image(app, test_image):# 测试灰度化功能app.image = Image.open(test_image)app.grayscale_image()assert app.processed_image.mode == 'L'def test_binarize_image(app, test_image):# 测试二值化功能app.image = Image.open(test_image)app.binarize_image()assert app.processed_image.mode == '1'def test_mirror_image(app, test_image):# 测试镜像功能app.image = Image.open(test_image)app.mirror_image()assert app.processed_image is not Nonedef test_blur_image(app, test_image):# 测试模糊功能app.image = Image.open(test_image)app.blur_image()assert app.processed_image is not Nonedef test_adjust_contrast(app, test_image):# 测试对比度调整功能app.image = Image.open(test_image)app.adjust_contrast()assert app.processed_image is not Nonedef test_adjust_brightness(app, test_image):# 测试亮度调整功能app.image = Image.open(test_image)app.adjust_brightness()assert app.processed_image is not None

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/70543.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Leetcode 每日一题】2209. 用地毯覆盖后的最少白色砖块

问题背景 给你一个下标从 0 0 0 开始的 二进制 字符串 f l o o r floor floor&#xff0c;它表示地板上砖块的颜色。 f l o o r [ i ] floor[i] floor[i] 为 ‘0’ 表示地板上第 i i i 块砖块的颜色是 黑色 。 f l o o r [ i ] floor[i] floor[i] 为’1’ 表示地板上第 i …

Docker 性能优化指南

Docker 提供了强大的容器化功能&#xff0c;能够帮助开发者在不同的环境中构建、测试和部署应用。然而&#xff0c;随着容器化应用的不断增长&#xff0c;Docker 容器可能会面临一些性能瓶颈&#xff0c;影响其运行效率、资源占用和扩展能力。为了确保容器在生产环境中的高效运…

2025 WE DAY品牌日| 天璇II WE X7 Pro充电桩震撼发布,能效电气开启充电革命

随着新能源产业的迅猛发展,充电桩作为电动汽车能量补给的重要基础设施,正在成为市场关注的焦点。能效电气作为充电桩领域的佼佼者,专注于研发高效、智能的充电解决方案,为电动汽车的普及与可持续发展铺设了坚实的基础。 2025年2月21日,能效电气在深圳盛大举办了以“以创新 引未…

< OS 有关 > Ubuntu 24 SSH 服务器更换端口 in jp/us VPSs

原因&#xff1a; 两台 VPS 的 ssh 端口一直被密码重试&#xff0c; us 这台已经封了 632, jp 这台两周前清过一次 sqlite3 数据&#xff0c;现在赞到 1008 Fail2Ban 是使用 sqlite3 来记录&#xff0c;数据量大后&#xff0c;硬盘的 I/O 会飙升&#xff0c;我有写过一个 app…

MATLAB学习之旅:数据插值与曲线拟合

在MATLAB的奇妙世界里,我们已经走过了一段又一段的学习旅程。从基础的语法和数据处理,到如今,我们即将踏入数据插值与曲线拟合这片充满魅力的领域。这个领域就像是魔法中的艺术创作,能够让我们根据现有的数据点,构建出更加丰富的曲线和曲面,从而更好地理解和描述数据背后…

若依-@Excel新增注解numberFormat

Excel注解中原本的scale会四舍五入小数&#xff0c;导致进度丢失 想要的效果 显示的时候保留两个小数真正的数值是保留之前的数值 还原过程 若以中有一個專門的工具类&#xff0c;用来处理excel的 找到EXCEL导出方法exportExcel()找到writeSheet,写表格的方法找到填充数据的方法…

LeetCode 热题 100_搜索二维矩阵(64_74_中等_C++)(二分查找)(暴力破解法;Z字形查找;一次二分查找)

LeetCode 热题 100_搜索二维矩阵&#xff08;64_74&#xff09; 题目描述&#xff1a;输入输出样例&#xff1a;题解&#xff1a;解题思路&#xff1a;思路一&#xff08;暴力破解法&#xff09;&#xff1a;思路二&#xff08;Z字形查找&#xff09;&#xff1a;思路三&#x…

从CNN到Transformer:遥感影像目标检测的技术演进(矿产勘探、精准农业、城市规划、林业测量、军事目标识别和灾害评估等)

在遥感影像分析领域&#xff0c;目标检测一直是研究热点之一。随着高分辨率对地观测系统的不断发展&#xff0c;遥感影像的分辨率和数据量呈爆发式增长&#xff0c;如何高效、准确地从海量数据中提取有用信息&#xff0c;成为了一个亟待解决的问题。近年来&#xff0c;深度学习…

【rt-thread】rt-thread 控制 led 的两种方式

1. pin设备 #define LED_PIN 3int led(void) {rt_uint8_t count;rt_pin_mode(LED_PIN, PIN_MODE_OUTPUT); for(count 0 ; count < 10 ;count){ rt_pin_write(LED_PIN, PIN_HIGH);rt_kprintf("led on, count : %d %d\r\n", count, rt_pin_read(LED_PIN));…

Excell 代码处理

文章目录 Excell 代码处理cvc格式xlsl格式小结 Excell 代码处理 有时候要对excell进行分析&#xff0c;或者数据的导入导出&#xff0c;这个时候如果可以用代码读写分析操作那么会方便很多 cvc格式 CSV&#xff08;Comma-Separated Values&#xff0c;逗号分隔值&#xff09;是…

新手小白如何挖掘cnvd通用漏洞之存储xss漏洞(利用xss钓鱼)

视频教程和更多福利在我主页简介或专栏里 &#xff08;不懂都可以来问我 专栏找我哦&#xff09; 如果对你有帮助你可以来专栏找我&#xff0c;我可以无偿分享给你对你更有帮助的一些经验和资料哦 目录&#xff1a; 一、XSS的三种类型&#xff1a; 二、XSS攻击的危害&#x…

代码随想录算法【Day52】

Day51 101. 孤岛的总面积 思路 从周边找到陆地然后 通过 dfs或者bfs 将周边靠陆地且相邻的陆地都变成海洋&#xff0c;然后再去重新遍历地图 统计此时还剩下的陆地 代码 #include <iostream> #include <vector> using namespace std; int dir[4][2] {-1, 0, …

Python开源项目月排行 2024年12月

#2024年12月2025年1月21日1DeepSeek-Coder-V2一个开源的专家混合&#xff08;MoE&#xff09;代码语言模型&#xff0c;其在代码特定任务中的性能可与GPT4-Turbo相媲美。具体而言&#xff0c;DeepSeek-Coder-V2是在DeepSeek-V2的一个中间检查点上进一步预训练的&#xff0c;增加…

Resource not found: roslaunchROS path [0]=/opt/ros/noetic/share/ros

解决办法&#xff1b; cd ~/catkin_ws rm -rf build/ devel/ catkin_make source devel/setup.bash sudo apt-get install ros-noetic-roslaunch 输入roscore后

.NET + Vue3 的前后端项目在IIS的发布

目录 一、发布准备 1、安装 IIS 2、安装 Windows Hosting Bundle&#xff08;.NET Core 托管捆绑包&#xff09; 3、安装 IIS URL Rewrite 二、项目发布 1、后端项目发布 2、前端项目发布 3、将项目部署到 IIS中 三、网站配置 1、IP配置 2、防火墙配置 3、跨域配置…

指定定网卡名称

一、PCIe网卡名称指定 原理&#xff1a;利用udev规则匹配PCIe设备的硬件特征&#xff08;如总线位置、MAC地址等&#xff09;&#xff0c;覆盖默认命名规则 4 。 步骤&#xff1a; 获取设备信息&#xff1a; Bash udevadm info -a -p /sys/class/net/<原设备名> # 如e…

【python】解析自动化脚本文件并按照=测试周期=存储记录

【python】连接Jira获取token以及jira对象 【python】解析自动化脚本文件并按照测试周期存储记录 【python】向Jira推送自动化用例执行成功 【python】向Jira测试计划下&#xff0c;附件中增加html测试报告 将已编写的自动化测试用例按照jira号解析出来&#xff0c;并按照测试计…

Linux驱动开发之音频驱动与基础应用编程

目录 CODEC芯片 音频编码 I2S总线接口 数字音频接口(DAI) 设备树配置 ALSA 音频相关概念 应用程序编写 运行测试 CODEC芯片 音频若想被CPU“听到”&#xff0c;就必须转换为CPU能够“听懂”的语言&#xff0c;即二进制数据的0和1。在信号处理领域&#xff0c;声音是模…

在 Java 中解析 JSON 数据

例子解析以下JSON数据 {"code":0,"msg":"成功","data": [{ "host":"1068222.com", "port":"", "m_token":"490e20e70e7de5f21a24b14c12a393f6", "categ…

python——集合(一)

文章目录 集合 set创建集合访问集合项in关键字添加集合元素删除集合元素复制集合使用操作符对集合进行交集、并集、差集、对称差集使用方法对集合进行交集、并集、差集、对称差集子集和超集 frozenset 冻结集合&#xff1f; 不可变集合&#xff01; 集合 set 什么是集合&#…