👏🏻 你好!欢迎访问IT教程网,0门教程,教程全部原创,计算机教程大全,全免费!

🔥 新增教程

《黑神话 悟空》游戏开发教程,共40节,完全免费,点击学习

《AI副业教程》,完全原创教程,点击学习

1 强化学习的基本概念和历史

基本概念

强化学习(Reinforcement Learning,RL)是一种机器学习的子领域,它关注如何通过与环境的互动来学习做出决策。强化学习的核心思想是通过试错(trial and error)的方法,让代理(agent)在给定的环境中进行探索,并根据环境反馈的奖励(reward)来优化其决策策略。

关键组成部分

强化学习的基本组成部分主要包括以下几个:

  1. 代理(Agent):执行动作并学习的实体。
  2. 环境(Environment):代理操作的外部系统,代理通过与环境的互动获得反馈。
  3. 状态(State):描述环境当前情况的信息。每个时间步,代理都会感知到当前的状态。
  4. 动作(Action):代理在当前状态下可以选择的行为。
  5. 奖励(Reward):环境对代理所采取动作的反馈,表明动作的好坏。
  6. 策略(Policy):代理在给定状态下选择动作的规则或函数。策略可以是确定的(deterministic)或随机的(stochastic)。
  7. 价值函数(Value Function):用于评估状态或状态-动作对的“好坏”,通常定义为从该状态或动作出发,未来所能获得的期望奖励。

奖励和惩罚

很重要的一点是,强化学习通过奖励惩罚来引导代理的学习,从而提升决策的质量。例如,在一个经典的强化学习任务——迷宫问题中,代理需要找到从起点到终点的路径。每当它走出正确的路径时,就会获得正向奖励,反之则会受到负向惩罚。代理的最终目标是通过最大化累积奖励来找到最优策略。

历史背景

强化学习的起源可以追溯到20世纪的心理学和生物学研究,尤其是巴甫洛夫的经典条件反射理论和斯金纳的操作性条件反射理论。这些理论启示了如何通过奖励惩罚来形成学习机制。

发展历程

  • 1950年代:一些早期的研究集中在通过马尔可夫决策过程(MDP)来建模决策问题,奠定了强化学习的数学基础。

  • 1980年代:随着计算机技术的发展,强化学习开始吸引研究者的注意。1989年,沃特金斯(Watkins)提出了Q-learning算法,这是一个重要的无模型强化学习方法,能够通过学习状态-动作值函数来指导策略优化。

  • 1990年代策略梯度方法和时间差分学习(TD Learning)等新方法相继出现,进一步丰富了强化学习的研究。

  • 2010年代至今:强化学习结合深度学习(Deep Learning),形成了深度强化学习(Deep Reinforcement Learning),这一领域的突破性成果使得代理可以在复杂环境中成功学习。例如,2015年DeepMind的AlphaGo系统通过深度强化学习在围棋上战胜了人类顶尖选手。

经典案例

下面是一个简单的强化学习例子——网格世界(Grid World)问题的代码实现。这个例子体现了如何通过强化学习来学习策略。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
import numpy as np

# 定义环境
class GridWorld:
def __init__(self, size):
self.size = size
self.grid = np.zeros((size, size))
self.state = (0, 0) # 初始状态

def reset(self):
self.state = (0, 0)
return self.state

def step(self, action):
if action == 0: # 向上
self.state = (max(self.state[0] - 1, 0), self.state[1])
elif action == 1: # 向下
self.state = (min(self.state[0] + 1, self.size - 1), self.state[1])
elif action == 2: # 向左
self.state = (self.state[0], max(self.state[1] - 1, 0))
elif action == 3: # 向右
self.state = (self.state[0], min(self.state[1] + 1, self.size - 1))

# 回合结束条件
if self.state == (self.size - 1, self.size - 1):
return self.state, 1, True # 到达目标,奖励+1
return self.state, -0.1, False # 每一步的惩罚

# 训练代理
def train(agent, env, episodes):
for episode in range(episodes):
state = env.reset()
done = False
while not done:
action = agent.select_action(state) # 代理选择动作
next_state, reward, done = env.step(action) # 环境反馈
agent.learn(state, action, reward, next_state) # 代理学习

# 代理定义略...

# 使用示例
grid = GridWorld(size=5)
# train(your_agent, grid, 1000)

通过这种方式,代理在多个回合(episode)中不断探索,通过获得的奖励来调整其策略,从而学习到如何在网格中达到目标。

总结

强化学习不仅是一种强大的学习机制,还在很多实际应用中得到了成功应用,如机器人控制、自动驾驶、游戏AI等。随着深度学习的发展,强化学习的应用前景越来越广阔,为我们提供了更多的可能性。下一篇,我们将探讨强化学习与监督学习的区别,以帮助读者更好地理解这两者之间的差异与联系。

分享转发

2 强化学习导论之强化学习与监督学习的区别

在上篇中,我们介绍了强化学习的基本概念和历史背景。在这一篇中,我们将深入探讨强化学习与监督学习之间的区别,以帮助读者更好地理解这两种机器学习范式的应用场景和适用条件。

强化学习与监督学习的基本定义

在进入详细比较之前,首先我们要明确这两种学习方式的基本定义:

  1. 监督学习:在监督学习中,算法学习一个输入到输出的映射关系。训练数据通常包含输入特征和对应的标签,例如给定邮件的文本内容(输入),算法需要判断这封邮件是“垃圾邮件”还是“正常邮件”(输出)。

  2. 强化学习:强化学习则是一种基于环境反馈的学习过程,学习者(代理)与环境进行交互,通过试错误获得反馈(奖励或惩罚),从而制定出最优的策略以最大化累计奖励。例如,训练一个游戏代理在围棋中选择最佳的下一步.

主要区别

1. 学习目标

  • 监督学习的目标是最小化预测误差,它通常侧重于对特定数据样本的准确预测。
  • 强化学习的目标是最大化长期累计奖励,它更关注在整个决策过程中如何做出最佳选择。

2. 数据类型

  • 监督学习中,数据是已标注的,意味着每一个输入都有一个对应的输出标签。比如,图像分类中的每一张图像都有一个对应的标签。
  • 而在强化学习中,代理通过与环境的交互获得奖惩信号,数据不是事先准备好的,而是通过代理的行为而动态生成的。

3. 反馈机制

  • 监督学习中的反馈是直接的,模型在训练时会得到明确的正确答案(标签),例如在“猫”与“狗”的分类任务中,算法会直接知道它预测错误的图片。
  • 强化学习中,反馈是间接的,代理需要探索环境,行为的后果往往是延迟反馈。例如,在某个时刻采取的行动可能在后续多个时间步后才会知道其效果如何。

4. 环境交互

  • 监督学习的训练一般是离线的,模型训练完成后再进行测试,模型不会在训练过程中与环境互动。
  • 强化学习是在线的,模型在训练过程中就与环境交互,通过不断尝试不同的行动来学习。

案例分析

监督学习示例

考虑以下监督学习的Python代码示例,使用sklearn库进行简单的分类任务:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
from sklearn.datasets import load_iris
from sklearn.model_selection import train_test_split
from sklearn.ensemble import RandomForestClassifier
from sklearn.metrics import accuracy_score

# 加载数据集
iris = load_iris()
X, y = iris.data, iris.target

# 切分训练集与测试集
X_train, X_test, y_train, y_test = train_test_split(X, y, test_size=0.2, random_state=42)

# 训练模型
model = RandomForestClassifier()
model.fit(X_train, y_train)

# 预测
y_pred = model.predict(X_test)

# 计算准确率
accuracy = accuracy_score(y_test, y_pred)
print(f"准确率: {accuracy * 100:.2f}%")

强化学习示例

下面是一个使用OpenAI Gymstable-baselines3库的强化学习示例,训练一个简单的代理在CartPole环境中保持平衡:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
import gym
from stable_baselines3 import PPO

# 创建环境
env = gym.make('CartPole-v1')

# 创建代理
model = PPO("MlpPolicy", env, verbose=1)

# 训练代理
model.learn(total_timesteps=10000)

# 测试代理
obs = env.reset()
for _ in range(1000):
action, _states = model.predict(obs)
obs, rewards, done, info = env.step(action)
env.render()
if done:
obs = env.reset()

env.close()

总结

通过上述比较,我们可以看到强化学习与监督学习在多个方面的显著区别。这两种方法各有优势和应用场景,监督学习在数据相对容易获取和标注时表现优越,而强化学习则在需要与环境进行互动并依据反馈不断调整策略的场景中脱颖而出。在下一篇中,我们将探讨强化学习的应用领域,以进一步理解其实际用途和发展方向。

分享转发

3 强化学习导论之强化学习的应用领域

在深入了解强化学习(RL)的背景和与监督学习的区别后,我们将探讨强化学习在现实世界中的多样化应用领域。强化学习以其自适应性和自主性,已经成为解决复杂问题的重要工具。以下是一些主要的应用领域,以及相关案例分析。

游戏

案例分析:AlphaGo

作为强化学习应用的一个标志性案例,谷歌的 AlphaGo 通过使用深度强化学习算法,成功击败了世界围棋冠军李世石。这一过程展示了强化学习在策略优化和决策制定中的强大能力。AlphaGo 利用蒙特卡洛树搜索和深度神经网络来评估局面,并选择最佳的下一步棋。

代码示例

以下是一个简单的强化学习游戏策略示例,使用 Python 和 OpenAI 的 Gym 库:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
import gym
import numpy as np

# 创建一个环境
env = gym.make('CartPole-v1')

# 简单的Q-learning算法
Q = np.zeros((env.observation_space.shape[0], env.action_space.n))
learning_rate = 0.1
discount_factor = 0.95

for episode in range(2000):
state = env.reset()
done = False
while not done:
action = np.argmax(Q[state]) # 选择贪心策略
next_state, reward, done, _ = env.step(action)

# 更新Q值
Q[state, action] += learning_rate * (reward + discount_factor * np.max(Q[next_state]) - Q[state, action])
state = next_state

机器人控制

强化学习已被广泛应用于机器人技术,尤其是在自主导航和操作任务中。通过与环境的交互,机器人能够逐步学习到最佳的操作策略。

案例分析:仿人机器人

在一个项目中,研究者使用强化学习训练仿人机器人在复杂地形中行走。机器人最初随机尝试不同的步态,通过不断的学习和优化,它逐渐掌握了稳定行走的技巧,这在特定的任务中展现出了极大的灵活性和自适应能力。

金融交易

强化学习同样应用于金融市场,以优化交易策略和风险管理。

案例分析:高频交易

在高频交易中,使用强化学习来实时分析市场数据和动态决策。交易代理可以基于历史数据和当前市场态势,学习如何选择买入、卖出或保持不动,以实现最大收益。这种方法的有效性已经在多个市场策略测试中得到验证。

健康医疗

在医疗决策中,强化学习能够帮助医生制定更加个性化的治疗计划。

案例分析:个性化药物治疗

在某些研究中,强化学习用于优化药物剂量和治疗方案,以提高患者的恢复率和治疗效果。通过分析不同患者的反应和历史治疗数据,算法可以迭代学习出最佳的治疗策略,从而改善健康结果。

自动驾驶

自动驾驶技术是强化学习的另一个重要应用领域。车辆通过与道路环境的交互学习驾驶策略。

案例分析:Waymo和特斯拉

Waymo 和特斯拉等公司利用强化学习来优化自动驾驶决策。他们的系统通过不断的道路行驶数据进行训练,使得车辆能够在复杂的交通环境中做出迅速而安全的决策。例如,如何在红绿灯前做出最佳决策,或者如何避开突发的障碍物。

结论

通过以上应用领域的探讨,强化学习展现出了其在不同场景下的灵活性与适应性。未来,随着技术的不断进步,强化学习在更多领域的应用将有可能改变我们的生活和工作方式。接下来,我们将进入《Markov决策过程之MDP的定义和基本要素》的主题,进一步探讨强化学习的理论基础。

分享转发

4 Markov决策过程(MDP)的定义与基本要素

在上一篇文章中,我们探讨了强化学习的应用领域,了解到强化学习在多种实际问题中的广泛应用,例如游戏、机器人控制、财务决策等。而在强化学习的核心中,“Markov决策过程”(Markov Decision Process,简称MDP)是理解强化学习算法的重要基础。本文将详细介绍MDP的定义及其基本要素。

什么是MDP?

Markov决策过程是一个数学框架,用于描述在某一环境中,智能体(agent)如何通过选择动作来最大化某一累积奖励。MDP提供了一种形式化的方式来建模智能体与环境之间的交互。

一个MDP由以下五个基本要素定义:

  1. 状态集(S):代表系统可能的状态集合。智能体在每个时间步骤上都处于某个状态中。
  2. 动作集(A):代表智能体在每个状态下可以采取的动作集合。动作决定了智能体的行为,进而影响环境的状态。
  3. 状态转移概率(P):定义为在状态$s_t$下采取动作$a_t$后转移到状态$s_{t+1}$的概率,记作$P(s_{t+1} | s_t, a_t)$。这体现了环境的动态性和不确定性。
  4. 奖励函数(R):在状态$s_t$下采取动作$a_t$后,获得的即时奖励,记作$R(s_t, a_t)$. 奖励函数为智能体的学习过程提供反馈信息。
  5. 折扣因子($\gamma$):一个在区间$[0, 1]$上的值,决定了未来奖励的当前价值。折扣因子越接近1,未来奖励在当前的影响越大;越接近0,则倾向于关注短期奖励。

MDP的数学形式化

结合以上基本要素,MDP可以用五元组表示为:

$$
MDP = (S, A, P, R, \gamma)
$$

示例:简化的格子世界

考虑一个简单的“格子世界”作为案例。假设有一个5x5的网格,智能体可以在其中移动。我们来看看如何用MDP来描述这个环境。

  • 状态集 $S$:该状态集包含25个状态,分别对应网格中的每一个格子。
  • 动作集 $A$:智能体在每个格子中可以选择的动作包括“上”、“下”、“左”、“右”四种移动。
  • 状态转移概率 $P$:假设智能体在状态$s_t$下选择“右”动作,概率1.0转移到状态$s_{t+1}$(即下一个格子),如果边界限制,则状态保持不变。
  • 奖励函数 $R$:智能体在到达某个目标格(例如位置(4, 4))时获得奖励+10;在每个时间步骤上移动的成本为-1。因此,对于每一步$R(s, a) = -1$,在到达目标后是$R(s, a) = 10$。
  • 折扣因子 $\gamma$:设定为0.9,以重视更长期的奖励。

总结

通过MDP的框架,我们可以清晰地对智能体的学习过程进行形式化描述。定义状态、动作、转移概率、奖励和折扣因子,使我们能够更好地理解和设计强化学习算法。

在下一篇文章中,我们将进一步探讨MDP的基本组成部分——状态、动作和奖励。这将为我们后续的强化学习算法实现奠定重要基础。

分享转发

5 Markov决策过程之状态、动作和奖励

在本篇教程中,我们将深入探讨Markov决策过程(MDP)的核心组成部分:状态动作奖励。这些元素是理解MDP的基础,也是强化学习中智能体决策的支柱。

一、状态(State)

在MDP中,状态是环境在某一时刻的描述。它应该能够提供足够的信息,以便智能体做出合理的决策。一个状态可以是任何对环境的表征,可能包括某个游戏中的棋盘状况、机器人在地图上的位置等。

案例:迷宫问题

假设我们有一个简单的迷宫,迷宫的不同位置可以表示为不同的状态。例如,迷宫由一个$3 \times 3$的网格构成,每个格子表示一个状态:

1
2
3
(0, 0) (0, 1) (0, 2)
(1, 0) (1, 1) (1, 2)
(2, 0) (2, 1) (2, 2)

在这个迷宫中,智能体的位置就是当前状态。

二、动作(Action)

动作是智能体可以在特定状态下采取的选择。在某一状态下,智能体可以执行一个或多个可用的动作,来改变其状态。每个动作都有可能导致智能体转移到另一个状态。

动作的选择

在迷宫的例子中,如果智能体在状态$(1, 1)$,它可以选择的动作可能是上(Up)下(Down)左(Left)右(Right)。每个动作都会导致状态的变化。例如:

  • 执行动作从$(1, 1)$到$(0, 1)$
  • 执行动作从$(1, 1)$到$(2, 1)$
  • 执行动作从$(1, 1)$到$(1, 0)$
  • 执行动作从$(1, 1)$到$(1, 2)$

可以使用一个字典来表示这些动作及其对应的状态转移,如下所示:

1
2
3
4
5
6
7
8
action_transition = {
(1, 1): {
'Up': (0, 1),
'Down': (2, 1),
'Left': (1, 0),
'Right': (1, 2)
}
}

三、奖励(Reward)

在MDP中,奖励是环境给予智能体的反馈,用于评估特定状态与动作的组合。奖励可以是正值、负值或零,反映了智能体在某个状态下执行某个动作的好坏。通过奖励,智能体能够学习哪些行为是有益的,哪些是有害的。

奖励的设计

在迷宫中,假设智能体到达出口会获得奖励+10,而走入死胡同会获得奖励-5,其他状态的奖励都是0。我们可以使用一个奖励函数来定义这一过程:

1
2
3
4
5
6
7
8
9
10
11
rewards = {
(0, 0): 0,
(0, 1): 0,
(0, 2): 0,
(1, 0): 0,
(1, 1): 0,
(1, 2): 0,
(2, 0): -5,
(2, 1): 0,
(2, 2): 10
}

总结

本篇教程中,我们详细介绍了Markov决策过程的三个关键要素:状态动作奖励。通过迷宫问题的示例,我们展示了这些要素是如何相互作用的。智能体在不同的状态下,通过执行不同的动作获取相应的奖励,从而学习到最优策略,为下一步的学习打下了基础。

在接下来的上一篇教程中,我们将讨论折扣因子与价值函数,进一步探讨如何评估和优化智能体的决策过程。

分享转发

6 Markov决策过程之折扣因子与价值函数

在上一篇文章中,我们探讨了 Markov决策过程(MDP)的基本概念,包括状态、动作和奖励。这些构成了强化学习的基础框架。在本篇中,我们将深入讨论 MDP 中的重要元素之一:折扣因子与价值函数。这些概念不仅是理论上的重要工具,而且在实际应用中也具有重要的意义。

价值函数

在强化学习中,价值函数用于评估某一状态或状态-动作对的价值。一般而言,价值函数可以分为两类:

  1. 状态价值函数 $V(s)$:给定一个状态 $s$,它表示从该状态出发,在策略 $\pi$ 下,未来所能获得的期望回报。
  2. 动作价值函数 $Q(s, a)$:给定一个状态 $s$ 和一个动作 $a$,它表示在状态 $s$ 下执行动作 $a$,然后遵循策略 $\pi$ 所能获得的期望回报。

状态价值函数和动作价值函数的计算公式如下:

  • 状态价值函数:
    $$
    V_\pi(s) = \mathbb{E}\pi \left[ \sum{t=0}^{\infty} \gamma^t R_t \mid S_0 = s \right]
    $$

  • 动作价值函数:
    $$
    Q_\pi(s, a) = \mathbb{E}\pi \left[ \sum{t=0}^{\infty} \gamma^t R_t \mid S_0 = s, A_0 = a \right]
    $$

在上面的公式中:

  • $R_t$ 是在时间步 $t$ 获得的即时奖励。
  • $\gamma$ 是 折扣因子,其值在 $[0, 1]$ 的范围内。

折扣因子

折扣因子 $\gamma$ 是 MDP 中一个重要的超参数,它决定了未来奖励的当前值。其物理意义在于,随着时间的推移,未来的奖励会被“折扣”到现在的价值。具体来说:

  • 当 $\gamma$ 接近 $1$ 时,未来的奖励与当前的奖励几乎同等重要,模型将倾向于追求长期回报。
  • 当 $\gamma$ 接近 $0$ 时,模型更看重当前的奖励,短期决策将成为优先考虑的因素。

案例分析

假设我们有一个简单的游戏,玩家在一个 1 到 10 的数字上进行“拿奖励”的游戏。每个时刻玩家都有机会选择一个数字 $x$,获取该数字的奖励,而游戏会在 $10$ 轮后结束。

选择一个折扣因子 $\gamma = 0.9$ 和一个奖励序列 $R_0, R_1, \ldots, R_9$,我们可以计算每一步的状态价值。假设奖励序列为 $R_t = 10 - t$,那么我们可以通过以下方式计算从初始状态 $S_0$ 取得的折现价值:

1
2
3
4
5
6
7
8
9
10
def compute_value(gamma, rewards):
total_value = 0
for t in range(len(rewards)):
total_value += (gamma ** t) * rewards[t]
return total_value

rewards = [10, 9, 8, 7, 6, 5, 4, 3, 2, 1]
gamma = 0.9
value = compute_value(gamma, rewards)
print("折现价值:", value)

执行上述代码,我们将获得该游戏的总折现价值。

小结

在本篇中,我们深入探讨了 Markov决策过程中的折扣因子与价值函数,它们不仅为我们提供了量化决策的工具,也为强化学习算法的设计奠定了基础。我们看到,折扣因子的选择直接影响到代理(Agent)的决策行为,在不同的环境中可能需要进行不同的调整。

下一篇文章将继续讨论动态规划,它是强化学习中的一种重要方法,我们将介绍其基本思想和框架,如何利用动态规划来解决 MDP 的问题。

分享转发

7 动态规划的基本思想和框架

在强化学习中,动态规划(Dynamic Programming, DP)是解决优化问题的重要方法。它为我们提供了一种系统的方法来处理具有阶段性决策的问题。在上一篇文章中,我们介绍了马尔可夫决策过程(MDP)中的折扣因子和价值函数,这些概念是理解动态规划的基础。在本篇中,我们将探讨动态规划的基本思想和框架,为后续的值迭代算法奠定基础。

动态规划的基本思想

动态规划的核心思想是“递归分解”。针对某个复杂问题,动态规划将其划分为多个子问题,解决这些子问题后再合并结果,从而获得原问题的解。这种方法特别适用于具有重叠子问题和最优子结构性质的问题。

  • 重叠子问题:子问题多次出现,通过保存已经解决的子问题的结果,可以避免重复计算,提高效率。
  • 最优子结构:一个问题的最优解包含了其子问题的最优解。

动态规划主要用于寻找在某种意义上最优的解决方案,通常涉及用了某种评价标准来进行优化,比如最小化成本,最大化收益等。

案例分析:最短路径问题

考虑一个简单的最短路径问题,假设我们有一个加权有向图,其中每个边都有一个非负的权重。我们的目标是找到从起点到终点的最短路径。

  1. 子问题定义:设cost(s, t)为从节点s到节点t的最短路径成本。对于每一个节点s,我们需要计算cost(s, t)

  2. 递归关系:对于节点st的路径,如果我们中途选择经过节点u,那么可以表示为:
    $$
    cost(s, t) = \min_u (cost(s, u) + cost(u, t))
    $$
    这个式子表明,从st的最短路径成本是通过某个中间节点u的路径成本的最小值。

  3. 边界条件:若cost(s, s) = 0(从一个点到自身不需要成本),如果节点之间没有连接,则cost(s, t) = \infty

动态规划的框架

在解决动态规划问题时,我们通常遵循以下步骤:

  1. 定义问题:明确需要解决的问题以及目标。
  2. 划分子问题:将原问题分解为多个更小的子问题。
  3. 建立递推关系:找出子问题之间的关系,通常使用递归公式。
  4. 计算顺序:根据递推关系,以合适的顺序计算所有子问题的解,确保每个子问题在需要使用之前都已解决。
  5. 构造最终解:根据子问题的结果构造出原问题的解。

动态规划的实现

以下是用 Python 实现的一个简化的动态规划算法,通过实例找出从起点到终点的最短路径。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
import numpy as np

def shortest_path(graph, start, end):
num_nodes = len(graph)
# 初始化成本矩阵
cost = np.full((num_nodes, num_nodes), np.inf)

# 从自身到自身的成本为0
for i in range(num_nodes):
cost[i][i] = 0

# 填充邻接矩阵
for u in range(num_nodes):
for v, weight in graph[u].items():
cost[u][v] = weight

# 动态规划计算最短路径成本
for k in range(num_nodes):
for i in range(num_nodes):
for j in range(num_nodes):
# 更新成本
if cost[i][j] > cost[i][k] + cost[k][j]:
cost[i][j] = cost[i][k] + cost[k][j]

# 返回从 start 到 end 的最短路径
return cost[start][end]

# 示例图:邻接矩阵表示
# graph[u] = {v: weight} 表示从 u 到 v 的边权重
graph = [
{1: 1, 2: 4},
{2: 2, 3: 6},
{3: 1},
{}
]

start_node = 0
end_node = 3
result = shortest_path(graph, start_node, end_node)
print(f"从节点 {start_node} 到节点 {end_node} 的最短路径成本是: {result}")

小结

动态规划为解决复杂的优化问题提供了一种有效的方法。通过系统地分解问题,建立递推关系,我们能够在合理的时间内找到最优解。接下来,我们将介绍动态规划中特别重要的一个算法——值迭代算法,它在基于动态规划的学习中起着重要作用。

如果你对动态规划的基本思想和框架有更深入的理解,将为后续学习值迭代算法做好准备。

分享转发

8 强化学习从零学教程系列之动态规划之值迭代算法

在上一篇教程中,我们探讨了动态规划的基本思想和框架,为我们后续学习强化学习打下了坚实的基础。在本篇中,我们将深入了解动态规划的一种具体实现——值迭代算法。这一算法在解决马尔可夫决策过程(MDP)中的最优策略时,提供了一种有效的计算手段。

值迭代算法概述

值迭代算法是一种在给定状态空间和动作空间的条件下,通过递归地更新状态价值,从而逐步收敛到最优值函数的动态规划方法。值迭代不仅可以用于求解最优价值函数,还能获取最优策略。

算法步骤

值迭代算法的主要步骤如下:

  1. 初始化:为所有状态初始化一个任意的价值函数,通常是将所有状态的价值设为0。
  2. 更新价值函数:根据贝尔曼方程更新每个状态的价值函数:
    $$
    V_{i+1}(s) = \max_{a \in A} \sum_{s’ \in S} P(s’|s, a)[R(s, a, s’) + \gamma V_i(s’)]
    $$
    其中,$V_i(s)$ 是状态$s$ 在第$i$次更新时的价值,$P(s’|s, a)$ 是在状态$s$下采取动作$a$后转移到状态$s’$的概率,$R(s, a, s’)$ 是在此转移中获得的奖励,$\gamma$ 是折扣因子。
  3. 收敛判断:不断迭代更新价值函数,直到价值函数的变化小于定义好的阈值(如0.01)。
  4. 获取策略:依据最终的价值函数,推导出最优策略:
    $$
    \pi^*(s) = \arg \max_{a \in A} \sum_{s’ \in S} P(s’|s, a)[R(s, a, s’) + \gamma V(s’)]
    $$

示例:网格世界

为更好地理解值迭代算法,我们通过一个简化的示例——网格世界来说明。

问题描述

假设我们有一个$4 \times 4$的网格世界,智能体在每个状态(网格单元)中可以选择向上、向下、向左或向右移动。目标是使智能体尽快到达目标状态(右下角),并且每次移动都要消耗$1$的奖励。目标状态的奖励为$0$,其他状态的奖励均为$-1$。

设置参数

  • 状态空间$S$:包含所有的$16$个网格。
  • 动作空间$A$:包含updownleftright
  • 折扣因子$\gamma$:取$0.9$。

值迭代实现

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
import numpy as np

# 定义环境参数
grid_size = 4
num_actions = 4
gamma = 0.9
theta = 0.01

# 定义奖励结构
R = -np.ones((grid_size, grid_size))
R[3, 3] = 0 # 目标状态的奖励

# 定义状态转移概率(简单处理,假设概率为1)
def get_next_states(s, a):
x, y = s
if a == 0: # up
return max(x-1, 0), y
elif a == 1: # down
return min(x+1, grid_size-1), y
elif a == 2: # left
return x, max(y-1, 0)
elif a == 3: # right
return x, min(y+1, grid_size-1)

# 值迭代函数
def value_iteration():
V = np.zeros((grid_size, grid_size))
while True:
delta = 0
for x in range(grid_size):
for y in range(grid_size):
v = V[x, y]
V[x, y] = max(sum(1 * (R[get_next_states((x, y), a)] + gamma * V[get_next_states((x, y), a)]))
for a in range(num_actions))
delta = max(delta, abs(v - V[x, y]))
if delta < theta:
break
return V

# 运行值迭代
V = value_iteration()
print(V)

在以上代码中,我们通过一个简单的数组V来维持各状态的价值。每次我们都会更新状态价值,直到所有状态的价值收敛。最终返回的V就是每个状态对应的最优价值。

策略提取

一旦获取到最终的价值函数,我们可以根据上述策略公式,从每个状态推导出最优策略。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
def extract_policy(V):
policy = np.zeros((grid_size, grid_size), dtype=int)
for x in range(grid_size):
for y in range(grid_size):
best_action = None
best_value = float('-inf')
for a in range(num_actions):
next_state = get_next_states((x, y), a)
value = R[next_state] + gamma * V[next_state]
if value > best_value:
best_value = value
best_action = a
policy[x, y] = best_action
return policy

# 提取最优策略
policy = extract_policy(V)
print(policy)

在策略提取的过程中,我们遍历每个状态,选择能够带来最大价值的动作,并形成最终的policy。每个动作可以用数字表示,比如0代表up1代表down2代表left3代表right

总结

本文详细介绍了值迭代算法的工作机制,并通过实践案例展示了如何在一个简单的网格世界中实现它。值迭代是动态规划的重要组成部分,广泛应用于强化学习中。掌握了这一算法后,我们将在下一篇教程中进一步探讨动态规划的另一个经典算法——策略迭代算法。希望大家能在实践中不断探索并深化对强化学习的理解!

分享转发

9 动态规划之策略迭代算法

在本篇中,我们将深入探讨强化学习中的策略迭代算法,这是动态规划的一种重要方法。在上一篇中,我们介绍了值迭代算法,并了解了如何通过计算状态值来优化策略。而在这一篇中,我们将重点关注如何通过“策略迭代”来直接改善策略。

策略与价值

在强化学习中,策略(Policy)是智能体在每个状态下所采取的行动的概率分布。策略可以是“确定性”的,即在某一状态下采取唯一的行动,也可以是“随机”的,即在某一状态下以一定概率随机选择行动。在策略迭代中,我们将交替进行策略评估和策略改进。

  1. 策略评估:给定一个策略,计算其在当前策略下每个状态的值。
  2. 策略改进:在评估基础上,通过选择最优的动作来改进该策略。

算法步骤

策略迭代算法的基本步骤如下:

  1. 初始化策略:随机选择一个初始策略。
  2. 策略评估:计算当前策略下每个状态的价值函数$V^\pi(s)$,直到收敛。
  3. 策略改进:通过选择使得价值函数最大的行动来改进策略,即
    $$
    \pi_{\text{new}}(s) = \arg\max_a Q(s, a)
    $$
    其中 $Q(s, a)$ 为动作价值函数。
  4. 重复步骤 2 和 3,直到策略不再改变。

案例:格子世界

假设我们有一个简单的格子世界,智能体在一个 $4 \times 4$ 的方格中行动。智能体的目标是从起始点(左上角)到达终点(右下角),在过程中获得奖励。我们设定在每个动作上都有一个$s$的奖励和一个小的惩罚。

环境设定

  • 状态 $S$: 四个位置的格子(共16个状态)
  • 动作 $A$: 上、下、左、右(4个动作)
  • 奖励: 到达终点的奖励 +1,其它状态-0.01

算法实现

下面是策略迭代算法的简单代码实现:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
import numpy as np

# 状态和动作定义
grid_size = 4
n_states = grid_size * grid_size
n_actions = 4 # 上、下、左、右

# 奖励设定
rewards = np.full((grid_size, grid_size), -0.01)
rewards[3, 3] = 1 # 终点奖励

# 初始化策略和价值函数
policy = np.zeros((grid_size, grid_size), dtype=int) # 随机初始化策略
V = np.zeros((grid_size, grid_size)) # 状态值初始化

def get_next_state(state, action):
row, col = divmod(state, grid_size)
if action == 0: # 上
row = max(0, row - 1)
elif action == 1: # 下
row = min(grid_size - 1, row + 1)
elif action == 2: # 左
col = max(0, col - 1)
elif action == 3: # 右
col = min(grid_size - 1, col + 1)
return row * grid_size + col

# 策略评估
def policy_evaluation(policy):
while True:
delta = 0
for state in range(n_states):
v = V[state // grid_size, state % grid_size]
action = policy[state // grid_size, state % grid_size]
V[state // grid_size, state % grid_size] = rewards[state // grid_size, state % grid_size] + \
V[get_next_state(state, action) // grid_size, get_next_state(state, action) % grid_size]
delta = max(delta, abs(v - V[state // grid_size, state % grid_size]))
if delta < 1e-4: # 收敛条件
break

# 策略改进
def policy_improvement():
policy_stable = True
for state in range(n_states):
old_action = policy[state // grid_size, state % grid_size]
action_values = np.zeros(n_actions)
for action in range(n_actions):
next_state = get_next_state(state, action)
action_values[action] = rewards[state // grid_size, state % grid_size] + \
V[next_state // grid_size, next_state % grid_size]
policy[state // grid_size, state % grid_size] = np.argmax(action_values)
if old_action != policy[state // grid_size, state % grid_size]:
policy_stable = False
return policy_stable

# 主循环
while True:
policy_evaluation(policy)
if policy_improvement():
break

print("最终策略:")
print(policy)
print("状态值:")
print(V)

结果分析

运行上述代码后,我们可以得到智能体的最终策略和对应的状态值。智能体将会通过策略迭代找到从起始点到达终点的最佳路径。

总结

策略迭代算法通过交替进行策略评估和策略改进,可以有效地找到最优策略。相较于值迭代,策略迭代在许多情况下收敛更快,因为它在每一步都在不断优化 “所有状态”的策略。

在接下来的章节中,我们将继续探讨蒙特卡罗方法的基本原理,进一步丰富我们的强化学习知识体系。通过不同方法的对比与结合,帮助我们更深入地理解强化学习的核心思想。

分享转发

10 蒙特卡罗方法的基本原理

在强化学习的领域,蒙特卡罗方法是评估和改进策略的重要工具。它利用随机采样的结果来估计状态价值或策略的价值,并通过对这些结果的分析来进行策略的更新。本章将详细介绍蒙特卡罗方法的基本原理,以及如何将其应用于具体的强化学习任务。

蒙特卡罗方法的基本概念

蒙特卡罗方法的核心思想是利用随机采样来解决问题。在强化学习中,通常会面临从环境中获取响应和奖励的任务。我们通常需要知道某一策略下,从某个状态开始,到达终局状态所获得的预期回报。这个过程可以通过多次实验来进行估计。

一、基本要素

在使用蒙特卡罗方法时,我们需要关注以下几个关键的要素:

  1. 试验(Episode): 一次完整的环境交互过程,从初始状态开始,直到达到终止状态。

  2. 回报(Return): 从某个状态出发获得的总奖励,通常定义为从该状态开始的所有未来奖励的折扣和。假设$\gamma$是折扣因子,则从某状态$s$开始的回报为:

    $$
    G_t = R_{t+1} + \gamma R_{t+2} + \gamma^2 R_{t+3} + \ldots
    $$

  3. 价值函数(Value Function): 为了评估某个策略的好坏,我们定义状态$s$的价值为在策略下从状态$s$出发的所有回报的期望值。我可以用如下公式表示:

    $$
    V(s) = \mathbb{E}[G_t | s]
    $$

二、蒙特卡罗估计

蒙特卡罗方法通过多次试验获得回报,然后计算这些回报的平均值来估计状态价值。假设对状态$s$进行$n$次独立的试验,得到的回报为$G_1, G_2, \ldots, G_n$,则状态$s$的价值估计可以表示为:

$$
V(s) \approx \frac{1}{n} \sum_{i=1}^n G_i
$$

三、算法步骤

以下是使用蒙特卡罗方法的基本步骤:

  1. 初始化:选择一个策略$\pi$,为所有状态初始化价值函数$V(s)$。
  2. 生成试验:与环境进行交互,生成多个完整的试验,记录状态及获得的奖励。
  3. 计算回报:对每一个状态$s$,记录其在试验中出现的情况,并计算回报$G_t$。
  4. 更新价值函数:根据采集到的回报更新价值函数。

案例分析

我们来看看一个具体的案例,通过一个简单的迷宫游戏来更好地理解蒙特卡罗方法的应用。在这个环境中,我们的目标是从起点到达终点,同时尽量减少获得的惩罚。

环境描述

假设我们有一个简单的$3 \times 3$的迷宫,每一个格子代表一个状态,起点在$(0, 0)$,终点在$(2, 2)$。每次移动都有概率获得相应的奖励或惩罚。我们给予到达终点一个奖励+1,走错路线的惩罚为-1,其他格子为0。

代码实现

下面的Python示例展示了如何使用蒙特卡罗方法来估计状态价值。在此示例中,我们将进行多次试验,模拟在迷宫中的随机行动。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
import numpy as np

# 定义奖励结构
rewards = np.array([[0, 0, 0],
[0, 0, 0],
[0, 0, 1]])

# 状态价值初始化
V = np.zeros((3, 3))
num_episodes = 1000

# 蒙特卡罗方法
for _ in range(num_episodes):
state = (0, 0) # 起始状态
episode_rewards = []

while state != (2, 2):
# 随机选择下一个动作
action = np.random.choice(["up", "down", "left", "right"])
if action == "up" and state[0] > 0:
state = (state[0] - 1, state[1])
elif action == "down" and state[0] < 2:
state = (state[0] + 1, state[1])
elif action == "left" and state[1] > 0:
state = (state[0], state[1] - 1)
elif action == "right" and state[1] < 2:
state = (state[0], state[1] + 1)

# 记录奖励
episode_rewards.append(rewards[state])

# 计算回报
G = sum(episode_rewards) # 简化的回报计算
V[0, 0] += G # 更新起始状态的价值(这里没有平均,作为基本示例)

# 输出价值函数
print("状态价值函数:")
print(V)

在上述代码中,我们简单模拟了在一个$3 \times 3$迷宫中行走的过程。通过$1000$次试验,我们不断更新状态价值函数$V$。虽然这里的更新方式是非常简单的,但可以通过引入更复杂的策略和更新规则来逐步改进。

四、总结

蒙特卡罗方法是强化学习中一种强大且灵活的工具,利用随机试验来估计策略的性能,并通过这些估计来改进策略。虽然简单的蒙特卡罗方法可能在效率上不如其他方法(如时间差分学习),但它的基本思想和应用场景在实际问题中非常重要。

在接下来的章节中,我们将探讨蒙特卡罗控制方法,以及如何通过这种方法来优化策略,使得我们能够在实际应用中获得更好的决策能力。

分享转发

11 蒙特卡罗控制方法概述

在上一篇中,我们探讨了蒙特卡罗方法的基本原理。这一部分将深入讨论蒙特卡罗控制方法,进一步拓展我们对强化学习的理解。蒙特卡罗控制是指通过蒙特卡罗方法进行策略评估和改进的过程,它主要用于策略的优化。

蒙特卡罗控制的基本概念

蒙特卡罗控制的目标是通过对状态-动作值函数($Q$值函数)的估计来找到最优策略。它的基本流程是:

  1. 采样:使用策略生成多个轨迹(episode),每个轨迹由状态、动作和奖励序列组成。
  2. 评估:计算每一对 $(s, a)$ 的$Q$值,即在状态$s$下采取动作$a$的期望回报。
  3. 改进:根据$Q$值更新策略,使概率更高地选择在给定状态下收益更高的动作。

蒙特卡罗控制的实现步骤

步骤 1: 生成轨迹

在强化学习中,我们需要通过与环境的交互来获得轨迹。以下是一个简单的示例,展示如何在一个简单的环境中生成轨迹。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
import numpy as np

def generate_episode(env, policy):
state = env.reset()
episode = []
done = False

while not done:
action = policy(state)
next_state, reward, done, info = env.step(action)
episode.append((state, action, reward))
state = next_state

return episode

在这个代码示例中,generate_episode 函数生成一个完整的轨迹,使用给定的策略与环境进行交互。

步骤 2: 评估$Q$值函数

一旦获得了多条轨迹,我们可以开始评估$Q$值。在这里,我们将计算每个状态-动作对的回报。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
def compute_Q(episodes, num_states, num_actions, discount_factor=0.9):
Q = np.zeros((num_states, num_actions))
returns = np.zeros((num_states, num_actions))
returns_count = np.zeros((num_states, num_actions))

for episode in episodes:
G = 0 # 回报
multiplier = 1 # 折扣因子
for state, action, reward in reversed(episode):
G += reward * multiplier
multiplier *= discount_factor

# 更新返回值与计数
returns[state, action] += G
returns_count[state, action] += 1
Q[state, action] = returns[state, action] / returns_count[state, action] # 计算平均

return Q

在这个函数中,我们计算每个状态-动作对的$Q$值,通过回报的累积进行评估。

步骤 3: 改进策略

基于更新后的$Q$值,我们可以通过$\epsilon$-贪婪策略来改进当前策略。这是强化学习中常用的策略改进方法。

1
2
3
4
5
6
7
def epsilon_greedy_policy(Q, epsilon=0.1):
def policy(state):
if np.random.rand() < epsilon:
return np.random.choice(len(Q[state])) # 随机动作
else:
return np.argmax(Q[state]) # 选择最佳动作
return policy

在上述代码中,epsilon_greedy_policy函数定义了一个$\epsilon$-贪婪策略,这种策略在探索 (选择随机动作) 和利用 (选择Q值最高的动作)之间平衡。

循环迭代

最终,我们可以将这些步骤放在一个循环中,迭代进行策略评估与改进,直到策略收敛。

1
2
3
4
5
6
7
8
9
10
def monte_carlo_control(env, num_episodes, discount_factor=0.9):
Q = np.zeros((env.observation_space.n, env.action_space.n))
policy = epsilon_greedy_policy(Q)

for episode_num in range(num_episodes):
episode = generate_episode(env, policy)
Q = compute_Q([episode], env.observation_space.n, env.action_space.n, discount_factor)
policy = epsilon_greedy_policy(Q)

return policy, Q

实际案例:简单的网格世界

为了更好地理解蒙特卡罗控制方法,我们可以考虑一个简单的网格世界环境,其中代理可以在一个$5 \times 5$的网格中移动,每个格子可以得到相应的奖励。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
class GridWorld:
def __init__(self):
self.grid_size = 5
self.state = (0, 0)

def reset(self):
self.state = (0, 0)
return self.state

def step(self, action):
# 定义动作:上, 下, 左, 右
if action == 0: # 上
next_state = (max(0, self.state[0] - 1), self.state[1])
elif action == 1: # 下
next_state = (min(self.grid_size - 1, self.state[0] + 1), self.state[1])
elif action == 2: # 左
next_state = (self.state[0], max(0, self.state[1] - 1))
else: # 右
next_state = (self.state[0], min(self.grid_size - 1, self.state[1] + 1))

self.state = next_state
# 设定奖励
reward = -1 if self.state != (self.grid_size - 1, self.grid_size - 1) else 0
done = self.state == (self.grid_size - 1, self.grid_size - 1)

return self.state, reward, done, {}

# 使用网格世界环境进行蒙特卡罗控制
env = GridWorld()
optimal_policy, Q_values = monte_carlo_control(env, num_episodes=5000)

在这个示例中,代理在网格中移动,获取奖励,最终通过蒙特卡罗控制方法学习到一个近似最优的策略。

总结

蒙特卡罗控制方法是通过采样生成轨迹并使用$Q$值进行策略评估与改进的有力工具。它稳健且易于实现,适合用于强化学习的各种应用场景。在我们下一篇的内容中,我们将进一步讨论如何进行区间估计,以提高对强化学习模型的评估与理解。

分享转发

12 区间估计

在上一篇中,我们探讨了蒙特卡罗控制方法的基本概念和应用。在进行强化学习时,我们经常需要对某些参数进行估计,而区间估计则是对这些估计结果不确定性的一种量化方式。接下来,我们将深入探讨蒙特卡罗方法中的区间估计。

区间估计的重要性

在强化学习中,尤其涉及到策略评估时,理解和量化一些量的不确定性是非常重要的。通过区间估计,我们可以为我们的估计值提供一个置信区间,这样可以更好地指导我们的决策。

蒙特卡罗方法的回顾

首先,我们快速回顾一下蒙特卡罗方法。蒙特卡罗方法是通过随机采样来估计函数的期望值。其基本思想是:

  1. 根据当前策略,生成多个轨迹(序列);
  2. 计算每个轨迹的回报;
  3. 从多个轨迹中提取信息以更新我们的估计。

例如,在一个简单的环境中,我们可能会从每个状态开始多次试验,并记录每次试验的总回报。

确定区间估计

在蒙特卡罗方法中,我们通常关注的是回报的均值。设 $R$ 为从某个状态下的回报的集合。我们可以用样本均值 $\bar{R}$ 来表示:

$$
\bar{R} = \frac{1}{N}\sum_{i=1}^{N} R_i
$$

其中 $N$ 是样本数量,$R_i$ 是第 $i$ 个样本的回报。

置信区间的构建

为了构建置信区间,我们需要用到样本标准差。样本标准差可以由下式计算:

$$
s = \sqrt{\frac{1}{N-1}\sum_{i=1}^{N}(R_i - \bar{R})^2}
$$

根据正态分布的性质,我们可以使用这个标准差来构建置信区间。对于一个给定的置信水平(例如 95%),置信区间可以表示为:

$$
\left[\bar{R} - t_{1-\alpha/2} \cdot \frac{s}{\sqrt{N}}, , \bar{R} + t_{1-\alpha/2} \cdot \frac{s}{\sqrt{N}}\right]
$$

其中 $t_{1-\alpha/2}$ 是 t 分布表中的临界值,它依赖于样本大小和所选择的置信水平。

实例:区间估计的实际应用

让我们通过一个简单的 Python 代码示例来看如何实现蒙特卡罗区间估计。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
import numpy as np
import scipy.stats as stats

# 设置随机种子以保证结果可重复
np.random.seed(42)

# 假设回报来自于某个分布的样本
N = 1000
true_mean = 10
true_std = 2
rewards = np.random.normal(true_mean, true_std, N)

# 计算样本均值和标准差
sample_mean = np.mean(rewards)
sample_std = np.std(rewards, ddof=1)

# 计算95%的置信区间
confidence_level = 0.95
alpha = 1 - confidence_level
t_critical = stats.t.ppf(1 - alpha/2, N - 1)

margin_of_error = t_critical * (sample_std / np.sqrt(N))
confidence_interval = (sample_mean - margin_of_error, sample_mean + margin_of_error)

print(f"Sample Mean: {sample_mean:.2f}")
print(f"95% Confidence Interval: {confidence_interval}")

在这个代码示例中,我们生成了 1000 个来自正态分布的回报样本,计算了样本均值和样本标准差,并基于这些数据构建了 95% 的置信区间。运行代码后会输出样本均值和相应的置信区间。

总结

通过使用蒙特卡罗方法的区间估计,我们能够为强化学习中的策略评估提供更强的理论支持与实用性。在实际应用中,引入区间估计的过程有助于我们更全面地理解模型的性能及其不确定性。在下一篇教程中,我们将探讨时序差分学习的基本概念,敬请期待!

分享转发