【强化学习系列】Gym库使用——创建自己的强化学习环境2:拆解官方标准模型源码/规范自定义类+打包自定义环境

       

目录

一、 官方标准环境的获取与理解

二、根据官方环境源码修改自定义

        1.初始化__init__()

        2.重置环境 reset()

三、打包环境

        1.注册与创建自定义环境

        2.环境规范化


         在本文的早些时候,曾尝试按照自己的想法搭建自定义的基于gym强化学习环境。

        【强化学习系列】Gym库使用——创建自己的强化学习环境1:单一环境创建测试

        但由于学艺不精,在矢量化环境时遇到诸多障碍。仔细检查发现自己编写的环境存在很多不符合gym官方规范的地方,因此,索性从零重新研读官方介绍文档,从头开始解读官方标准环境源码,以帮助规范化自己搭建环境的类参数设置。

        官方文档地址:https://gymnasium.farama.org/

一、 官方标准环境的获取与理解

        通过打印 gym.envs.registry.keys() 获得的字典,就可以获得官方默认的所有可以直接调用的环境。

import gymnasium as gym# 官方支持的所有游戏环境
print(gym.envs.registry.keys())

        通过官方给出的运行和可视化代码即可,先通过“人为”(‘human’)来玩游戏。

import gymnasium as gym# 官方支持的所有游戏环境
print(gym.envs.registry.keys())# 可视化
env = gym.make("MountainCarContinuous-v0", render_mode="human")
observation, info = env.reset()for _ in range(1000):action = env.action_space.sample()  # agent policy that uses the observation and infoobservation, reward, terminated, truncated, info = env.step(action)if terminated or truncated:observation, info = env.reset()
env.close()

        下面展示一些官方游戏的可视化窗口的部分画面。只需根据最开始获得的游戏包装好的字典名,修改上述代码中环境实例化的make中第一个参数名即可,实际运行是动态的。

 

        首先从官方给出的Github地址下载好Gymnaium的源码——Gymnaium官方Github地址

        在官方文档目录的 ENVIRONMENTS 中介绍相关环境的一些基本关键信息,包括其状态动作空间、交互逻辑、奖励逻辑等。

        其中所有环境的源码保存在Github文件的 gymnaium/envs/ 目录下。

        本文选择经典环境下的山地车(Mountain Car)为源码拆解的例子。首先看起官方文档的环境介绍,其动作空间是一个三种可能的离散空间,分别对应小车的向左向右加速或不加速。

        其状态空间是一个连续的二维空间,分别是小车在x轴的位置,取值范围是(-1.2,0.6);小车速度,取值范围是(-0.07,0.07)。

        目标是让小车登顶,初始化是将小车随机放在山谷的一个位置,初始速度为0,如果小车登顶或超过200步操作,则退出游戏。

二、根据官方环境源码修改自定义

        在理解游戏逻辑的基础上,进入其源码处查看并学习其代码类编写的规范性。

        1.初始化__init__()

        首先是环境的初始化。其在传入参数上设计两个内容:一是选择环境可视化,二是设置汽车的初始速度为0。

        还可以看到其可视化模式的选择是定义了一个metadata的字典,其定义状态空间和动作空间的逻辑和我们之前无异。

        值得注意的规范细节是,对于初始化是numpy数组向量时,都需标准其定义的数据类型dtype

        总结官方环境源码初始化的内容,可以得出初始化的个关键内容——状态环境的取值范围(如所处空间的大小——游戏里的地图,到边缘就不能探索了)、实体状态属性值(如小车的速度、或者游戏里角色的血量蓝条等等)、可视化窗口的一些变量。       

        现在回到自定义环境的逻辑(参考文章——单一自定义环境测试),新的环境中状态空间是图片上的目标检测框,是一个四维的(左上右下四坐标列表)连续空间。动作空间是一个四种可能的离散空间——上扩或缩、下扩或缩。其框坐标的取值范围受到图片高宽的影响。因此初始化必须传入原图地址信息,以获取高宽范围。

        参考总结的内容,可以梳理清晰自定义环境的一些设置,状态空间上肯定超出图片高宽或者小于0的框无意义,因此很容易定义状态空间。更关键的是实体属性,在这里原始的框就是需要移动的实体,像小车会有速度一样,这里我们定义初始框包含几个实际物体对应框(虚拟的框,在状态空间环境中不存在)视为该框的属性值。同时一张图里也可能有多个框存在,如果不希望当前框和环境中其他存在的框存在高度重叠,也可以设置重叠度检查属性。

        2.重置环境 reset()

        查看官方的初始化环境函数,可以看到其对于函数传参有规范化的标准,其初始的状态控制放在options字典数据中传入,并且采取了强制关键字的模式(*后的参数需要以关键字对应方式传入)因此改进我们自定义的reset函数。

         再将真实框可视化进背景,即可运行reset得到可视化结果。

         以上可视化所有完整代码如下。

import pygame
import gym
from gym import spaces
import numpy as np
from typing import Optionalclass My_Env(gym.Env):metadata = {"render_modes": ["human", "rgb_array"], "render_fps": 1}def __init__(self, render_mode: Optional[str] = None, image_path: Optional[str] = None):super(My_Env, self).__init__()# 读取当前图像信息if image_path != None:self.img = pygame.image.load(image_path)self.width, self.height = self.img.get_width(), self.img.get_height()else:self.width, self.height = 100, 100print('env need background')self.real = None  # 真实框数据# 定义框属性值self.box_num = None  # 初始预测框数量self.overlap = None  # 包含真实框个数self.box_iou = None  # 重叠度检查# 创建动作和状态空间范围self.action_space = spaces.Discrete(4)self.observation_space = spaces.Box(low=np.array([0,0,0,0],dtype=np.float32), high=np.array([self.width, self.height,self.width, self.height],dtype=np.float32), shape=(4,), dtype=np.float32)# 定义模式:人类可视化or机器人训练assert render_mode is None or render_mode in self.metadata["render_modes"]self.render_mode = render_modeself.window = None  # 可视化窗口self.clock = None   # 可视化时钟self.window_size = (600,600)  # 窗口大小self.background = None  # 背景图self.scale_x = None     # x横轴缩放比self.scale_y = None     # y竖轴缩放比def reset(self,*,seed: Optional[int] = None,options: Optional[dict] = None) :real = options['real_box']   # 真实框数据box = options['box']   # 初始框数据self.real = np.array(real, dtype=np.float32).reshape(-1, 4)self.state = np.array(box,dtype=np.float32).reshape(-1,4)   # 规范数据格式和形状if self.render_mode == 'human':self.background = pygame.transform.scale(self.img, self.window_size) # 设置背景图# 计算x和y方向的缩放比例self.scale_x = self.window_size[0]/ self.widthself.scale_y = self.window_size[1] / self.heightself.render()return self.statedef step(self, action):  # 动作序列{0:上扩, 1:上缩, 2:下扩, 3:下缩}# 根据action生成移动数组movement_np = np.zeros_like(self.state,dtype=np.float32)for i,act in enumerate(action):if act == 0:movement_np[i,1] = 10elif act == 1:movement_np[i,1] = -10elif act == 2:movement_np[i,3] = 10elif act == 3:movement_np[i,3] = -10# 移动当前状态框self.state += movement_npself.state = np.clip(self.state, self.observation_space.low, self.observation_space.high)return self.statedef render(self):# 初始化窗口和时钟if self.window is None and self.render_mode == 'human':pygame.init()pygame.display.init()self.window = pygame.display.set_mode(self.window_size)if self.clock is None and self.render_mode == 'human':self.clock = pygame.time.Clock()# 重新绘制背景,以清除上一层if self.background is not None:self.window.blit(self.background,(0,0))for real in list(self.real):rect_real = [real[0], real[1], abs(real[2]-real[0]), abs(real[3]-real[1])]# 根据x和y方向的缩放比例计算每个矩形框的新位置stretched_rectangle = [rect_real[0] * self.scale_x,  # x 坐标rect_real[1] * self.scale_y,  # y 坐标rect_real[2] * self.scale_x,  # 宽度rect_real[3] * self.scale_y  # 高度]pygame.draw.rect(self.window, (255, 0, 0), stretched_rectangle, 2)  # 绘制矩形框,线宽为2# 绘制框for box in list(self.state):rect = [box[0], box[1], abs(box[2]-box[0]), abs(box[3]-box[1])]# 根据x和y方向的缩放比例计算每个矩形框的新位置stretched_rectangle = [rect[0] * self.scale_x,  # x 坐标rect[1] * self.scale_y,  # y 坐标rect[2] * self.scale_x,  # 宽度rect[3] * self.scale_y  # 高度]pygame.draw.rect(self.window, (0, 255, 0), stretched_rectangle, 3)  # 绘制矩形框,线宽为3# 更新显示pygame.display.flip()self.clock.tick(self.metadata["render_fps"])def close(self):pygame.quit()if __name__=='__main__':# 加载本地图片image_path = './jpg/000000000025.jpg'  # 替换为你的图片路径# 加载本地框box_path = './box/000000000025.json'with open(box_path, 'r') as f:box_dict_list = json.load(f)box_list = []for box_dict in box_dict_list:box = box_dict['box']box_list.append(box)box2 = [(0,3),(1,0),(1,1),(1,2),(1,3)]box1 = [(0,0),(0,1),(1,0),(1,1),(2,1),(3,1),(4,1),(4,2),(4,3),(5,1),(5,2),(5,3),(5,4),(6,1),(6,2),(6,3),(6,4),(7,1),(7,2),(7,3),(7,4),(7,5),(8,1),(8,2),(8,3),(8,4),(8,5)]real_list = []for i in range(len(box_list)):bbox = box_list[i]wid = bbox[2]-bbox[0]hgt = bbox[3]-bbox[1]gap = 30for w in range(wid//gap):for h in range(hgt//gap):if i ==0:if (h,w) in box1:real = [bbox[0]+w*gap, bbox[1]+h*gap, bbox[0]+(w+1)*gap, bbox[1]+(h+1)*gap]real_list.append(real)elif i ==1:if (h,w) in box2:real = [bbox[0] + w * gap, bbox[1] + h * gap, bbox[0] + (w + 1) * gap, bbox[1] + (h + 1) * gap]real_list.append(real)env = My_Env(render_mode='human', image_path=image_path)state = env.reset(options={'real_box':real_list, 'box':box_list})while True:env.render()

三、打包环境

        1.注册与创建自定义环境

        为了更方便的加载环境以及为后续矢量化做准备,需要将自定义的环境打包并使用官方推荐的gymnasium.make来加载环境,这样的加载方式会帮助检查环境的合规性。

        首先新建项目文件,将所有环境代码放入一个子文件中方便修改,然后再新建一个注册环境的子文件用于测试环境能否被成功注册。

        在测试环境下(有图片和框数据的文件夹),先注册环境在创建环境。

import gymnasium as gym
import json
from gymnasium.envs.registration import register# 注册环境
register(id='detect_env-v0',entry_point='my_gym_env.detect_env:Detect_Env',
)# 加载本地图片
image_path = './jpg/000000000025.jpg'  # 替换为你的图片路径
# 加载本地框
box_path = './box/000000000025.json'
real_path = './real/000000000025.json'
with open(box_path, 'r') as f:box_dict_list = json.load(f)
box_list = []
for box_dict in box_dict_list:box = box_dict['box']box_list.append(box)
with open(real_path, 'r') as f:real_list = json.load(f)env = gym.make('detect_env-v0', render_mode='human', image_path=image_path)state = env.reset(options={'real_box':real_list, 'box':box_list})
while True:env.render()

        2.环境规范化

        运行上面的测试代码,虽然能加载出环境演示,但是make会检测环境创建源码,并对不规范报错,这方便我们进行修改规范源码。

        ① reset 报错与规范

         这里报错的原因是,在定义环境重置 reset 时,没有返回 info 信息。补一个空字典即可。

UserWarning: WARN: The result returned by `env.reset()` was not a tuple of the form `(obs, info)`, where `obs` is a observation and `info` is a dictionary containing additional information. Actual type: `<class 'numpy.ndarray'>`

        ② observation_space 状态空间报错与规范

        改完后报新的错。这是说在重置环境时传入的状态空间与__init__ 中定义的状态空间形状不匹配。

UserWarning: WARN: The obs returned by the `reset()` method is not within the observation space.logger.warn(f"{pre} is not within the observation space.")

        检查环境中定义的空间必须是固定形状的,(,4)的定义会默认为(1,4),其不能自由匹配维度,但是作为图片中的目标框,其第一个维度数量是不固定的。因此一种解决方式是选择所有图片内的最大框数作为初始化状态空间第一维度,不够的地方用零填充。

# 设置最大框数
self.box_max_num = 20
# 创建动作和状态空间范围
self.action_space = spaces.Discrete(4)
self.observation_space = spaces.Box(low=np.zeros((self.box_max_num,4),dtype=np.float32),high=np.tile([self.width, self.height,self.width, self.height],(self.box_max_num,1)),shape=(self.box_max_num,4), dtype=np.float32)

# 初始传入框信息
state = np.array(box,dtype=np.float32).reshape(-1,4)   # 规范数据格式和形状# 填充零使得状态达到指定维度
self.state = np.pad(state,((0,self.box_max_num-state.shape[0]),(0,0)), mode='constant', constant_values=0)

        规范后不再报错,此时打印环境的状态检查。

        ③ step 报错与规范

        继续在加入动作空间的输入,解决step中的规范性问题。首先如果要输入动作,需要知道当前图片中实际需要操作的框的数量(不能根据状态空间判断,因为状态中框数(第一行维度)被强制设为最大框数), 因此在重置环境时就要返回当前图片的框数,作为每个框生成多少个动作的标准,在官方的规范代码中,可将这些额外信息写入info字典中返回。

info = {'box_num':state.shape[0]}
return self.state, info

        在测试代码中添加采样动作和step操作,并打印状态的变化numpy数组。注意此处环境还没有设计奖励函数,因此定义环境操作reward都是0。

env = gym.make('detect_env-v0', render_mode='human', image_path=image_path)state,info = env.reset(options={'real_box':real_list, 'box':box_list})
print(0, state)
for epid in range(50):action = [env.action_space.sample() for _ in range(info['box_num'])]state, reward, _, _, _ = env.step(action)print(epid+1, state)env.render()

         下图中,红色箭头表示右较大框的状态变化,黄色箭头代表左较小框的状态变化。此处为方便可视化,将最大框数量设为3。以20像素为步长变化检测框。

         加上具体动作打印结果可视化。

        至此,环境规范化告一段落,完整代码放在下面方便复制取走。终于可以开始后续矢量化环境操作了。

        自定义检测环境完整代码。

import pygame
import gymnasium as gym
from gymnasium import spaces
import numpy as np
from typing import Optionalclass Detect_Env(gym.Env):metadata = {"render_modes": ["human", "rgb_array"], "render_fps": 1}def __init__(self, render_mode: Optional[str] = None, image_path: Optional[str] = None):super(Detect_Env, self).__init__()# 读取当前图像信息if image_path != None:self.img = pygame.image.load(image_path)self.width, self.height = self.img.get_width(), self.img.get_height()else:self.width, self.height = 100, 100print('env need background')self.real = None  # 真实框数据# 定义框属性值self.box_num = None  # 初始预测框数量self.overlap = None  # 包含真实框个数self.box_iou = None  # 重叠度检查# 设置最大框数self.box_max_num = 3# 创建动作和状态空间范围self.action_space = spaces.Discrete(4)self.observation_space = spaces.Box(low=np.zeros((self.box_max_num,4),dtype=np.float32),high=np.tile([self.width, self.height,self.width, self.height],(self.box_max_num,1)),shape=(self.box_max_num,4), dtype=np.float32)# 定义模式:人类可视化or机器人训练assert render_mode is None or render_mode in self.metadata["render_modes"]self.render_mode = render_modeself.window = None  # 可视化窗口self.clock = None   # 可视化时钟self.window_size = (600,600)  # 窗口大小self.background = None  # 背景图self.scale_x = None     # x横轴缩放比self.scale_y = None     # y竖轴缩放比def reset(self,*,seed: Optional[int] = None,options: Optional[dict] = None) :real = options['real_box']   # 真实框数据box = options['box']   # 初始框数据self.real = np.array(real, dtype=np.float32).reshape(-1, 4)state = np.array(box,dtype=np.float32).reshape(-1,4)   # 规范数据格式和形状# 填充零使得状态达到指定维度self.state = np.pad(state,((0,self.box_max_num-state.shape[0]),(0,0)), mode='constant', constant_values=0)if self.render_mode == 'human':self.background = pygame.transform.scale(self.img, self.window_size) # 设置背景图# 计算x和y方向的缩放比例self.scale_x = self.window_size[0]/ self.widthself.scale_y = self.window_size[1] / self.heightself.render()info = {'box_num':state.shape[0]}return self.state, infodef step(self, action):  # 动作序列{0:上扩, 1:上缩, 2:下扩, 3:下缩}# 根据action生成移动数组movement_np = np.zeros_like(self.state,dtype=np.float32)for i,act in enumerate(action):if act == 0:movement_np[i,1] = 20elif act == 1:movement_np[i,1] = -20elif act == 2:movement_np[i,3] = 20elif act == 3:movement_np[i,3] = -20# 移动当前状态框self.state += movement_npself.state = np.clip(self.state, self.observation_space.low, self.observation_space.high)return np.array(self.state, dtype=np.float32),0,False,False,{}def render(self):# 初始化窗口和时钟if self.window is None and self.render_mode == 'human':pygame.init()pygame.display.init()self.window = pygame.display.set_mode(self.window_size)if self.clock is None and self.render_mode == 'human':self.clock = pygame.time.Clock()# 重新绘制背景,以清除上一层if self.background is not None:self.window.blit(self.background,(0,0))for real in list(self.real):rect_real = [real[0], real[1], abs(real[2]-real[0]), abs(real[3]-real[1])]# 根据x和y方向的缩放比例计算每个矩形框的新位置stretched_rectangle = [rect_real[0] * self.scale_x,  # x 坐标rect_real[1] * self.scale_y,  # y 坐标rect_real[2] * self.scale_x,  # 宽度rect_real[3] * self.scale_y  # 高度]pygame.draw.rect(self.window, (255, 0, 0), stretched_rectangle, 2)  # 绘制矩形框,线宽为2# 绘制框for box in list(self.state):rect = [box[0], box[1], abs(box[2]-box[0]), abs(box[3]-box[1])]# 根据x和y方向的缩放比例计算每个矩形框的新位置stretched_rectangle = [rect[0] * self.scale_x,  # x 坐标rect[1] * self.scale_y,  # y 坐标rect[2] * self.scale_x,  # 宽度rect[3] * self.scale_y  # 高度]pygame.draw.rect(self.window, (0, 255, 0), stretched_rectangle, 3)  # 绘制矩形框,线宽为3# 更新显示pygame.display.flip()self.clock.tick(self.metadata["render_fps"])def close(self):pygame.quit()

        项目文件层级结构。

        测试可视化完整代码。

import gymnasium as gym
import json
from gymnasium.envs.registration import register# 注册环境
register(id='detect_env-v0',entry_point='my_gym_env.detect_env:Detect_Env',
)# 加载本地图片
image_path = './jpg/000000000025.jpg'  # 替换为你的图片路径
# 加载本地框
box_path = './box/000000000025.json'
real_path = './real/000000000025.json'
with open(box_path, 'r') as f:box_dict_list = json.load(f)
box_list = []
for box_dict in box_dict_list:box = box_dict['box']box_list.append(box)
with open(real_path, 'r') as f:real_list = json.load(f)env = gym.make('detect_env-v0', render_mode='human', image_path=image_path)state,info = env.reset(options={'real_box':real_list, 'box':box_list})
print(0, state)
for epid in range(50):action = [env.action_space.sample() for _ in range(info['box_num'])]print('action', action)state, reward, _, _, _ = env.step(action)print(epid+1, state)env.render()

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

IEEE-754 32位十六进制数 转换为十进制浮点数

要将 IEEE-754 32位十六进制数 转换为 十进制浮点数&#xff0c;可以使用LabVIEW中的 Type Cast 函数。以下是一些具体步骤&#xff0c;以及相关实例的整理&#xff1a; 实现步骤&#xff1a; 输入十六进制数&#xff1a;在LabVIEW中&#xff0c;首先需要创建一个输入控制器&am…

剃(磨)前插齿刀设计计算开发第二步:

从刀具上的各段齿形计算加工出的齿轮端面齿形&#xff0c;下一步进行细化处理[开心]&#xff0c;去掉一些线头&#xff0c;增加一些关键参数的计算及标准&#xff0c;例如&#xff1a;SAP、UnderCut、EAP、Chamfer等等&#xff0c;祝我好运吧&#xff0c;谢谢&#xff01;

MySQL系列—11.Redo log

1.简介 概念 redo log用于记录事务操作变化&#xff0c;记录的是数据被修改之后的值&#xff0c;&#xff08;tbs space id page no action&#xff09;。 作用 尚未完成的DML,数据库崩溃则用log恢复。保证事务持久性。 ( 1 ) 在页面修改完成之后&#xff0c;脏页刷入磁盘之…

ZYNQ FPGA自学笔记~点亮LED

一 ZYNQ FPGA简介 ZYNQ FPGA主要特点是包含了完整的ARM处理系统&#xff0c;内部包含了内存控制器和大量的外设&#xff0c;且可独立于可编程逻辑单元&#xff0c;下图中的ARM内核为 ARM Cortex™-A9&#xff0c;ZYNQ FPGA包含两大功能块&#xff0c;处理系统Processing System…

路由原理介绍

定义与过程 定义&#xff1a;是指导IP报文发送的路径信息 过程&#xff1a; 检查数据包的目的地确定信息源发现可能的路径选择最佳路径验证和维护路由信息 路由来源 直连路由&#xff1a;不需配置&#xff0c;路由器配置IP后自动生效 静态路由&#xff1a;手动配置 ip r…

Ubantu LLaMA-Factory实战

一、Ubantu LLaMA-Factory实战安装&#xff1a; CUDA 安装 CUDA 是由 NVIDIA 创建的一个并行计算平台和编程模型&#xff0c;它让开发者可以使用 NVIDIA 的 GPU 进行高性能的并行计算。 首先&#xff0c;在 https://developer.nvidia.com/cuda-gpus 查看您的 GPU 是否支持CU…

壹嘉情,中国与世界经济文化交流的新桥梁

壹嘉情正在全球华商领域迅速崛起。作为意大利华商总会的中国分部&#xff0c;壹嘉情承载着推动两岸及全球华商深度合作、实现资源共享和互利共赢的使命。它的成立标志着意大利华商总会在全球战略布局上的重要一步&#xff0c;同时也昭示了全球化浪潮中&#xff0c;华人企业正加…

苹果电脑也可以清除垃圾吗?苹果电脑清理垃圾用什么软件哪个好?

相对于Windows电脑&#xff0c;目前专注于苹果电脑清理的软件不算多&#xff0c;那么&#xff0c;苹果电脑垃圾清理软件哪个好&#xff1f;本文经过对比给大家推荐几款好用的软件。另外&#xff0c;我们还会进行苹果电脑垃圾清理方法盘点&#xff0c;让大家更了解电脑的清理方法…

从零开始讲DDR(0)——DDR的前世今生

一、计算机组成 计算机组成结构&#xff08;Computer Architecture&#xff09;是计算机系统的核心&#xff0c;它定义了计算机的基本工作原理和设计模式。计算机的组成可以分成以下3大类&#xff1a;中央处理器&#xff08;CPU&#xff09;、存储器和输入/输出子系统。 1.1 中…

达梦数据库DM8使用介绍

达梦数据库DM8使用介绍 达梦数据库DM8使用介绍一、安装达梦数据库二、初始化数据库实例三、SQL 分类DML(Data Mannipulation Language)数据操纵语言&#xff1a;DDL(Data Definition Language)数据定义语言&#xff1a;DCL(Data Control Language)数据控制语言&#xff1a;TCL(…

springboot实训学习笔记(5)(用户登录接口的主逻辑)

接着上篇博客学习。上篇博客是已经基本完成用户模块的注册接口的开发以及注册时的参数合法性校验。具体往回看了解的链接如下。 springboot实训学习笔记&#xff08;4&#xff09;(Spring Validation参数校验框架、全局异常处理器)-CSDN博客文章浏览阅读576次&#xff0c;点赞7…

城市级河流三维处理及展示的一些技术

本文是一些算法技术的初探分析&#xff0c;会陆续修订。 1、问题 河流是一种非常复杂的多边形。在二维地图可以采用多边形填充算法(DDA)对任意复杂的多边形进行绘制与填充。但是三维引擎只能采纳三角面进行渲染。但在如此复杂的多边形面前&#xff0c;简单的三角化算法不能解…

Java 回顾方法的定义

一、方法的定义 1&#xff0e;修饰符&#xff08;public static…&#xff09;详见博客【Java 方法的定义】 2&#xff0e;返回值&#xff08;int, double, char[],…., void&#xff09;详见博客【Java 方法的定义】 3. break&#xff1a;跳出switch 结束循环&#xff0c;详…

2024年09月18日《每日一练》

1、 智慧城市建设参考模型包括有依赖关系的5层结构和对建设有约束关系的3个支撑体系&#xff0c;5层结构包括物联感知层、通信网络层、计算与存储层、数据及服务支撑层、智慧应用层;3个支撑体系除了建设和运营管理体系、安全保障体系之外还包括&#xff08;&#xff09;。 A 人…

使用arduino玩基于esp8266的nodemcu开发板

一、简介 中秋节到图书馆看书&#xff0c;看到了arduino方面的书籍&#xff0c;里面有提到ESP8266模块。让我想起我抽屉里吃灰很久了的基于esp8266的nodemcu开发板。于是把书借回家研究了一下。这里做个记录。 二、我目前在使用的云服务器推荐 学Linux不搞个云服务器始终感觉…

进行直流充电桩测试仪的步骤和规范

直流充电桩测试仪是一种用于检测和评估直流充电桩性能的设备。它能够测量充电桩的输出电压、电流、功率等参数&#xff0c;并能够模拟各种负载条件&#xff0c;以测试充电桩的稳定性和可靠性。下面是进行直流充电桩测试仪的步骤和规范&#xff1a; 确保测试环境安全&#xff0…

2024.9.18 作业+思维导图

练习&#xff1a;提示并输入一个字符串&#xff0c;统计该字符串中字母、数字、空格、其他字符的个数并输出 #include <iostream>using namespace std;int main() {cout << "请输入一个字符串:" << endl;string str;int shuzi 0,zimu0,space 0,q…

MAGDA:多智能体指南驱动的诊断助手

MAGDA&#xff1a;多智能体指南驱动的诊断助手 秒懂大纲提出背景精细拆解输入输出全流程创意视角中文意译 论文&#xff1a;MAGDA: Multi-agent guideline-driven diagnostic assistance 秒懂大纲 ├── MAGDA: Multi-agent guideline-driven diagnostic assistance【研究主…

安全运维教程(非常详细)从零基础入门到精通,看完这一篇就够了

一、安全运维-网络 1、IP地址相关 IP地址属于网络层地址&#xff0c;用于标识网络中的节点设备。 IP地址由32bit构成&#xff0c;每8bit一组&#xff0c;共占用4个字节。 IP地址由两部分组成&#xff0c;网络位和主机位。 IP地址分类&#xff1a; 类别网络位子网掩码私有地…

嵌入式基本知识梳理

一、CPU的组成 CPU&#xff1a;中央处理器-----》soc(片上系统)&#xff08;描述的是一种芯片&#xff0c;这个芯片具有运算程序的能力&#xff09;、 UART: Universal Asynchronous Receiver/Transmitter&#xff0c;通用异步收发传输器&#xff09;是一种广泛使用的串行通信协…