深度强化学习DQN训练避障

news/2024/12/29 12:25:01/

目录

一.前言

二.代码

2.1完整代码

2.2运行环境

2.3动作空间

2.4奖励函数

2.5状态输入

2.6实验结果


一.前言

深度Q网络(DQN)是深度强化学习领域的一项革命性技术,它成功地将深度学习的强大感知能力与强化学习的决策能力相结合。在过去的几年里,DQN已经在许多复杂的问题上展示了其卓越的性能,从经典的Atari游戏到更复杂的机器人控制任务。特别值得一提的是,DQN在处理高维状态空间和动作空间的问题时表现出了显著的优势,使得它能够处理传统强化学习方法难以解决的难题。

二维环境避障问题是强化学习领域的一个经典问题,它要求智能体在二维空间中移动,同时避免与障碍物碰撞。这个问题看似简单,但实际上涉及到了许多复杂的因素,如环境的不确定性、部分可观察性、以及智能体的感知和决策能力限制等。因此,开发一种能够有效解决二维环境避障问题的方法具有重要的实际意义和应用价值。

近年来,随着深度学习和强化学习技术的不断发展,基于DQN的二维环境避障方法已经取得了显著的进展。通过利用深度学习的强大特征提取能力,DQN能够从原始的环境状态中学习到有用的特征表示,并基于这些特征进行高效的决策。此外,DQN还引入了经验回放和目标网络等关键技术,进一步提高了算法的稳定性和收敛速度。

本文研究如何让智能体在一个二维障碍物环境中进行避障。

二.代码

2.1完整代码

import gym  
import torch,copy
import torch.nn as nn
import torch.nn.functional as F
import numpy as np  
from gym import spaces  
from gym.utils import seeding  
import matplotlib.pyplot as plt  
#超参数
BATCH_SIZE = 32
LR = 0.01
EPSILON = 0.9 #随机选取的概率,如果概率小于这个随机数,就采取greedy的行为
GAMMA = 0.9
TARGET_REPLACE_ITER = 100
MEMORY_CAPACITY = 2000
N_ACTIONS=len(np.arange(0, 2*np.pi, 0.1))
N_STATES=4
ENV_A_SHAPE =0
class Net(nn.Module):def __init__(self, ):super(Net, self).__init__()self.fc1 = nn.Linear(N_STATES, 500)self.fc1.weight.data.normal_(0, 0.1)   # initializationself.out = nn.Linear(500, N_ACTIONS)self.out.weight.data.normal_(0, 0.1)   # initializationdef forward(self, x):x = self.fc1(x)x = F.relu(x)actions_value = self.out(x)return actions_value
class DQN(object):def __init__(self):#DQN是Q-Leaarning的一种方法,但是有两个神经网络,一个是eval_net一个是target_net#两个神经网络相同,参数不同,是不是把eval_net的参数转化成target_net的参数,产生延迟的效果self.eval_net=Net()self.target_net =copy.deepcopy(self.eval_net)self.learn_step_counter = 0 #学习步数计数器self.memory_counter = 0 #记忆库中位值的计数器self.memory = np.zeros((MEMORY_CAPACITY,N_STATES * 2 + 2)) #初始化记忆库#记忆库初始化为全0,存储两个state的数值加上一个a(action)和一个r(reward)的数值self.optimizer = torch.optim.Adam(self.eval_net.parameters(),lr = LR)self.loss_func = nn.MSELoss() #优化器和损失函数    #接收环境中的观测值,并采取动作def choose_action(self,x):global EPSILON#x为观测值x = torch.unsqueeze(torch.FloatTensor(x),0)if np.random.uniform() < EPSILON:#随机值得到的数有百分之九十的可能性<0.9,所以该if成立的几率是90%#90%的情况下采取actions_value高的作为最终动作actions_value = self.eval_net.forward(x)action = torch.max(actions_value,1)[1].data.numpy()action = action[0] if ENV_A_SHAPE == 0 else action.reshape(ENV_A_SHAPE) # return the argmax indexelse:#其他10%采取随机选取动作action = np.random.randint(0,N_ACTIONS) #从动作中选一个动作action = action if ENV_A_SHAPE == 0 else action.reshape(ENV_A_SHAPE)return action    #记忆库,存储之前的记忆,学习之前的记忆库里的东西def store_transition(self,s,a,r,s_):transition = np.hstack((s, [a, r], s_))# 如果记忆库满了, 就覆盖老数据index = self.memory_counter % MEMORY_CAPACITYself.memory[index, :] = transitionself.memory_counter += 1def learn(self):# target net 参数更新,每隔TARGET_REPLACE_ITE更新一下if self.learn_step_counter % TARGET_REPLACE_ITER == 0:self.target_net.load_state_dict(self.eval_net.state_dict())self.learn_step_counter += 1#targetnet是时不时更新一下,evalnet是每一步都更新# 抽取记忆库中的批数据sample_index = np.random.choice(MEMORY_CAPACITY, BATCH_SIZE)b_memory = self.memory[sample_index, :] #打包记忆,分开保存进b_s,b_a,b_r,b_sb_s = torch.FloatTensor(b_memory[:, :N_STATES])b_a = torch.LongTensor(b_memory[:, N_STATES:N_STATES+1].astype(int))b_r = torch.FloatTensor(b_memory[:, N_STATES+1:N_STATES+2])b_s_ = torch.FloatTensor(b_memory[:, -N_STATES:])# 针对做过的动作b_a, 来选 q_eval 的值, (q_eval 原本有所有动作的值)q_eval = self.eval_net(b_s).gather(1, b_a)  # shape (batch, 1)q_next = self.target_net(b_s_).detach()     # q_next 不进行反向传递误差, 所以 detachq_target = b_r + GAMMA * q_next.max(1)[0]   # shape (batch, 1)loss = self.loss_func(q_eval, q_target)# 计算, 更新 eval netself.optimizer.zero_grad()loss.backward() #误差反向传播self.optimizer.step()class RobotEnv(gym.Env):  metadata = {'render.modes': ['human']}  def __init__(self):  super(RobotEnv, self).__init__()  self.grid_size = 1000  self.viewobs = np.zeros((self.grid_size, self.grid_size))  self.robot_radius = 10  self.obstacle_radius = 50  self.robot_pos = np.array([self.grid_size // 8, self.grid_size // 2])  self.obstacle_pos = np.array([self.grid_size *3// 4, self.grid_size // 2])  self.target_pos = np.array([self.grid_size *7 // 8, self.grid_size // 2])  self.action_space =np.arange(0, 2*np.pi, 0.1)  self.observation_space = spaces.Box(low=0, high=1, shape=(self.grid_size, self.grid_size))  self._seed()  def _reset(self):self.robot_pos = np.array([self.grid_size  // 8, self.grid_size // 2])  self.obstacle_pos = np.array([self.grid_size *3// 4, self.grid_size // 2])  self.observation_space = spaces.Box(low=0, high=1, shape=(self.grid_size, self.grid_size))  return self._get_obs()def _seed(self, seed=None):  self.np_random, seed = seeding.np_random(seed)  return [seed]  def _step(self, action):  angle = action*0.1  dx = self.robot_radius * np.cos(angle)  dy = self.robot_radius * np.sin(angle)  new_pos = self.robot_pos + np.array([dx, dy])  if np.linalg.norm(new_pos - self.obstacle_pos) <= self.obstacle_radius:  reward = -1done = True  elif self.robot_pos[0]<=self.robot_radius+self.robot_radius or self.robot_pos[0]>=self.grid_size-self.robot_radius-self.robot_radius:reward=-1done = True  elif self.robot_pos[1]<=self.robot_radius+self.robot_radius or self.robot_pos[1]>=self.grid_size-self.robot_radius-self.robot_radius:reward=-1done = True  else:  reward = 1 / (1 + np.linalg.norm(new_pos - self.target_pos))  done = False  self.robot_pos = new_pos  return self._get_obs(), reward, done, {}  def _get_obs(self):  self.viewobs = np.zeros((self.grid_size, self.grid_size))  self.viewobs[int(self.robot_pos[0])][int(self.robot_pos[1])] = 1.0self.viewobs[int(self.obstacle_pos[0] - self.obstacle_radius):int(self.obstacle_pos[0] + self.obstacle_radius),   int(self.obstacle_pos[1] - self.obstacle_radius):int(self.obstacle_pos[1] + self.obstacle_radius)] = 0.5  obs=[self.robot_pos[0],self.robot_pos[1],self.obstacle_pos[0],self.obstacle_pos[1]]return obsdef _render(self, mode='human', close=False):  if close:  return  plt.imshow(self.viewobs, cmap='gray')  plt.scatter(self.robot_pos[1], self.robot_pos[0], c='red')  plt.scatter(self.obstacle_pos[1], self.obstacle_pos[0], c='blue')  plt.xlim([0, self.grid_size])  plt.ylim([0, self.grid_size])  #plt.show()  plt.pause(0.01)  plt.clf()  if __name__ == "__main__":  env = RobotEnv()  dqn = DQN()#for i in range(400000):i=0while True:s=env.reset()for j in range(300):  # Run for 100 steps as an example  action  = dqn.choose_action(s) s_, reward, done, info = env.step(action)  print(i)if i>=200000:env.render()  dqn.store_transition(s,action,reward,s_)if dqn.memory_counter > MEMORY_CAPACITY:dqn.learn()if done:  print("Episode finished after {} timesteps".format(i+1))  breaks = s_ # 现在的状态赋值到下一个状态上去i=i+1

2.2运行环境

gym== 0.7.0

torch==2.1.1

2.3动作空间

    在深度强化学习领域中,动作空间的设计对于智能体的决策能力和学习效果具有至关重要的影响。针对连续动作空间的问题,一种常见的处理方式是将其离散化,以便应用离散动作空间的强化学习算法。本实验提出了一种将0-2π的连续动作空间等分为63个离散动作的方法。

    通过采用等分为63个离散动作的策略。使得实验在保留了足够的动作分辨率的同时,避免了过度离散化可能带来的维度灾难问题。通过将动作空间划分为足够细密的离散区间,智能体能够更精确地控制自身的行为,从而在面对复杂环境时实现更好的性能。

2.4奖励函数

在本研究中,我们设计的奖励函数主要基于两个核心原则:

  1. 靠近障碍物的奖励:当智能体靠近障碍物后方时,将获得一个与距离大小成反比的奖励值。具体而言,随着智能体逐渐接近障碍物,奖励值将逐渐增大,以鼓励智能体保持与障碍物的安全距离。这种设计策略旨在通过提供即时的正反馈来促使智能体学习避免与障碍物发生碰撞的行为。
  2. 违反仿真边界或进入障碍物区域的惩罚:一旦智能体离开仿真边界或进入障碍物区域,将获得一个负奖励(惩罚),并且仿真将立即停止。这种严厉的惩罚机制确保了智能体能够清晰地认识到这些行为的不可取性,从而在学习过程中避免重复犯下相同的错误。

2.5状态输入

    在本研究中,我们将智能体的当前位置以及障碍物的位置信息作为状态输入。具体来说,状态空间包括智能体的二维坐标和障碍物的二维坐标。这样的设计使得智能体能够直接感知到周围环境的信息,并根据这些信息做出相应的决策。

2.6实验结果

    无论是在静态还是动态环境中,智能体都能够准确地感知到障碍物的存在,并有效地规划出避开障碍物的路径。


http://www.ppmy.cn/news/1287634.html

相关文章

数据特征工程 | PSO粒子群算法的特征选择原理及python代码实现

粒子群优化(Particle Swarm Optimization,PSO)是一种基于群体智能的优化算法,常用于解决搜索和优化问题。在特征选择问题中,PSO可以用于选择最佳的特征子集,从而提高模型的性能和效果。 PSO的特征选择原理如下: 表示特征子集:PSO中的每个粒子表示一个特征子集,其中每…

K8S中将一个Deployment暴露出一个指定的外部端口,生成yaml

K8S中将一个Deployment暴露出一个指定的外部端口 如何固定NodePort端口 如何固定NodePort端口 - 知乎 kubectl expose deployment quickstart-kb --namekibana-svc --port5601 --target-port5601 --typeNodePort --dry-runclient -o yaml > quickstart-kb-svc.yaml [root…

.NET进阶篇06-async异步、thread多线程2

知识须要不断积累、总结和沉淀&#xff0c;思考和写做是成长的催化剂web 内容目录 1、线程Thread 一、生命周期 二、后台线程 三、静态方法 1.线程本地存储 2.内存栅栏 四、返回值 2、线程池ThreadPool 一、工做队列 二、工做线程和IO线程 三、和Thread区别 四、定时器 1、线…

开源安全测试工具 | 网络安全工具列表

自动化渗透测试 • AttackSurfaceMapper (https://github.com/superhedgy/AttackSurfaceMapper) - 自动化渗透测试工具, 使用手册/测试流程 (https://www.uedbox.com/post/59110/)。 • vajra (https://github.com/r3curs1v3-pr0xy/vajra) - 自动化渗透测试. • Savior (https…

Oracle database 静默安装 oracle12c 一键安装 12.1.0.2

基于oracle安装包中应答文件实现一键安装 注意此安装脚本基于12.1.0.2 安装包 原始安装包结构为两个压缩包 此脚本使用安装包为原始压缩包解压后、 重新封装为一个.zip压缩包 建议在linux 环境下解压重新压缩后 使用该脚本 支持环境: Linux :centerOS 7 oracle :12.1.0.…

【Linux】Linux 下基本指令 -- 详解

无论是什么命令&#xff0c;用于什么用途&#xff0c;在 Linux 中&#xff0c;命令有其通用的格式&#xff1a; command [-options] [parameter] command&#xff1a;命令本身。-options&#xff1a;[可选&#xff0c;非必填]命令的一些选项&#xff0c;可以通过选项控制命令的…

【后端】Docker学习笔记

文章目录 Docker一、Docker安装&#xff08;Linux&#xff09;二、Docker概念三、Docker常用命令四、数据卷五、自定义镜像六、网络七、DockerCompose Docker Docker是一个开源平台&#xff0c;主要基于Go语言构建&#xff0c;它使开发者能够将应用程序及其依赖项打包到一个轻…

Tomcat与Netty比较

Tomcat介绍Tomcat支持的协议Tomcat的优缺点Netty介绍Netty支持的协议Netty的优点和缺点Tomcat和Netty的区别Tomcat和Netty的应用场Tomcat和Netty来处理大规模并发连接的优化Tomcat与Netty的网络模型的区别Tomcat与Netty架构设计拓展 Tomcat介绍 Tomcat是一个免费的、开放源代码…