前面几章已经讲了model-based (如:动态规划)和model-free (蒙特卡洛、时序差分)方法。这两种方法的区别在于计算价值函数的时候是否已知模型(这里指状态转移概率)。虽然这两种强化学习方法不同,但也有相同的地方,比如:两种方法的核心都在于计算价值函数,还都是根据未来的奖励来估算当前状态价值。
既然有相同之处,这两种强化学习方法能否统一起来呢?这就是本章的内容,其中,model-based方法视为规划方法,model-free方法视为学习方法。
模型是指能够对agent所处状态和动作做出反应的东西。也可以理解为模型是模仿环境来做一些反应的。
常见的有两种不同类型的模型,一种是抽样模型,一种是分布模型。
模型可以用来仿真经验,即agent不需要和真实环境交互,而是跟模型交互。对于抽样模型来说,给定初始状态和动作,与模型交互,就可以得到这个经验序列(episode)。
规划是指通过agent和模型交互,产生仿真的经验序列,根据经验序列更新价值函数,更新策略。过程如下:
动态规划就满足上面的规划过程,其模型是分布模型。对于一个状态-动作对,模型会遍历所有可能的奖励和所有可能到达的下一个状态。利用这些状态转移来计算价值函数,并且迭代得到最优策略。
可以看出,规划和学习的区别在于:规划使用的是模型产生的模拟的经验;学习是使用agent和环境交互产生的真实经验。
下面给出了利用表格的一步规划算法过程,可以看出跟之前学过的Q-learning算法的唯一区别就在于状态转移是根据采样模型得到的。而Q-learning中,状态转移是和真实环境交互得到的。
从上面的内容中可以看出,规划和学习的区别仅在于真实经验和仿真经验上。但两者都是可以用来计算价值函数、更新策略的,所以基于这样同一个目标,我们可以将规划和学习统一起来。框架如下:
上图可以看到,experience扮演了两个角色,一个是direct RL,即直接根据经验计算价值函数、更新策略。另一个是利用经验学习模型,然后利用模型进行规划,简介计算价值函数、更新策略。
图中中间部分是和环境交互得到的真实经验,向左的箭头是direct RL,向右的indirect RL。
下面给出Dyna计算价值函数的伪代码,(d)是根据真实经验计算动作价值的,(f)是根据模拟经验计算动作价值的。
例子:方格游戏
任务目标是从起点S以最短的路径到达终点G,其中阴影的部分表示障碍物。这里主要想用这个例子来展示规划对于该任务的作用。
在上图中,蓝色的线是指没有进行规划的,仅使用真实经验计算价值函数、更新策略。绿色的线表示每进行一个direct RL,会进行5次规划。红色的线则是每进行一个direct RL,会进行50次规划。
可以看出,红色的线在刚开始的几个episode中就能够以很少的步数到达终点。
原因如下:
在第二个episode时,对于没有规划的学习来说,只有到达终点之前的一个状态的价值会更新,其他的还是0。但是对于有50步规划的方法来说,第二个episode就已经可以更新大部分状态的价值。
模型其实是对环境的一个估计,模拟环境生成经验。但环境通常是随机的,会动态变化的。当环境发生变化,但模型还没有及时检测并调整,或是模型学到的价值函数不准确时,就会发生模型的错误。
当模型不正确时,规划过程会计算得到一个次优的策略,该策略会使得模型的错误更快的被发现和修正。下面给出两个例子,阐述在环境发生变化时,Dyna-Q等算法的表现。
下图展示环境变化情况和各算法的调整。上左图是初始环境,从起始点S到终点G之间有右边的一条路可以连通。但在第1000个时间步,环境发生改变,原来的障碍物右移了一格,导致连通的路径改变。图的下半部分展示了两种算法随时间步的增加,累计奖励的变化情况。
其中,Dyna-Q算法是上一节中介绍的结合学习和规划的算法。Dyna-Q+算法是在Dyna-Q算法的基础上,考虑各个状态转移之间距离上次访问的时间,将奖励设置为 r + k τ r+k\sqrt{\tau} r+kτ,这里 r r r是即时奖励, k k k是一个很小的值, τ \tau τ表示已经有多少个时间步没有进行这个状态转移了。这样一个改变,就会鼓励agent去探索,相应的也能更好地应对环境的变化。
在上图中,1-1000个时间步,Dyna-Q和Dyna-Q+的累积奖励都在不断增加,且Dyna-Q+增加速度大于Dyna-Q。第1000个时间步时,环境发生变化,两个算法开始时都没有及时检测到变化,而是在尝试的过程中发现变化并调整。所以中间的一段累积奖励没有变化。调整后,各状态的价值函数发生了变化,从而适应了新的环境,累积奖励又不断增加。
初始环境中从起点S到终点G在左边有一条通路,在时间步3000时,环境发生变化,在S和G右边也出现了一条通路,这条路使得S和G之间路径更短。
上图给出了Dyna-Q和Dyna-Q+在环境变化前后得到的累积奖励。可以看出Dyna-Q在环境改变前后,并没有发生太大变化,Dyna-Q+累积奖励增加。
Dyna-Q中,模型已经探索到了右边路径是可以到达终点的, 那么就认为右边的相关状态价值更高,所以在规划的过程中,模型能给出的转移一直是右边通路的,即使在学习过程中使用了 ϵ \epsilon ϵ-greddy,也无法扭转局面。
但在Dyna-Q+中,agent会去尝试长时间没有访问到的动作,所以可以较快的检测到环境的变化,找到了一条更近的路径,累积奖励增加。
前面涉及到规划过程总是在之前经历过的state-action pair中随机选择一个进行状态转移。但很多时候,并不是所有的state-action pair都要参与这一次的更新,可能更新之后值还是没有变化。比如方格游戏中,初始时,所有状态价值都为0,下一次更新时,只有终止状态的前一个状态的价值会得到更新,其他状态的价值仍然为0。所以这样的情况如果随机选择state-action pair就会导致计算浪费。
所以,每次更新的时候,应该对state-action pair有所侧重,怎样去选择影响最大的状态去更新呢?在方格游戏中,我们可以从终点状态倒推,去更新其他状态的价值。但有些情况下不存在终点状态,那么这种方法就不适用了。
所以,我们需要找到一种泛化性比较强的方法去计算所有的state-action pair的优先级,先更新优先级高的state-action pair。
要做到这件事情,我们需要做两件事情:一是要找到一种优先级计算的方法;二是要维护一个优先级队列,队列中的state-action pair根据其优先级排序。
优先级计算方法:根据之前的学习,可以知道如果一个状态的价值变化很大,那么它前面的状态价值也会有很大的变化。所以我们可以预测每个state-action pair的价值变化,根据价值变化大小插入到优先级队列中。
优先级队列:每次会选择队首的state-action pair进行更新,更新后调整队列。
上面的算法中,(e)步计算了状态S执行A动作的价值价值变化,如果变化大于一个阈值,就根据该变化值插入到优先级队列PQueue中。(g)步是进行规划的过程,取出队首的state-action pair,根据模型给出对应的状态转移,更新动作价值。然后对于所有可能到达状态S的state-action pair,计算其动作价值变化,如果超过阈值,也插入到优先级队列。
例子
下图展示了Dyna-Q算法和优先级排序算法在第2节的方格游戏中的表现。横轴是方格的大小(也是状态数),纵轴是在收敛到最优之前,两个算法对应的更新次数。
可以看出优先级遍历始终要比Dyna-Q的更新次数少。
值函数的更新在强化学习的任务中是非常重要的,更新放方法也不相同。对于一步更新,总结来看,共有三个维度上的不同:
根据以上的三个维度,可以得到8种不同类别的更新方式,对应于下面给出的7种特定的算法(因为采样更新的方式没有办法直接计算最优策略的价值)。
期望更新相比采样更新会得到一个更好的估计,因为它不受采样的影响。但它往往需要更多的计算资源。所以需要权衡场景需求再选择这两种方式。
例子: Q ∗ Q^* Q∗的期望更新和采样更新
期望更新:
Q ( s , a ) ← ∑ s ′ P s s ′ a [ R s s ′ a + γ max a ′ Q ( s ′ , a ′ ) ] Q(s, a) \leftarrow \sum_{s'}P_{ss'}^a[R_{ss'}^a + \gamma \max_{a'}Q(s', a')] Q(s,a)←s′∑Pss′a[Rss′a+γa′maxQ(s′,a′)]抽样更新:
Q ( s , a ) ← Q ( s , a ) + α [ R s s ′ a + γ max a ′ Q ( s ′ , a ′ ) − Q ( s , a ) ] Q(s, a) \leftarrow Q(s, a) + \alpha[R_{ss'}^a + \gamma \max_{a'}Q(s', a')-Q(s, a)] Q(s,a)←Q(s,a)+α[Rss′a+γa′maxQ(s′,a′)−Q(s,a)]
可以看出,期望更新需要知道分布模型的转移概率。在环境随机的情况下,期望更新和采样更新差别较大。对于一个状态动作对(s, a),如果其后有b个可能的状态,那么期望更新的计算量大概是抽样更新的b倍。
书中也给出了期望更新和采样更新的实验结果:
初始误差为1,对于期望更新来说,因为后续有b中可能的状态,在b次备份之后,误差减少为0。抽样备份则是根据 1 t b − 1 b \frac{1}{\sqrt{t}}\frac{b-1}{b} t1bb−1减小误差,并且可以发现,在b较大的时候,采样更新的误差急剧下降。
轨迹采样是根据一定的分布在状态空间或者状态-行为空间中采样。比如:根据给定的一个策略,我们就可以得到一系列的轨迹。这就叫做轨迹采样(trajectory sampling)。
还有一种采样方式叫做均匀采样(uniform sampling),是从整个状态-动作空间中随机采样,并进行更新操作。
对于以上的两种采样方式,下图展示了其对比的实验结果。
第一张图展示了状态空间中包含1000个状态,抽样200次,分支数分别为1,3,10时的结果。
第二张图展示了分支数为1,具有10000个状态的结果。
从第一张图中可以看出:
- 轨迹采样收敛更快,在计算初期就可以得到一个较好的值。
- 轨迹采样后期的表现可能不如均匀采样,并且b越大这种趋势越明显。
从第二张图可以看出:
- 状态数增加,轨迹采样相对较好。
实时动态规划(RTDP)是动态规划值迭代算法的on-policy轨迹采样版本。RTDP的状态价值更新是根据DP中的值迭代方法来做的,公式如下。但不同于DP的一点是更新轨迹中出现的状态的价值。
v k + 1 ≐ max a E [ R t + 1 + γ v k ( S t + 1 ) ∣ S t = s , A t = a ] = max a ∑ s ′ , r p ( s ′ , r ∣ s , a ) [ r + γ v k ( s ′ ) ] v_{k+1} \doteq \max_{a} \mathbb{E}[R_{t+1} + \gamma v_k(S_{t+1})|S_t = s, A_t = a]\\ = \max_{a} \sum_{s', r}p(s', r| s, a)[r+\gamma v_k(s')] vk+1≐amaxE[Rt+1+γvk(St+1)∣St=s,At=a]=amaxs′,r∑p(s′,r∣s,a)[r+γvk(s′)]
在第四章 动态规划中,讲到了异步动态规划的方法,其中一个是in-place DP,这里的RTDP也是一种异步DP。异步DP是指状态价值的更新顺序并非一个接一个的,而是不固定的。在RTDP中价值更新的顺序是根据其出现在轨迹(真实轨迹或仿真轨迹都可)中的顺序来的。
可以看出,RTDP的特点包括:
RTDP的优点:
规划的方式可以有两种,一种是像之前的Dyna和动态规划类似的方法,利用模型产生的仿真经验计算价值函数。是对所有状态(或状态-动作对)的价值都进行计算,并且计算结果保存在表格中。这种方式关注全局的状态,叫做background planning。
另一种规划是对于特定的状态(或状态-动作对)而言的。对于当前状态 S t S_t St,会计算出agent当前要执行的动作 A t A_t At。但进行到下一个状态时,又重新开始规划。这种方式仅关注当前状态,称为决策时规划(decision-time planning)。
因为决策时规划是关注于当前状态的,所以从当前状态计算出来的价值和策略通常在选择完动作之后就被丢弃。这对于一些任务来说没有很大损失,并且这种方法非常适用于不需要快速响应的情况下。比如:国际象棋游戏程序中,每次移动可以允许几秒甚至几分钟的计算。
决策时规划统称为启发式搜索。在启发式搜索里,对于每一个面临的状态,考虑一个可能的延续树。跟上一章中提到的数备份算法类似,区别在于不保存树中的值,当前状态做完决策之后,就可以将这棵树丢弃。
启发式搜索在一些对响应时间要求不太高的任务重非常高效,比如:围棋比赛等。因为搜索树密切关注那些当前状态之后可能出现的状态和行为,内存和计算资源仅为当前时刻的状态服务,可以大幅提高效率。
下图展示了利用启发式规则构建的一棵搜索树,我们可以自底向上各自执行单步备份。最终得到根节点状态的决策。
Rollout算法是依据蒙特卡洛控制的决策时规划算法。具体的,对于当前状态,根据蒙特卡洛采样从当前状态开始的一些轨迹序列。要计算当前状态的价值,只需要将所有序列中得到的奖励求和取平均。如果想要得到动作价值,那就计算每个状态-动作对得到的平均奖励。
得到动作价值之后,就可以选择当前状态下,具有最高估计值的动作,执行这个动作,也就是和环境交互,转移到下一个状态,再对下一个状态进行同样的采样,以规划下一步的动作。
这种方法的本质是在提升当前的策略,而不是找到一个最优策略。因为它只对当前状态进行提升,其他状态的策略不变。在实际中这种算法十分有效。
rollout算法的效率受到以下几方面的影响:
当然,我们也可以通过并行仿真轨迹,以及使用截断的轨迹来提升算法效率。
蒙特卡洛树(MCTS)是一个决策时规划算法,与rollout思想类似,对于当前的状态,它会采样一些轨迹序列,来帮助决策。区别在于MCTS在采样轨迹的过程中构建搜索树,并且搜索树可以引导采样的方向,使得采样出来的轨迹对于决策更有效果。
所以,MCTS非常适用于那些状态、动作空间较大的情况,例如:围棋游戏。因为它可以做到有效的剪枝。
具体的构建搜索树的过程包括四步:
下面给出一个下棋的例子
首先假设已经根据一些轨迹经验得到了部分的搜索树,在此基础上,对搜索树进行进一步扩展。
每个节点表示为A/B,其中A表示黑棋获胜次数,B表示访问次数。
【下图是从我之前做过的ppt中截取的】
注意:这里我们是站在黑棋的立场,所以每一步是黑棋或者白棋时,UCB的计算方法也不相同。