【转载】论文阅读:《A Contextual-Bandit Approach to Personalized News Article Recommendation》

这篇文章属于推荐领域采用强化学习方法的文章。与其他推荐算法相比,基于强化学习的推荐算法更多地关注了explore/exploit问题,即探索/利用问题。也就是强调推荐算法不应该仅仅基于历史数据中用户的偏好进行推荐,而应该给用户更多的新鲜事物,引导和发掘用户的爱好。而强化学习恰恰是基于系统采取的动作所接收到的环境的反馈,来不断训练系统的。因此,在强化学习的过程中适度的选择一些探索性的动作,并观察反馈进一步调整后续的动过选择,天然地具有解决该问题的潜质。

参考:
https://zhuanlan.zhihu.com/p/34940176
https://zhuanlan.zhihu.com/p/129336798
https://www.yuque.com/el1s/paper-reading/vai69h
https://mp.weixin.qq.com/s/boUy1h_SuiMohLBu1Tatkg
https://zhuanlan.zhihu.com/p/162804829

论文中把文章推荐看作是多臂老虎机问题,一种有效的做法是根据用户和文章的上下文信息,选择文章推荐给用户,同时基于用户对文章的点击行为动态调整策略,追求点击次数最大化。实验证明加入上下文信息的Bandit算法比传统的Bandit算法CTR提升在12.5%以上,如果是基于稀疏的数据,效果会变得更好。

论文分析了已有的Bandit算法,包括UCB、E-Greedy、Thompson Smapling,然后提出了LinUCB算法,LinUCB分为两种:

  • 简单的线性不相交模型 disjoint LinUCB
  • 混合相交的线性模型 hybrid LinUCB

概述

人生中有很多选择问题,当每天中午吃饭的时候,需要选择吃饭的餐馆,那么就面临一个选择,是选择熟悉的好吃的餐馆呢,还是冒风险选择一个没有尝试过的餐馆呢。同样的,推荐系统处处也面临着这样的选择,是推荐一个已经熟悉的点击率很高的物品呢,还是选择一个新的物品呢。这些都可以泛化成一个经典问题,多臂老虎机问题,也是一个研究很广的问题。


DZTIKX87kwnUnXlY.png!thumbnail_fileGuid=RR3Vv6gjtQwp9kcW.png

下面介绍一些常用的bandit算法。

Epsilon-Greedy 算法

  • 选一个 (0,1) 之间较小的数作为 epsilon
  • 每次以概率 epsilon 做一件事:所有臂中随机选一个
  • 每次以概率 1-epsilon 选择截止到当前,平均收益最大的那个臂。

算法存在的缺点:

  • 算法中需要人工调的参数比较多,比如epsilon控制explore比例,还有对于一个物品探索到那个界限K才会让利用起来比较置信
  • 没有利用历史的信息,完全有可能探索我们已知的不好的行动

汤普森采样

  • 假设每个臂是否产生收益,其背后有一个概率分布,产生收益的概率为 p。
  • 我们不断地试验,去估计出一个置信度较高的 “概率 p 的概率分布” 就能近似解决这个问题了。
  • 怎么能估计 “概率 p 的概率分布” 呢? 答案是假设概率 p 的概率分布符合 beta(wins, lose)分布,它有两个参数: wins, lose。
  • 每个臂都维护一个 beta 分布的参数。每次试验后,选中一个臂,摇一下,有收益则该臂的 wins 增加 1,否则该臂的 lose 增加 1。
  • 每次选择臂的方式是:用每个臂现有的 beta 分布产生一个随机数 b,选择所有臂产生的随机数中最大的那个臂去摇。


    YTaQcWZ7tRyeCOeM.png!thumbnail_fileGuid=RR3Vv6gjtQwp9kcW.png
  • 注:
    • beta分布中的两个参数可以根据先验知识去设置,比如物料平均点击率3%, 那么我可以设置beta分布参数α=3,β=97,这样更贴近真实情况,可能更有效
    • 为什么可以这样做的理解:
      • 伯努利分布(二项分布):P\{X=k\}=\left(\begin{array}{c} n \\ k \end{array}\right) p^{k}(1-p)^{n-k}
      • Beta分布:f(x ; \alpha, \beta)=\frac{x^{\alpha-1}(1-x)^{\beta-1}}{\int_{0}^{1} u^{\alpha-1}(1-u)^{\beta-1} d u}=\frac{\Gamma(\alpha+\beta)}{\Gamma(\alpha) \Gamma(\beta)} x^{\alpha-1}(1-x)^{\beta-1}=\frac{1}{B(\alpha, \beta)} x^{\alpha-1}(1-x)^{\beta-1}
      • 贝叶斯更新:P(\theta \mid x)=\frac{P(x \mid \theta) P(\theta)}{P(x)},先验分布 * 最大似然 = 后验分布。
        • 对随机变量X,假设服从参数为theta1的分布XiP1(theta1),且theta1P2(theta2)。
        • 当对X抽样得到似然概率后,可以用似然和theta1的先验分布计算出theta1的后验分布
      • 如果P1和P2属于同一种分布,则称P2是P1的共轭先验分布
      • Beta分布是伯努利分布的共轭先验

UCB算法

  • 初始化:先对每一个臂都试一遍

  • 按照如下公式计算每个臂的分数,然后选择分数最大的臂作为选择:U C B_{j}=\bar{x}_{j}(t)+\sqrt{\frac{2 l n t}{T_{j, t}}}

  • 观察选择结果,更新t和Tjt。

加号前面是这个臂到目前的收益均值,后面的叫做 bonus,本质上是均值的标准差,t 是目前的试验次数,Tj,t是这个臂被试次数。

  • 注:

    • 公式前面一项是收益均值,可以理解为点击率

    • 核心是第二项,我们知道当x取值大于1时ln函数的作用是平滑,如果实验次数很多,那么后一项就会趋近很小,从而收敛到前一项收益均值,如果是一个新物料或者explore次数很少,那么后一项很大甚至大于1从而优先进行explore,支持探索到这个臂到达置信的次数。

    • UCB其实是一个算法框架,即每次选择动作的时候,采用乐观态度去衡量每个动作的预期收益。即预期收益有一个置信区间,以置信区间的上限作为动作的预期收益进行比较选择。所谓置信区间,其本质就是偏离预期收益(历史上的每次选择该动作的收益的均值)某个范围(例如,95%)的偏差。而这个偏差会随着选择该动作次数的增加,而逐渐收缩,即置信区间的上限会减小。也就是该动作被选择次数很多后,其预期收益会趋于稳定在一个范围(减小了噪音等因素引起的偏差)。可以被视为,该动作没有潜力了。如果该动作稳定在一个高收益,则仍会被选择,而如果该动作稳定在一个低收益,则又没有潜力,则不会再被选择。因此,UCB这种以置信区间上限作为选择动作依据的方式,自然而然地对探索的方向进行调整,使其偏向于高收益或有潜力的动作。

    • 公式推导:

      • 观测到行动a的奖励的均值Qt(a),假设奖励的真实值变化范围在[Qt(a)-delta, Q(t)+Ut(a)]之间,我们乐观的认为行动a的奖励是Qt(a)+Ut(a),每一次都按这个值来选择行动a,其中Ut(a)是关于Nt(a)的函数,Nt(a)越大,Ut(a)越小:N_{t}(a)=\sum_{i=1}^{t} 1\left[a_{i}=a\right]。Ut(a)总是采取最贪心的行动:a_{t}^{U C B}=\operatorname{argmax}_{a \in \mathcal{A}} \hat{Q}_{t}(a)+\hat{U}_{t}(a)

      • Hoeffding’s Inequality:不对数据分布进行任何假设,设X1, …, Xt为独立同分布的随机变量,他们的值在[0,1]之间,样本平均值为\overline{X_{t}}=\frac{1}{t} \sum_{i}^{t} \mathrm{X}_{i},对于\delta>0以下不等式总是成立:P\left\{\left|E[X]-\overline{X_{t}}\right| \leq \delta\right\} \geq 1-2 e^{-2 t \delta^{2}}

        • \delta的取值为\sqrt{\frac{2 \ln N_{t}(a)}{t}}时,2 e^{-2 t \delta^{2}}=\frac{2}{N_{t}(a)^{4}}
  • \overline{X_{t}}-\sqrt{\frac{2 \ln N_{t}(a)}{t}} \leq E[X] \leq \overline{X_{t}}+\sqrt{\frac{2 \ln N_{t}(a)}{t}}是以1-\frac{2}{N_{t}(a)^{4}}的概率成立的:

  • N_{t}(a)=2时,成立的概率为0.88
  • N_{t}(a)=3时,成立的概率为0.98
  • N_{t}(a)=4时,成立的概率为0.99
  • 根据Hoeffding’s Inequality:a_{t}^{U C B}=\arg \max _{a \in \mathcal{A}} \hat{Q}_{t}(a)+\sqrt{\frac{2 \log t}{N_{t}(a)}}

本文贡献

LinUCB with Disjoint Linear Model

本文正是在UCB框架下进行的研究工作,即在衡量每个动作预期收益时,引入了个性化因素——上下文环境,使每个动作的预期收益与上下文环境特征有相关性。相比于普通UCB,本文提出了一种LinUCB的方法。LinUCB改变的是预期收益(均值)这一部分。论文给出了下面的公式:

  • (1): R_{\mathrm{A}}(T) \quad \stackrel{\text { def }}{=} \quad \mathbf{E}\left[\sum_{t=1}^{T} r_{t, a_{t}^{*}}\right]-\mathbf{E}\left[\sum_{t=1}^{T} r_{t, a_{t}}\right]
  • (2): \mathbf{E}\left[r_{t, a} \mid \mathbf{x}_{t, a}\right]=\mathbf{x}_{t, a}^{\top} \boldsymbol{\theta}_{a}^{*}

通常UCB在预期收益部分采用的是(1)式左边的E值,通常是历史数据统计出来的收益均值。而本文LinUCB采用(2)式的线性回归模型对这部分进行了拟合,使得(1)式的差最小化。即(1)式左边的E值为Label,右边的E值为线性回归的预测值,RA(T)可看作是Loss。(2)式中的X为与用户和物品(arm)相关的特征值向量,里面可能包括了用户的年龄、性别以及新闻的推荐位置,类型等。而\theta_{a}^{*}则是待训练的参数。

使用岭回归作为loss,通过岭回归公式,可以给出theta的计算公式:\hat{\boldsymbol{\theta}}_{a}=\left(\mathbf{D}_{a}^{\top} \mathbf{D}_{a}+\mathbf{I}_{d}\right)^{-1} \mathbf{D}_{a}^{\top} \mathbf{c}_{a}其中Da为m*d的矩阵,包含m个训练数据x_{t, a}c_a为m维的向量,每一个值为m个训练数据对应的行动产生的相应奖励值。

这种回归拟合粗看似乎跟强化学习没有什么关系,因为已经知道了历史数据统计出来的预期收益,直接就能训练出参数\theta_{a}^{*},那么选择下一个动作(推荐哪个新闻),直接就比较预测值就可以了。然而,在采取动作以后,还是要收集反馈(用户点击情况),并纳入历史数据中,以更新预期收益的值,并用于下一次动作的选择。这是强化学习与普通的监督学习最大的区别,即根据环境的反馈进行学习,而没有固定的样本数据和对应的标签值。而(1)式中左边的E作为Label,这个Label其实就是环境的反馈。

在将线性回归与MAB结合后,没有解决探索/利用问题,即此时选择下一个动作完全是根据历史最优,选择了历史看来收益最好的动作。因此,论文采用了UCB的架构,利用下面的公式(另一篇论文Exploring compact reinforcement-learning representations with linear regression,这里将其作为一个定理使用):\left|\mathbf{x}_{t, a}^{\top} \hat{\boldsymbol{\theta}}_{a}-\mathbf{E}\left[r_{t, a} \mid \mathbf{x}_{t, a}\right]\right| \leq \alpha \sqrt{\mathbf{x}_{t, a}^{\top}\left(\mathbf{D}_{a}^{\top} \mathbf{D}_{a}+\mathbf{I}_{d}\right)^{-1} \mathbf{x}_{t, a}}构造了UCB的置信区间上限。在此基础上,给出了下面的动作选择依据公式:a_{t} \stackrel{\text { def }}{=} \arg \max _{a \in \mathcal{A}_{t}}\left(\mathbf{x}_{t, a}^{\top} \hat{\boldsymbol{\theta}}_{a}+\alpha \sqrt{\mathbf{x}_{t, a}^{\top} \mathbf{A}_{a}^{-1} \mathbf{x}_{t, a}}\right),其中\alpha=1+\sqrt{\ln (2 / \delta) / 2}\mathbf{A}_{a} \stackrel{\text { def }}{=} \mathbf{D}_{a}^{\top} \mathbf{D}_{a}+\mathbf{I}_{d}也即是,选择括号内部分得分最大的那个arm进行执行。括号内左边就是回归拟合的预期收益,右边是置信区间的上限。

在这个基本思想下,算法的伪代码可以写成:

3.png

LinUCB with Hybrid Linear Model

前面讲的是LinUCB的一种实现方式,对于每个臂维护theta参数。当我们希望解决共享类context的需求时,即一个用户可能有偏好,只喜欢体育,那么这个参数就应该被所有待推荐给他的文章池(arm池)所共享。即在t时刻为该用户推荐文章时,所有待选文章(arm)都要用到这个context。因此,作者引入了下面的公式:\mathbf{E}\left[r_{t, a} \mid \mathbf{x}_{t, a}\right]=\mathbf{z}_{t, a}^{\top} \boldsymbol{\beta}^{}+\mathbf{x}_{t, a}^{\top} \boldsymbol{\theta}_{a}^{}。其中z为共享类context(当前用户和文章的共享类特征的组合向量),而\beta为待学习的权重,可以看到它没有下标,即所有arm(所有待选文章)都只学习且采用这一个权重,从而实现共享类context的信息交互。与之相对的是权重\theta_{a},每个arm都会学到一个,各arm之间不会共享context的信息。

在这个思想下,算法的伪代码可以写成:

4.png

评估方法

由于是探测问题,如果不上线,就没办法探到算法希望的内容。有点像鸡生蛋、蛋生鸡的问题。也可以看作强化学习里的“off-policy evaluation problem”问题。一个方式是根据离线数据构造一个模拟器,但是这种模拟器会引入bias。论文提出了一个没有bias的简单实现:

  • 假设有若干未知的分布D服从独立同分布\left(x_{1}, \ldots, x_{K}, r_{1}, \ldots, r_{K}\right)。每个分布包含了所有arm的可观测特征向量、隐含的奖励

  • 假设要处理现实世界中,根据日志策略的交互结果生成的一系列事件:
    每个事件包含了上下文特征向量x_{1}, \ldots, x_{K},一个选择的arm a,以及观察到的奖励ra。
    简单起见,我们把实验的事件流看作是无限的,但是实验所需的事件是有上限的。
    目标是利用这份数据来评估一个bandit算法\pi。形式上\pi是从历史t-1个事件h_{t-1}以及当前的上下文特征x_{1}, \ldots, x_{K}到当前时刻t选择arm at的映射。

  • 论文提出的评估策略如下:
    5.png
  • 把策略\pi和一系列事件T作为输入。然后逐一考察日志事件。如果,给定当前历史h_{t-1},如果策略\pi和日志策略选择同一个arm,那么保留这个事件,也就是把它加到历史中,更新总体奖励Rt;否则,如果策略\pi和日志策略选择的arm不一样,则整个事件被忽略,也不更新状态。

  • 日志策略选择arm的策略是uniformly at random,所以每个事件被算法保留的概率是\frac{1}{K}并且各自是独立的。

  • 也就是说,被D选择并保留的事件有相同的分布。

实验细节

6.png

数据收集

  • 随机抽取用户
  • 曝光的文章是从内容池中随机选取的
  • 为了避免曝光位置bias,我们只考察F1位置的文章
  • 每个用户交互事件包含三部分内容
    • 曝光的文章
    • 用户、文章信息
    • 是否点击
  • 选了5月1日470万事件作为训练集,5月3日~9日的3600万个事件作为验证集

特征构造

  • 用“支持度”来选择特征,支持度指的是包含这个特征的用户占比。为了减少噪声,选择支持度大于0.1的特征。
  • 用户特征的向量超过1000维。包括(且只包括):
    • 人口属性学:性别(2种)、年龄(离散化为10段)
    • 位置信息:美国200个大城市
    • 行为类型: 1000种行为标签
  • 类似的,文章通常用100种类目特征向量来表示:
    • URL种类:10种类型
    • 编辑类型:人工标注的数十种文章类型
  • 用论文Personalized recommendation on dynamic content using predictive bilinear models中的编码方式,将特征向量统一长度。同时,我们会为每个特征向量增加一个常数特征1。这样,每个文章和user被表示成83维和1193维的向量
  • 为了降维和获取非线性特征,我们用08年9月份的数据进行联合分析,然后利用论文A case study of behavior-driven conjoint analysis on Yahoo!中的降维方法。
    • 把用户特征投射到文章分类上,然后根据用户兴趣聚类。
      • 首先训练一个CTR预估模型:\varphi_{u}^{\top} W \varphi_{a}算出的是用户u对文章a的点击率。\varphi_{u}, \varphi_{a}分别表示用户和文章的特征向量。W是LR学到的权重矩阵
      • 原始的用户特征可以用\psi_{u}=\varphi^{\top} W\psi_{u}中的第i维表示用户u对文章i的兴趣度。然后用K-means把用户在\psi_{u}空间中聚成5类。最终的用户特征是一个6维向量,前五个维是对应5个聚类的相应(用高斯核计算,然后归一化,相加为1)、第六维是常数1。
    • 每个文章a有个单独的6维向量x_{t, a},类似用户特征计算方法。因为文章特征是不重合的,他们是属于disjoint模型特征。
    • 对于每个文章a,我们把它的6维向量和用户的6维向量做外积,得到6*6=36维特征,表示成z_{t, a} \in R^{36},代表用户和文章的交叉信息,属于hybrid模型特征。

算法比较

本文对比了如下算法的性能差异:

  • random:从内容池中随机选取一个,每个内容选中的概率相同。
  • Epsilon-Greedy
  • UCB
  • disjoint LinUCB
  • Hybrid LinUCB

学习桶和部署桶

在线上执行时,虽然线性拟合和统计等步骤计算量不大,但是用户和待选文章的数量增大后,仍然是比较大的代价。因此,论文给出了学习桶和部署桶的概念,即利用学习桶更新参数,而在部署桶中利用学习桶得到的参数进行推荐(动作的选择)。

实验结果

处于保密原因,使用相对的CTR,也就是算法的CTR除以随机策略的。假设随机策略的CTR等于1。论文中直接用CTR代替“相对CTR”。

由于部署集的数据量大于训练集,所以,部署集的CTR更重要一些。但是学习集中的高CTR意味着更快的学习率(或者等价的,更小的regret)。所以论文把训练集和部署集中的CTR都列出来了。

训练集数据

7.png
  • 每个对比的算法(除random和无所不知算法外)都有一个参数:如Epsilon-Greedy中的\epsilon和UCB算法中的\alpha
  • 学习集中的CTR都是呈倒U型。当参数较小时,探索不充分,算法不能发现CTR较好的文章,导致点击量较低。同理,当参数很大时,探索过度,导致浪费了流量在CTR较低的文章上。论文为每种算法选择最合适的参数,在下一节的测试数据上跑一遍。
  • 相比于没有上下文特征的Epsilon-Greedy算法,UCB算法,热启动算法Epsilon-Greedy(warm)和UCB(warm)在最好的参数时,可以击败无所不知算法。但是热启动算法的稳定性不如在线算法,因为热启动算法都是离线用的完全随机数据训练的。Epsilon-Greedy算法在\epsilon接近1的时候性能比较稳定。热启动可以帮助ucb(warm)为用户选择更多吸引力的文章,但是不能保证线上的效果。因为ucb(warm)依赖于探测的置信区间,但是不太容易纠正热启动的初始偏置。因此,论文没有在验证集上跑热启动算法。
  • 在部署集上,Epsilon-Greedy和UCB算法的效果差不多,但是在训练集上的效果差一些,跟上下文无关的算法表现一致。
  • 为了验证数据稀疏性对模型的影响,论文减少了训练集的样本数量,30%、20%、10%、5%、1%等,部署集样本量不变。

验证集数据

8.png

特征对结果的影响

从上表中明显看出,无论是 Epsilon-Greedy (seg/disjoint/hybrid)还是UCB(ucb(seg)、linucb(disjoint/hybrid))都能使CTR提升10%左右。


9.png

为了更好的可视化特征的效果,上图描述的是对比base CTR是如何提升的。base CTR是描述的文章是否被随机用户感兴趣。提升的比例越高,说明算法把文章推给了潜在感兴趣的用户。

  • (a)说明没有用上下文特征的算法也能提升CTR。其他三张图更明显的表明了个性化的效果。极端的例子,3(c)中最高的那个点代表的文章,CTR从1.31到3.03,提升了132%

样本大小对结果的影响

论文减少了训练集的样本数量,30%、20%、10%、5%、1%等,来模拟一个较大的内容池,但是流量固定的情况。


10.png

上图中描述了不同数据量对CTR的影响:

  • 在所有数量级中,特征仍然是有作用的。
  • 在部署集上,ucb比Epsilon-Greedy算法的效果普遍要好,在1%流量时更明显。
  • 相比与 ucb (seg) 和 linucb (disjoint),linucb(hybrid)在数据量小时优势更明显。原因是混合模型的特征是共享的,CTR的信息可以从一个文章传递到另一个文章,当内容池越大的时候,这个优势越明显。相反,在独立模型中,没法传递这种信息。
  • 图4(a)描述了迁移学习在稀疏数据中的作用。对比ucb(seg)和linucb(disjoint):从图4(a)中看出,这两个算法的效果差不多。我们认为这不是巧合。是用户聚类的作用。
  • 图5描述的是用户最近临(非常数的5个维度)的聚类的直方图。可以看出,大部分用户非常接近某一个聚类,比如85%的用户大于0.5,40%的用户大于0.8。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,417评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,921评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,850评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,945评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,069评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,188评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,239评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,994评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,409评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,735评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,898评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,578评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,205评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,916评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,156评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,722评论 2 363
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,781评论 2 351

推荐阅读更多精彩内容