机器视觉 AI 数据集制作

工业中,机器视觉物体分拣时,需要制作,数据集,那么,一般情况下,可以选择几个物体的几张图片,或者视频,将待识别的物体的掩模扣取出来,随机的贴在 传送带背景中,并批量自动的写成 VOC 数据集

在这里插入图片描述

使用图像处理的技术手段,将上述的目标的掩模扣取出来,或者使用 ps 的技术扣取掩模均可。

# -*- coding :  utf-8 -*-
# @Data      :  2019-08-16
# @Author    :  xm
# @Email     :
# @File      :  image_process.py
# Desctiption:  求取图像中物体的边界矩形import numpy as np
import cv2
import osdef calculatBoundImage(src_Image):"""求取图像中物体的边界矩形框:param src_Image: 输出的源图像:return: 返回图像中的物体边界矩形"""tmp_image = src_Image.copy()#print(tmp_image)if (len(tmp_image.shape) == 3):tmp_image = cv2.cvtColor(tmp_image, cv2.COLOR_BGR2GRAY)# 自适应阈值进行二值化thresh_image = cv2.adaptiveThreshold(tmp_image, 255, cv2.ADAPTIVE_THRESH_GAUSSIAN_C, cv2.THRESH_BINARY_INV, 71, 10)thresh_image = cv2.morphologyEx(thresh_image, cv2.MORPH_CLOSE, cv2.getStructuringElement(cv2.MORPH_RECT, (25, 25)))# 寻找最外层轮廓contours_ls, hierarchy = cv2.findContours(thresh_image, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_TC89_KCOS)pnt_cnt_ls = np.array([tmp_contour.shape[0] for tmp_contour in contours_ls])contour_image = src_Image.copy()contours_idx = np.argmax(pnt_cnt_ls)contour_image = cv2.drawContours(contour_image, contours_ls, contours_idx, (0, 0, 255))longest_contour = contours_ls[contours_idx]countour_image_gray = np.zeros(src_Image.shape, dtype=np.uint8)countour_image_gray = cv2.drawContours(countour_image_gray, contours_ls, contours_idx, (1, 1, 1), cv2.FILLED)obj_image = src_Image * countour_image_graybound_box = cv2.boundingRect(longest_contour)return bound_box, contour_image, obj_imagedef rotateImage(src_Image, angle_deg, rotate_center=None):"""对目标图像进行旋转:param src_Image: 输入的源图像:param angle_deg: 旋转的角度:param rotate_center: 旋转的中心:return: 旋转后的图片"""(h, w) = src_Image.shape[:2]if rotate_center is None:rotate_center = ((w -1) / 2, (h - 1) / 2)rot_mat = cv2.getRotationMatrix2D(rotate_center, angle_deg, 1.0)rot_iamge = cv2.warpAffine(src_Image, rot_mat, (w, h))return rot_iamgedef VideotoImage(video_file, folder_path):"""数据的视频保存为提取之后的物体图:param video_file: 视频文件:param folder_path: 保存图片的路径:return: 保存的图片"""video_cap = cv2.VideoCapture(video_file)image_idx = 2000while True:ret, frame = video_cap.read()if (frame is None):continuebound_box, contour_image, obj_image = calculatBoundImage(frame)bound_thres = 4500if (bound_box[2] > bound_thres or bound_box[3] > bound_thres):continuecontour_image = cv2.rectangle(contour_image, (bound_box[0], bound_box[1]),(bound_box[0] + bound_box[2],bound_box[1] + bound_box[3]), (225, 0, 0), thickness=2)#cv2.imshow('frame', contour_image)image_name = str(image_idx).zfill(6) + '.jpg'image_idx += 1if image_idx % 2 == 0:cv2.imwrite(folder_path + image_name, obj_image)cv2.waitKey(25)if 0xFF & cv2.waitKey(5) == 27:breakvideo_cap.release()def BatchImageProcess(image_path, folder_path):"""批量图片物体提取,背景为黑色:param Image_path: 图片的路径:param folder_path: 图像处理之后的保存路径:return: 保存的图片"""image_file_list = os.listdir(image_path)# 获取物体图像的文件名image_idx = 0for image_name in range(len(image_file_list)):obj_image_path = image_path + image_file_list[image_idx]src_Image = cv2.imread(obj_image_path)bound_box, contour_image, obj_image = calculatBoundImage(src_Image)bound_thres = 4500if (bound_box[2] > bound_thres or bound_box[3] > bound_thres):continuecontour_image = cv2.rectangle(contour_image, (bound_box[0], bound_box[1]), (bound_box[0] + bound_box[2], bound_box[1] + bound_box[3]), (225, 0, 0), thickness=2)#cv2.imshow('frame', contour_image)image_name = str(image_idx).zfill(6) + '.jpg'cv2.imwrite(folder_path + image_name, obj_image)image_idx += 1def main():image_path = "/home/xm/workspace/ImageProcess/tmp/circle/"folder_path = "/home/xm/workspace/ImageProcess/tmp/"BatchImageProcess(image_path, folder_path)# def main():
#     src_Image = cv2.imread("./Images/00001.png")
#     bound_box, contour_image, obj_image = calculatBoundImage(src_Image)
#     print("bound_box", bound_box)
#
#     cv2.namedWindow("input image", cv2.WINDOW_AUTOSIZE)
#     cv2.imshow("input image", contour_image)
#
#
#     # 一般源图像进行旋转再提取轮廓
#     rot_image = rotateImage(src_Image, 20, rotate_center=None)
#     cv2.imshow("obj image", obj_image)
#     cv2.imshow("rot image", rot_image)
#     cv2.waitKey(0)
#
#     # vide_file = "./Images/blue_1_82.mp4"
#     # folder_path = "./results/"
#     #
#     # VideotoImage(vide_file, folder_path)if __name__ == "__main__":main()

在这里插入图片描述

# -*- coding :  utf-8 -*-
# @Data      :  2019-08-17
# @Author    :  xm
# @Email     :
# @File      :  ImageDataSetGeneration.py
# Desctiption:  生成物体分类的图像数据集import numpy as np
import cv2
import os
from lxml import etree, objectifydef rotateImage(src_image, rotate_deg):"""对图像进行旋转:param src_image: 输入源图像:param rotate_dog: 旋转角度:return: 旋转后的图像"""img_h, img_w = src_image.shape[0:2]rotate_mat = cv2.getRotationMatrix2D((img_w / 2.0, img_h / 2.0), rotate_deg, 1.0)dst_image = cv2.warpAffine(src_image, rotate_mat, (img_w, img_h))return dst_imagedef calculateBoundImage(src_image):"""求图像中物体的边界矩形:param src_image: 源图像:return: 图像中物体的边界矩形、轮廓图、目标图像"""tmp_image = src_image.copy()if len(tmp_image.shape) == 3:tmp_image = cv2.cvtColor(tmp_image, cv2.COLOR_BGR2GRAY)ret, thresh_images = cv2.threshold(tmp_image, 0, 255,cv2.THRESH_BINARY)contours_ls, _ = cv2.findContours(thresh_images, cv2.RETR_EXTERNAL,cv2.CHAIN_APPROX_SIMPLE)all_points = np.concatenate(contours_ls, axis=0)bound_box = cv2.boundingRect(all_points)return bound_boxdef randomMoveObjectInImage(src_image, src_bound_box):"""将物体在图像中随机摆放:param src_image: 背景图 COCO/VOC:param src_bound_box: 原始边界框:return: 相机旋转后的边界框"""x, y, w, h = src_bound_boximg_h, img_w = src_image.shape[0:2]img_h -= himg_w -= wrandom_array = np.random.uniform(0.0, 1.0, 2)bbox_x = np.floor(img_w * random_array[0])bbox_y = np.floor(img_h * random_array[1])return np.array([bbox_x, bbox_y, w, h])def calculateIOU(bound_box_1, bound_box_2):"""计算两个 bound_box 之间的 IOU:param bound_box_1: 边界框 1, shape [x, y, w, h]:param bound_box_2: 边界框 2,shape [x, y, w, h]:return: 两个 bound box 之间的 IOU 值"""min_xy = np.maximum(bound_box_1[0:2], bound_box_2[0:2])max_xy = np.minimum(bound_box_1[0:2] + bound_box_2[2:4],bound_box_2[0:2] + bound_box_2[2:4])delta_xy = max_xy - min_xyintersection_area = delta_xy[0] * delta_xy[1]if (intersection_area < 0):returnbox_area_1 = bound_box_1[2] * bound_box_1[3]box_area_2 = bound_box_2[2] * bound_box_2[3]union_area = box_area_1 + box_area_2 - intersection_areareturn intersection_area / union_areadef resizeObjectImage(src_image, max_min_box_size):"""对物体图像进行随机缩放:param src_image: 原始图像:param max_min_box_size: 缩放后图像中的物体的 bound box 的最大边的范围:return: 缩放后的图像"""src_bbox = calculateBoundImage(src_image)src_bbox_max = np.max(src_bbox[2:4])cur_bbox_max = np.random.uniform(max_min_box_size[1], max_min_box_size[0], 1)[-1]cur_ratio = cur_bbox_max / src_bbox_maxsrc_h, src_w = src_image.shape[0:2]dst_h, dst_w = np.floor(src_h * cur_ratio), np.floor(src_w * cur_ratio)dst_image = cv2.resize(src_image, (np.int(dst_w), np.int(dst_h)))return dst_imagedef addObjectToImage(backgroup_image, obj_image, bound_box):"""将目标物体添加到背景图中:param backgroup_image: 背景图:param obj_image: 目标物体图:param bound_box: 边界矩形框:return: 添加了目标物体的背景图"""tmp_image = obj_image.copy()if len(tmp_image.shape) == 3:tmp_image = cv2.cvtColor(tmp_image, cv2.COLOR_BGR2GRAY)mask = tmp_image > 5min_x, min_y, max_x, max_y = bound_box[0], bound_box[1], bound_box[0] + bound_box[2], bound_box[1] + bound_box[3]backgroup_image[np.int(min_y):np.int(max_y), np.int(min_x):np.int(max_x)][mask] = obj_image[mask]return backgroup_imagedef formImageAndlabel(background_image, obj_ls, max_min_size_ration, iou_thres):"""形成训练图像,并生成对应的 label 列表:param background_image: 输入背景图:param obj_ls: 目标 list:param max_min_size_ration: 最大最小旋转角度:param iou_thres: IOU 阈值:return: 返训练的图像,对应的 label"""max_ratio, min_ratio = max_min_size_rationimage_size = np.min(background_image.shape[0:2])dst_image = background_image.copy()max_min_box_size = [np.floor(max_ratio * image_size), np.floor(min_ratio * image_size)]label_ls = []for obj_image, obj_name in obj_ls:# 对图像进行随机缩放resize_obj_image = resizeObjectImage(obj_image, max_min_box_size)# 对图像进行随机旋转rotate_image = rotateImage(resize_obj_image, np.random.uniform(0, 360, 1)[-1])# 多次迭代, 直到将图像平移到适当位置为止src_bbox = calculateBoundImage(rotate_image)sub_obj_image = rotate_image[src_bbox[1]:src_bbox[1] + src_bbox[3], src_bbox[0]:src_bbox[0] + src_bbox[2]]iter_cnt = 100if len(label_ls) == 0:iter_cnt = 1for iter_idx in range(iter_cnt):dst_bbox = randomMoveObjectInImage(dst_image, src_bbox)if len(label_ls) != 0:is_fit = Truefor tmp_box, tmp_obj_name in label_ls:#print("....", tmp_box)#print("+++++", dst_bbox)IOU = calculateIOU(tmp_box, dst_bbox)if (IOU is not None) and (IOU > iou_thres):is_fit = Falsebreakif is_fit == False:continueelse:breakdst_image = addObjectToImage(dst_image, sub_obj_image, dst_bbox)label_ls.append([dst_bbox, obj_name])return dst_image, label_lsdef formImageLableXML(src_image, image_file_name, label_info, label_path):"""生成图片的 label XML:param src_image: 原始图像:param image_file_name: 图像的文件名:param label_infor: 标签信息:param label_path: 标签的路径:return: XML"""ele = objectify.ElementMaker(annotate=False)anno_tree = ele.annotation(ele.folder('VOC2019_xm'),ele.filename(image_file_name),ele.source(ele.database('The VOC2019 Database'),ele.annotation('PASCAL VOC2019'),ele.image('flickr'),ele.flickrid('264265361')),ele.owner(ele.flickrid('xm'),ele.name('xm')),ele.size(ele.width(str(src_image.shape[0])),ele.height(str(src_image.shape[1])),ele.depth(str(src_image.shape[2]))),ele.segmented('0'))for cur_box, cur_obj_name in label_info:cur_ele = objectify.ElementMaker(annotate=False)cur_tree = cur_ele.object(ele.name(cur_obj_name),ele.pose('Frontal'),ele.truncated('0'),ele.difficult('0'),ele.bndbox(ele.xmin(str(cur_box[0])),ele.ymin(str(cur_box[1])),ele.xmax(str(cur_box[0] + cur_box[2])),ele.ymax(str(cur_box[1] + cur_box[3]))))anno_tree.append(cur_tree)etree.ElementTree(anno_tree).write(label_path, pretty_print=True)def main():obj_name_ls = ['circle', 'square']# 各种物体对应的图像的路径base_obj_file_name = '/home/xm/workspace/ImageProcess/DataSet/'obj_file_name = [base_obj_file_name + cur_obj for cur_obj in obj_name_ls]print(obj_file_name)# 每个种类的样本数量obj_count = 600# 图像中物体最大的数量image_max_obj_cnt = 2# 图像中物体的 bound box 的最大尺寸点,整个图像最小尺寸比例,max_size_radio = 0.45min_size_radio = 0.20# 图像的总数image_count = len(obj_name_ls) * 600# 数据集的保存路径dataset_basic_path = '/home/xm/workspace/ImageProcess/COCO/VOCdevkit/VOC2019/'image_folder = dataset_basic_path + 'JPEGImages/'#print(image_folder)label_folder = dataset_basic_path + 'Annotations/'#print(label_folder)image_set_folder = dataset_basic_path + 'ImageSets/Main/'#print(image_set_folder)for data_idx in range(image_count):# 获取 VOC 数据集中图像文件夹中所有文件的名称voc_folder_dir = '/home/xm/workspace/ImageProcess/VOC'voc_image_file_list = os.listdir(voc_folder_dir)#获取物体图像的文件名列表obj_image_ls_ls = []for obj_image_dir in obj_name_ls:cur_image_dir = base_obj_file_name + obj_image_dirobj_image_ls_ls.append(os.listdir(cur_image_dir))# 随机取一张 VOC 图做背景background_image_file = voc_image_file_list[np.random.randint(0, len(voc_image_file_list), 1)[-1]]background_image_file = voc_folder_dir + '/' + background_image_filebackground_image = cv2.imread(background_image_file)# 随机取若干物体obj_image_name_ls = []obj_cnt = np.random.randint(1, image_max_obj_cnt, 1)[-1]for obj_idx in range(obj_cnt):cur_obj_idx = np.random.randint(0, len(obj_image_ls_ls), 1)[-1]cur_obj_image_ls = obj_image_ls_ls[cur_obj_idx]cur_obj_file = cur_obj_image_ls[np.random.randint(0, len(cur_obj_image_ls), 1)[-1]]cur_obj_image = cv2.imread(base_obj_file_name + obj_name_ls[cur_obj_idx] + '/' + cur_obj_file)obj_image_name_ls.append([cur_obj_image, obj_name_ls[cur_obj_idx]])# 随机生成图像get_image, label_ls = formImageAndlabel(background_image, obj_image_name_ls, [max_size_radio, min_size_radio], iou_thres=0.05)## # 保存图像与标签cur_image_name = str(data_idx).zfill(6) + '.jpg'#print(cur_image_name)cur_label_name = str(data_idx).zfill(6) + '.xml'#print(cur_label_name)cv2.imwrite(image_folder + cur_image_name, get_image)formImageLableXML(get_image, cur_image_name, label_ls, label_folder + cur_label_name)for obj_bbox, obj_name in label_ls:pnt_1 = tuple(map(int, obj_bbox[0:2]))pnt_2 = tuple(map(int, obj_bbox[0:2]))cv2.rectangle(get_image, pnt_1, pnt_2, (0, 0, 255))print(cur_image_name)cv2.imshow("get image", get_image)cv2.waitKey(10)train_set_name = 'train.txt'train_val_name = 'val.txt'test_set_name = 'test.txt'idx_thre = np.floor(0.6 * image_count)idx_thre_ = np.floor(0.8 * image_count)train_file = open(image_set_folder + train_set_name, 'w')for line_idx in range(int(idx_thre)):line_str = str(line_idx).zfill(6) + '\n'train_file.write(line_str)train_file.close()train_val_file = open(image_set_folder + train_val_name, 'w')for line_idx in range(int(idx_thre), int(idx_thre_)):line_str = str(line_idx).zfill(6) + '\n'train_val_file.write(line_str)train_val_file.close()test_file = open(image_set_folder + test_set_name, 'w')for line_idx in range(int(idx_thre_), image_count):line_str = str(line_idx).zfill(6) + '\n'test_file.write(line_str)test_file.close()if __name__ == '__main__':main()

在这里插入图片描述

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

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

相关文章

VMware上面安装部署centos7镜像系统【详细含镜像】

VMware上面安装部署centos7镜像系统【详细含镜像】 废话不多说直接开始 下载centos7镜像 网上有好多&#xff0c;但是我相信来看小编文章的基本上应该都有centos7的镜像了吧&#xff0c;毕竟咱们都是同一类人&#xff0c;哈哈不卖关子了&#xff0c;小编直接给大家一个百度云盘…

深度学习大数据物流平台 python 计算机竞赛

文章目录 0 前言1 课题背景2 物流大数据平台的架构与设计3 智能车货匹配推荐算法的实现**1\. 问题陈述****2\. 算法模型**3\. 模型构建总览 **4 司机标签体系的搭建及算法****1\. 冷启动**2\. LSTM多标签模型算法 5 货运价格预测6 总结7 部分核心代码8 最后 0 前言 &#x1f5…

【日常总结】优雅升级Swagger 2 升至 3.0, 全局设置 content-type application/json

目录 一、场景 二、问题 三、解决方案 四、延伸 上一节&#xff1a;【日常总结】Swagger-ui 导入 showdoc &#xff08;优雅升级Swagger 2 升至 3.0&#xff09;-CSDN博客 一、场景 接上一节&#xff1a;在 Swagger3Config extends WebMvcConfigurationSupport&#xff0c…

P2704 [NOI2001] 炮兵阵地 题解

P2704 题目题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1提示 解题思路分析Code更多方法 题目 原题链接 题目描述 司令部的将军们打算在 N M N\times M NM 的网格地图上部署他们的炮兵部队。 一个 N M N\times M NM 的地图由 N N N 行 M M M 列组成&#x…

BP算法推导

例子1&#xff1a; 解&#xff1a; 首先有如下结论&#xff1a; E k 1 2 ∑ k 1 n ( z k − f k ( x k ) ) 2 \color{green} E_k \frac{1}{2}\sum_{k1}^{n} ( z_k - f_k(x_k))^2 Ek​21​k1∑n​(zk​−fk​(xk​))2 z k ∑ i 1 , j 1 3 , 2 y i v i j z_k \sum_{i1,j …

C#常见的设计模式-创建型模式

引言 在软件开发过程中&#xff0c;设计模式是一种被广泛采用的思想和实践&#xff0c;可以提供一种标准化的解决方案&#xff0c;以解决特定问题。设计模式分为三种类型&#xff1a;创建型模式、结构型模式和行为型模式。本篇文章将重点介绍C#中常见的创建型模式。 目录 引言…

数据结构与算法编程题26

计算二叉树深度 #define _CRT_SECURE_NO_WARNINGS#include <iostream> using namespace std;typedef char ElemType; #define ERROR 0 #define OK 1 #define Maxsize 100 #define STR_SIZE 1024typedef struct BiTNode {ElemType data;BiTNode* lchild, * rchild; }BiTNo…

传音荣获2023首届全国人工智能应用场景创新挑战赛“智能家居专项赛”三等奖

近日&#xff0c;中国人工智能学会与科技部新一代人工智能发展研究中心联合举办2023首届全国人工智能应用场景创新挑战赛&#xff08;2023 1st China’s Innovation Challenge on Artificial Intelligence Application Scene&#xff0c;以下简称CICAS 2023)&#xff0c;吸引了…

供配电系统智能化监控

供配电系统智能化监控是指利用先进的监测技术、自动化控制技术、计算机网络技术等&#xff0c;对供配电系统进行实时、全方位的监测和控制&#xff0c;以实现供配电系统的安全、稳定、高效运行。 供配电系统智能化监控的主要功能包括&#xff1a; 实时数据采集&#xff1a;通过…

Cpython编译后再使用Pyinstaller打包

一、Cpython Python是一门解释型语言&#xff0c;当我们想让其他人运行我们的代码时&#xff0c;如果直接将.py源代码发送给他人&#xff0c;那么源代码将没有任何安全性可言&#xff0c;也就是任何一个人都可以打开源代码一看究竟&#xff0c;任何人都可以随意修改源代码。 …

Linux(CentOS7.5):新增硬盘分区纪实

一、服务器概述 1、既有一块系统硬盘&#xff0c;新增一块100G硬盘。 2、要求&#xff0c;将新插入硬盘分为&#xff1a;20G、30G、50G。 二、操作步骤 1、确认新硬盘是否插入成功&#xff1a; fdisk -l# 红色框出来的&#xff0c;为识别出来的新硬盘信息 # 黄色框出来的&#…

idea创建不了spring2.X版本,无法使用JDK8,最低支持JDK17 , 如何用idea创建spring2.X版本,使用JDK8解决方案

&#x1f9f8;欢迎来到dream_ready的博客&#xff0c;&#x1f4dc;相信您对博主首页也很感兴趣o (ˉ▽ˉ&#xff1b;) &#x1f4dc;jdk17安装全方位手把手安装教程 / 已有jdk8了&#xff0c;安装JDK17后如何配置环境变量 / 多个不同版本的JDK&#xff0c;如何配置环境变量&a…

【开源】基于Vue.js的高校大学生创业管理系统的设计和实现

项目编号&#xff1a; S 027 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S027&#xff0c;文末获取源码。} 项目编号&#xff1a;S027&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 系统公告模块2.2 创业项目模块2.3 创…

直播预告 | AR眼镜在现代医疗中究竟有哪些妙用?11.28晚八点虹科直播间为您揭晓!

直播预告 | AR眼镜在现代医疗中究竟有哪些妙用&#xff1f;11.28晚八点虹科直播间为您揭晓&#xff01; 什么是AR眼镜&#xff1f; AR眼镜&#xff0c;即增强现实眼镜&#xff0c;是一种结合虚拟信息与真实世界的创新医疗工具。 通过集成高科技传感器和实时数据处理技术&…

【Python】使用globals()函数成功解决tkinter多个新窗口问题

我在近期的一个项目&#xff08;tkinter复刻记事本&#xff09;上遇到一个很有意思的问题&#xff1a;如何在创建多个新窗口后&#xff0c;每个窗口还能独立运行&#xff1f;当时我尝试几种方法&#xff0c;奈何实力不足&#xff0c;于是便下定结论非使用线程不可&#xff0c;至…

通付盾Web3专题 | SharkTeam:起底朝鲜APT组织Lazarus Group,攻击手法及洗钱模式

国家级APT&#xff08;Advanced Persistent Threat&#xff0c;高级持续性威胁&#xff09;组织是有国家背景支持的顶尖黑客团伙&#xff0c;专门针对特定目标进行长期的持续性网络攻击。朝鲜APT组织Lazarus Group就是非常活跃的一个APT团伙&#xff0c;其攻击目的主要以窃取资…

JavaScript WebApi(二) 详解

监听事件 介绍 事件监听是一种用于在特定条件下执行代码的编程技术。在Web开发中&#xff0c;事件监听器可以用于捕获和响应用户与页面交互的各种操作&#xff0c;如点击、滚动、输入等。 事件监听的基本原理是&#xff0c;通过在特定元素上注册事件监听器&#xff0c;当事件…

免费SSL证书有效期只有90天?太短?

随着网络安全问题日益受到重视&#xff0c;SSL证书成为了网站安全的必需品。然而&#xff0c;在许多情况下&#xff0c;免费提供的SSL证书往往只有90天的有效期&#xff0c;这种期限对于很多用户来说显得过于短暂。 首先&#xff0c;我们要理解为什么 SSL 证书的有效期设定为90…

基于单片机DHT11湿度测量与控制-CO2-光照报警系统程序和仿真

一、系统方案 1、本设计采用这51单片机作为主控器。 2、DHT11温湿度、CO2、光照强度送到液晶1602显示。 3、按键设置报警值。 4、蜂鸣器报警。 二、硬件设计 原理图如下&#xff1a; 三、单片机软件设计 1、首先是系统初始化 //初始化LCD*********************************…

LCR 047. 二叉树剪枝 和 leetCode 1110. 删点成林 + 递归 + 图解

给定一个二叉树 根节点 root &#xff0c;树的每个节点的值要么是 0&#xff0c;要么是 1。请剪除该二叉树中所有节点的值为 0 的子树。节点 node 的子树为 node 本身&#xff0c;以及所有 node 的后代。 示例 1: 输入: [1,null,0,0,1] 输出: [1,null,0,null,1] 解释: 只有红…