当前位置: 首页 > news >正文

一步到位:用Python实现PC屏幕截图并自动发送邮件,实现屏幕监控

在当前的数字化世界中,自动化已经成为我们日常生活和工作中的关键部分。它不仅提高了效率,还节省了大量的时间和精力。在这篇文章中,我们将探讨如何使用Python来实现一个特定的自动化任务 - PC屏幕截图自动发送到指定的邮箱。

这个任务可能看起来很复杂,但是通过Python,我们可以将其分解为几个简单的步骤并逐一实现。首先,我们需要一个能够捕获屏幕截图的工具。其次,我们需要一个能够发送电子邮件的服务。最后,我们需要将这两个步骤组合在一起,创建一个可以自动执行这些任务的脚本。

在本文中,我们将详细介绍这个过程,并提供相应的Python代码示例。无论你是Python初学者,还是寻求新的自动化项目的经验开发人员,都可以从中受益。让我们开始吧。

主要功能

1.通过使用pyautogui库来进行屏幕截图。

2.使用smtplib库来发送电子邮件,以将截图发送给收件人。

3.使用tkinter库创建一个简单的图形用户界面(GUI),用于配置应用程序的设置。

4.通过使用logging库来记录日志,将日志保存到文件中。

5.使用configparser库来读取和保存应用程序的配置设置。

6.实现了开机自动启动功能,可以将应用程序设置为开机自动启动。

7.实现了隐藏和显示应用程序窗口的功能。

8.收件邮箱默认等于发件邮箱。

此外,代码还实现了一些其他功能,如数据加密和解密、删除已发送的截图文件等。

应用程序在为用户提供一个便捷的方式来定时截图并将截图发送给指定的收件人,适用于需要定期截图的监控、远程监视等场景。用户可以通过图形界面设置截图的间隔时间、截图的次数、发件人和收件人的电子邮件地址等。

具体代码

# coding=utf-8 
'''@Author  : TesterRoad@Time    : 2023/7/9 15:43@Desc    : 用python实现PC屏幕截图自动发送邮箱@Software: PyCharm
'''
import smtplib
import time
import pyautogui
from email.mime.multipart import MIMEMultipart
from email.mime.image import MIMEImage
from email.mime.text import MIMEText
import logging
import configparser
import os
import sys
import ctypes
from Crypto.Cipher import AES
from Crypto.Util.Padding import pad, unpad
import base64import tkinter as tk
from tkinter import ttk
import datetime
import threading
import winreg
import globKEY = b'MySuperSecretKey'def encrypt_data(data):cipher = AES.new(KEY, AES.MODE_CBC)ct_bytes = cipher.encrypt(pad(data.encode('utf-8'), AES.block_size))iv = base64.b64encode(cipher.iv).decode('utf-8')ct = base64.b64encode(ct_bytes).decode('utf-8')return iv + ctdef decrypt_data(data):try:iv = base64.b64decode(data[:24])ct = base64.b64decode(data[24:])cipher = AES.new(KEY, AES.MODE_CBC, iv=iv)pt = unpad(cipher.decrypt(ct), AES.block_size)return pt.decode('utf-8')except:return "Decryption Error!"class ScreenshotApp:def __init__(self):self.root = tk.Tk()self.root.title("Screen")self.config = configparser.ConfigParser()self.config_file = os.path.join(os.path.dirname(os.path.abspath(sys.argv[0])), "config.ini")if not os.path.exists(self.config_file):self.create_default_config()self.config.read(self.config_file)  # 读取配置文件self.sender_email_label = ttk.Label(self.root, text="发件邮箱:")self.sender_email_label.grid(row=0, column=0, padx=5, pady=5)self.sender_email_entry = ttk.Entry(self.root)self.sender_email_entry.grid(row=0, column=1, padx=5, pady=5)self.sender_password_label = ttk.Label(self.root, text="发件邮箱密码:")self.sender_password_label.grid(row=1, column=0, padx=5, pady=5)self.sender_password_entry = ttk.Entry(self.root, show="*")self.sender_password_entry.grid(row=1, column=1, padx=5, pady=5)self.interval_label = ttk.Label(self.root, text="截图间隔时间:")self.interval_label.grid(row=2, column=0, padx=5, pady=5)self.interval_entry = ttk.Entry(self.root)self.interval_entry.grid(row=2, column=1, padx=5, pady=5)self.count_label = ttk.Label(self.root, text="发送截图数量:")self.count_label.grid(row=3, column=0, padx=5, pady=5)self.count_entry = ttk.Entry(self.root)self.count_entry.grid(row=3, column=1, padx=5, pady=5)self.start_button = ttk.Button(self.root, text="开始截图", command=self.start_screenshot)self.start_button.grid(row=4, column=0, padx=5, pady=5)self.stop_button = ttk.Button(self.root, text="停止截图", command=self.stop_screenshot)self.stop_button.grid(row=4, column=1, padx=5, pady=5)self.stop_button.configure(state="disabled")self.save_button = ttk.Button(self.root, text="save", command=self.save_settings)self.save_button.grid(row=5, column=0, padx=5, pady=5)self.autostart_var = tk.BooleanVar()self.autostart_checkbutton = ttk.Checkbutton(self.root, text="开机自动启动", variable=self.autostart_var,command=self.save_settings)self.autostart_checkbutton.grid(row=6, column=0, columnspan=2, padx=5, pady=5)self.toggle_visibility_button = ttk.Button(self.root, text="显示/隐藏", command=self.toggle_visibility)self.toggle_visibility_button.grid(row=7, column=0, columnspan=2, padx=5, pady=5)# 创建日志记录器self.log_file_path = os.path.join(os.path.dirname(os.path.abspath(sys.argv[0])), "screenshot.log")self.logger = logging.getLogger("ScreenshotApp")self.logger.setLevel(logging.INFO)self.logger.addHandler(logging.FileHandler(self.log_file_path))self.screenshot_running = Falseself.screenshot_thread = Noneself.stop_event = threading.Event()# 初始化输入框的值self.sender_email_entry.insert(0, self.config.get("Settings", "sender_email", fallback=""))self.sender_password_entry.insert(0, self.get_decrypted_password())self.interval_entry.insert(0, self.config.get("Settings", "interval", fallback=""))self.count_entry.insert(0, self.config.get("Settings", "count", fallback=""))# 初始化开机自动启动选项self.autostart_var.set(self.is_autostart_enabled())self.root.protocol("WM_DELETE_WINDOW", self.on_close)self.root.bind("<F12>", self.toggle_visibility)# 初始化窗口可见性visibility = self.config.get("Settings", "visibility", fallback="visible")if visibility == "hidden":self.root.withdraw()if self.autostart_var.get():self.start_screenshot()self.root.mainloop()def on_close(self):self.stop_screenshot()self.save_settings()self.delete_screenshots()self.root.quit()def create_default_config(self):if not os.path.exists(self.config_file):self.config["Settings"] = {"sender_email": "","sender_password": "","interval": "","count": "","autostart": "False","visibility": "visible"}config_file_path = os.path.join(os.path.dirname(os.path.abspath(sys.argv[0])), "config.ini")with open(config_file_path, "w") as configfile:self.config.write(configfile)def start_screenshot(self):interval_text = self.interval_entry.get()count_text = self.count_entry.get()if not interval_text or not count_text:self.logger.error("请提供Screen间隔时间和Screen次数")returntry:interval = int(interval_text)count = int(count_text)except ValueError:self.logger.error("Screen间隔时间和Screen次数必须是有效的整数")returnif not self.screenshot_running:sender_email = self.sender_email_entry.get()sender_password = self.sender_password_entry.get()interval = int(self.interval_entry.get())count = int(self.count_entry.get())receiver_email = sender_email  # 收件邮箱地址默认等于发件邮箱地址self.logger.info("开始Screen")self.start_button.configure(state="disabled")self.stop_button.configure(state="normal")self.screenshot_running = Trueself.stop_event.clear()self.screenshot_thread = threading.Thread(target=self.screenshot_loop, args=(receiver_email, sender_email, sender_password, interval, count))self.screenshot_thread.start()def stop_screenshot(self):if self.screenshot_running:self.screenshot_running = Falseself.stop_event.set()self.screenshot_thread.join()self.logger.info("停止Screen")self.start_button.configure(state="normal")self.stop_button.configure(state="disabled")def screenshot_loop(self, receiver_email, sender_email, sender_password, interval, count):screenshot_count = 0screenshots = []# 获取用户主目录,并创建'Screenshots'文件夹user_dir = os.path.expanduser('~')screenshot_dir = os.path.join(user_dir, 'Screenshots')os.makedirs(screenshot_dir, exist_ok=True)# 在开始Screen前清空'Screenshots'文件夹self.delete_screenshots()while screenshot_count < count and not self.stop_event.is_set():try:# Screenscreenshot = pyautogui.screenshot()# 生成文件名,格式为“Screen时间.png”current_time = datetime.datetime.now().strftime("%Y%m%d_%H%M%S")filename = f"Screen_{current_time}.png"# 保存Screen到'Screenshots'文件夹中screenshot_path = os.path.join(screenshot_dir, filename)screenshot.save(screenshot_path)screenshots.append(screenshot_path)screenshot_count += 1# 设置文件为隐藏FILE_ATTRIBUTE_HIDDEN = 0x02ctypes.windll.kernel32.SetFileAttributesW(screenshot_path, FILE_ATTRIBUTE_HIDDEN)self.logger.info(f"Screen成功: {screenshot_path}")if screenshot_count == count:  # 达到指定Screen次数后发送Screenscreenshot_count = 0self.send_email(receiver_email, sender_email, sender_password, screenshots)self.logger.info(f"Screen发送成功,共发送了 {len(screenshots)} 张Screen")self.delete_screenshots(screenshots)screenshots = []  # 清空已发送的Screen列表except Exception as e:self.logger.error(f"Screen失败: {str(e)}")time.sleep(interval)def send_email(self, receiver_email, sender_email, sender_password, filenames):msg = MIMEMultipart()msg["From"] = sender_emailmsg["To"] = receiver_emailmsg["Subject"] = "Screen"# 添加邮件正文msg.attach(MIMEText("请查看附件中的Screen。", "plain"))# 添加Screen作为附件for filename in filenames:with open(filename, "rb") as f:image = MIMEImage(f.read())image.add_header('Content-Disposition', 'attachment', filename=os.path.basename(filename))msg.attach(image)try:# 发送邮件with smtplib.SMTP_SSL("smtp.qq.com", 465) as smtp:smtp.login(sender_email, sender_password)smtp.send_message(msg)self.logger.info(f"邮件发送成功,收件人: {receiver_email}")except Exception as e:self.logger.error(f"邮件发送失败: {str(e)}")def save_settings(self):self.config.set("Settings", "sender_email", self.sender_email_entry.get())self.config.set("Settings", "interval", self.interval_entry.get())self.config.set("Settings", "count", self.count_entry.get())self.config.set("Settings", "autostart", str(self.autostart_var.get()))visibility = "visible" if self.root.state() == "normal" else "hidden"self.config.set("Settings", "visibility", visibility)if self.sender_password_entry.get() != self.get_decrypted_password():encrypted_password = encrypt_data(self.sender_password_entry.get())self.config.set("Settings", "sender_password", encrypted_password)config_file_path = os.path.abspath(self.config_file)with open(config_file_path, "w") as configfile:self.config.write(configfile)self.logger.handlers.clear()self.logger.addHandler(logging.FileHandler(self.log_file_path))self.set_autostart(self.autostart_var.get())def delete_screenshots(self, filenames=None):# 获取'Screenshots'文件夹路径user_dir = os.path.expanduser('~')screenshot_dir = os.path.join(user_dir, 'Screenshots')if filenames is None:filenames = glob.glob(os.path.join(screenshot_dir, "Screen*.png"))for filename in filenames:try:os.remove(filename)self.logger.info(f"删除Screen: {filename}")except Exception as e:self.logger.error(f"删除Screen失败: {str(e)}")def get_decrypted_password(self):encrypted_password = self.config.get("Settings", "sender_password", fallback="")if encrypted_password:return decrypt_data(encrypted_password)else:return ""def toggle_visibility(self, event=None):if self.root.state() == "withdrawn":self.root.deiconify()else:self.root.withdraw()self.save_settings()def set_autostart(self, enabled):key = winreg.HKEY_CURRENT_USERrun_key = r"Software\Microsoft\Windows\CurrentVersion\Run"app_name = "Screen"app_path = sys.executable  # 获取当前脚本的绝对路径try:with winreg.OpenKey(key, run_key, 0, winreg.KEY_SET_VALUE) as reg_key:if enabled:winreg.SetValueEx(reg_key, app_name, 0, winreg.REG_SZ, app_path)self.logger.info("已设置开机自动启动")else:winreg.DeleteValue(reg_key, app_name)self.logger.info("已取消开机自动启动")except FileNotFoundError as e:self.logger.error(f"找不到注册表路径: {str(e)}")except PermissionError as e:self.logger.error(f"没有足够的权限访问注册表: {str(e)}")except Exception as e:self.logger.error(f"设置开机自动启动失败: {str(e)}")def is_autostart_enabled(self):key = winreg.HKEY_CURRENT_USERrun_key = r"Software\Microsoft\Windows\CurrentVersion\Run"app_name = "Screen"app_path = sys.executable  # 获取当前脚本的绝对路径try:with winreg.OpenKey(key, run_key, 0, winreg.KEY_READ) as reg_key:try:value, value_type = winreg.QueryValueEx(reg_key, app_name)return value == app_pathexcept FileNotFoundError:return Falseexcept FileNotFoundError as e:self.logger.error(f"找不到注册表路径: {str(e)}")except PermissionError as e:self.logger.error(f"没有足够的权限访问注册表: {str(e)}")except Exception as e:self.logger.error(f"读取开机自动启动设置失败: {str(e)}")return Falseif __name__ == "__main__":app = ScreenshotApp()

打开CMD,输入python ScreenCaptureSendEmail.py

图片

我们输入发件邮箱,发件邮箱密码(QQ邮箱则是授权码),截图间隔时间,发送截图数量,然后点击开始截图,稍后我们会收到一封QQ邮件

如下图所示

图片

图片

看到这里,我突然有个大胆的想法。

在本文中,我们详细讨论了如何使用Python实现PC屏幕截图并自动发送至邮箱的功能。我们探讨了相关库的使用。并通过编写实际的代码,我们一步步演示了如何将这些功能整合在一起。希望这篇文章能够帮助你在自动化处理任务、提高工作效率方面取得进步。如果你在实践过程中遇到任何问题,或者有任何建议,欢迎随时与我交流。记住,编程就是解决问题的艺术,不断学习和实践才能更好地掌握它。

最后感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:

这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你! 

相关文章:

一步到位:用Python实现PC屏幕截图并自动发送邮件,实现屏幕监控

在当前的数字化世界中&#xff0c;自动化已经成为我们日常生活和工作中的关键部分。它不仅提高了效率&#xff0c;还节省了大量的时间和精力。在这篇文章中&#xff0c;我们将探讨如何使用Python来实现一个特定的自动化任务 - PC屏幕截图自动发送到指定的邮箱。 这个任务可能看…...

Spring Boot+RocketMQ 实现多实例分布式环境下的事件驱动

为什么要使用MQ&#xff1f; 在Spring Boot Event这篇文章中已经通过Guava或者SpringBoot自身的Listener实现了事件驱动&#xff0c;已经做到了对业务的解耦。为什么还要用到MQ来进行业务解耦呢&#xff1f; 首先无论是通过Guava还是Spring Boot自身提供的监听注解来实现的事…...

oracle ORA-01704: string literal too long ORACLE数据库clob类型

当oracle数据表中有clob类型字段时候&#xff0c;insert或update的sql语句中&#xff0c;超过长度就会报错 ORA-01704: string literal too long update xxx set xxx <div><h1>123</h1></div> where id 100;可以修改为 DECLAREstr varchar2(10000…...

微星主板强刷BIOS(以微星X370gaming plus 为例)

(前两天手欠&#xff0c;用U盘通过微星的M-flash升级BIOS 升级过程中老没动静就强制关机了 然后电脑就打不开了) 几种强刷主板BIOS的方式 在网上看到有三种强刷BIOS的方式分别是: 使用夹子编程器 (听说不太好夹)使用微星转接线编程器&#xff08;只能用于微星主板&#xff0…...

matlab 图像上生成指定中心,指定大小的矩形窗

用matlab实现在图像上生成指定中心,指定大小的矩形窗(奇数*奇数) function PlaneWin PlaneWindow(CentreCoorX,CentreCoorY,RadiusX,RadiusY,SizeImRow,SizeImColumn) % 在图像上生成指定中心,指定大小的矩形窗(奇数*奇数) % % Input: % CentreCoorX(1*1) % CentreCoorY(1*1)…...

❀My学习小记录之算法❀

目录 算法:) 一、定义 二、特征 三、基本要素 常用设计模式 常用实现方法 四、形式化算法 五、复杂度 时间复杂度 空间复杂度 六、非确定性多项式时间&#xff08;NP&#xff09; 七、实现 八、示例 求最大值算法 求最大公约数算法 九、分类 算法:) 一、定义 …...

Hive-high Avaliabl

hive—high Avaliable ​ hive的搭建方式有三种&#xff0c;分别是 ​ 1、Local/Embedded Metastore Database (Derby) ​ 2、Remote Metastore Database ​ 3、Remote Metastore Server ​ 一般情况下&#xff0c;我们在学习的时候直接使用hive –service metastore的方式…...

码住!8个小众宝藏的开发者学习类网站

1、simplilearn simplilearn是全球排名第一的在线学习网站&#xff0c;它的课程由世界知名大学、顶级企业和领先的行业机构通过实时在线课程设计和提供&#xff0c;其中包括顶级行业从业者、广受欢迎的培训师和全球领导者。 2、VisuAlgo VisuAlgo是一个免费的在线学习算法和数…...

Postman常见问题及解决方法

1、网络连接问题 如果Postman无法发送请求或接收响应&#xff0c;可以尝试以下操作&#xff1a; 检查网络连接是否正常&#xff0c;包括检查网络设置、代理设置等。 确认请求的URL是否正确&#xff0c;并检查是否使用了正确的HTTP方法&#xff08;例如GET、POST、PUT等&#…...

ubuntu图形化登录默认只有guest session账号解决方法

新安装的ubuntu16.x 图形化界面登录默认只有guest账号&#xff0c;只有进入guest账号之后再去手动切换root账号很麻烦&#xff0c;但是这样确实很安全。为了方便希望能够在登录图形化界面的时候以root身份/或者自定义其他身份登录。做一下简单的记录。 使用终端命令行编辑文件…...

全国计算机等级考试| 二级Python | 真题及解析(1)

一、选择题 1. 按照“后进先出”原则组织数据的数据结构是____ A栈 B双向链表 C二叉树 D队列 正确答案: A 2. 以下选项的叙述中,正确的是 A在循环队列中,只需要队头指针就能反映队列中元素的动态变化情况 B在循环队列中,只需要队尾指针就能反映队列中元素的动态变…...

Java开发框架和中间件面试题(9)

目录 102.你了解秒杀吗&#xff1f;怎么设计&#xff1f; 103.什么是缓存穿透&#xff1f;怎么解决&#xff1f; 102.你了解秒杀吗&#xff1f;怎么设计&#xff1f; 1.设计难点&#xff1a;并发量大&#xff0c;应用&#xff0c;数据库都承受不了。另外难控制超卖。 2.设计…...

【ARMv8M Cortex-M33 系列 2 -- Cortex-M33 JLink 连接 及 JFlash 烧写介绍】

文章目录 Jlink 工具JLink 命令行示例JFlash 烧写问题Jlink 工具 J-Link 是 SEGGER 提供的一款流行的 JTAG 调试器,它支持多个平台和处理器。JLink.exe 是 J-Link 调试器的命令行接口,它允许用户通过命令行执行一系列操作,例如编程、擦除、调试等。 工具链接: https://ww…...

react pwa应用示例

创建一个基于React的PWA应用&#xff0c;你可以使用create-react-app&#xff0c;它自带PWA支持&#xff0c;但默认是关闭的。以下是创建React PWA应用的步骤&#xff1a; 安装create-react-app 如果你还没有安装&#xff0c;你可以通过npm来安装&#xff1a; npm install -…...

python如何通过日志分析加入黑名单

python通过日志分析加入黑名单 监控nginx日志&#xff0c;若有人攻击&#xff0c;则加入黑名单&#xff0c;操作步骤如下&#xff1a; 1.读取日志文件 2.分隔文件&#xff0c;取出ip 3.将取出的ip放入list&#xff0c;然后判读ip的次数 4.若超过设定的次数&#xff0c;则加…...

RabbitMq知识概述

本文来说下RabbitMq相关的知识与概念 文章目录 概述AMQP协议Exchange 消息如何保证100&#xff05;投递什么是生产端的可靠性投递可靠性投递保障方案 消息幂等性高并发的情况下如何避免消息重复消费confirm 确认消息、Return返回消息如何实现confirm确认消息return消息机制 消费…...

专业级A链接测试特有

A链接普通 A链接添加链接描述带有blank...

Spring Boot 入参校验及全局异常处理

版本依赖 JDK 17 Spring Boot 3.2.0 源码地址&#xff1a;Gitee Spring Boot validation spring-boot-starter-validation是基于hibernate-validator的实现&#xff0c;在Spring Boot项目中直接导入spring-boot-starter-validation即可。 Valid 和 Validated 的区别 适用范围…...

MySQL 和 MySQL2 的区别

MySQL是最流行的开源关系型数据库管理系统,拥有大量的使用者和广泛的应用场景。而MySQL2是MySQL官方团队推出的新一代MySQL驱动&#xff0c;用于取代老版的MySQL模块&#xff0c;提供更好的性能和更丰富的功能。 本文将介绍MySQL2相较于MySQL有哪些优势以及具体的技术区别。 …...

AutoCAD图纸打印后内容不见

用户反映&#xff0c;在CAD里的对象打印出来不显示。其实&#xff0c;这是在CAD的打印对象颜色的问题。&#xff08;在9.2以下版本有这种问题&#xff0c;9.2及以上版本已默认此种颜色&#xff09; 1.当背景色为黑色的时候&#xff0c;这里的颜色是白&#xff0c;如下图 2.当C…...

iOS 26 携众系统重磅更新,但“苹果智能”仍与国行无缘

美国西海岸的夏天&#xff0c;再次被苹果点燃。一年一度的全球开发者大会 WWDC25 如期而至&#xff0c;这不仅是开发者的盛宴&#xff0c;更是全球数亿苹果用户翘首以盼的科技春晚。今年&#xff0c;苹果依旧为我们带来了全家桶式的系统更新&#xff0c;包括 iOS 26、iPadOS 26…...

FFmpeg 低延迟同屏方案

引言 在实时互动需求激增的当下&#xff0c;无论是在线教育中的师生同屏演示、远程办公的屏幕共享协作&#xff0c;还是游戏直播的画面实时传输&#xff0c;低延迟同屏已成为保障用户体验的核心指标。FFmpeg 作为一款功能强大的多媒体框架&#xff0c;凭借其灵活的编解码、数据…...

python/java环境配置

环境变量放一起 python&#xff1a; 1.首先下载Python Python下载地址&#xff1a;Download Python | Python.org downloads ---windows -- 64 2.安装Python 下面两个&#xff0c;然后自定义&#xff0c;全选 可以把前4个选上 3.环境配置 1&#xff09;搜高级系统设置 2…...

JVM垃圾回收机制全解析

Java虚拟机&#xff08;JVM&#xff09;中的垃圾收集器&#xff08;Garbage Collector&#xff0c;简称GC&#xff09;是用于自动管理内存的机制。它负责识别和清除不再被程序使用的对象&#xff0c;从而释放内存空间&#xff0c;避免内存泄漏和内存溢出等问题。垃圾收集器在Ja…...

(转)什么是DockerCompose?它有什么作用?

一、什么是DockerCompose? DockerCompose可以基于Compose文件帮我们快速的部署分布式应用&#xff0c;而无需手动一个个创建和运行容器。 Compose文件是一个文本文件&#xff0c;通过指令定义集群中的每个容器如何运行。 DockerCompose就是把DockerFile转换成指令去运行。 …...

智能分布式爬虫的数据处理流水线优化:基于深度强化学习的数据质量控制

在数字化浪潮席卷全球的今天&#xff0c;数据已成为企业和研究机构的核心资产。智能分布式爬虫作为高效的数据采集工具&#xff0c;在大规模数据获取中发挥着关键作用。然而&#xff0c;传统的数据处理流水线在面对复杂多变的网络环境和海量异构数据时&#xff0c;常出现数据质…...

OPENCV形态学基础之二腐蚀

一.腐蚀的原理 (图1) 数学表达式&#xff1a;dst(x,y) erode(src(x,y)) min(x,y)src(xx,yy) 腐蚀也是图像形态学的基本功能之一&#xff0c;腐蚀跟膨胀属于反向操作&#xff0c;膨胀是把图像图像变大&#xff0c;而腐蚀就是把图像变小。腐蚀后的图像变小变暗淡。 腐蚀…...

Golang——6、指针和结构体

指针和结构体 1、指针1.1、指针地址和指针类型1.2、指针取值1.3、new和make 2、结构体2.1、type关键字的使用2.2、结构体的定义和初始化2.3、结构体方法和接收者2.4、给任意类型添加方法2.5、结构体的匿名字段2.6、嵌套结构体2.7、嵌套匿名结构体2.8、结构体的继承 3、结构体与…...

Caliper 配置文件解析:fisco-bcos.json

config.yaml 文件 config.yaml 是 Caliper 的主配置文件,通常包含以下内容: test:name: fisco-bcos-test # 测试名称description: Performance test of FISCO-BCOS # 测试描述workers:type: local # 工作进程类型number: 5 # 工作进程数量monitor:type: - docker- pro…...

Kubernetes 网络模型深度解析:Pod IP 与 Service 的负载均衡机制,Service到底是什么?

Pod IP 的本质与特性 Pod IP 的定位 纯端点地址&#xff1a;Pod IP 是分配给 Pod 网络命名空间的真实 IP 地址&#xff08;如 10.244.1.2&#xff09;无特殊名称&#xff1a;在 Kubernetes 中&#xff0c;它通常被称为 “Pod IP” 或 “容器 IP”生命周期&#xff1a;与 Pod …...