断路器Hystrix与Turbine集群监控-Spring Cloud学习第三天(非原创)

文章大纲

一、Hystrix基础介绍
二、断路器Hystrix简单使用
三、自定义Hystrix请求命令
四、Hystrix的服务降级与异常处理
五、Hystrix的请求缓存与请求合并
六、Hystrix仪表盘与Turbine集群监控
七、项目源码与参考资料下载
八、参考文章

一、Hystrix基础介绍

1. Hystrix简介

??一个用户管理项目,里边就三个功能:用户注册、用户登录、用户详情浏览。按照传统的软件开发方式直接创建一个Web项目,分分钟就把这三个功能开发出来了,但是我现在想使用微服务+服务治理的方式来开发:首先我将这个项目拆分为四个微服务,四个微服务各建一个模块,分别是用户注册模块、用户登录???、用户详情浏览??楹褪菘獠僮髂??,这四个??橥ü诓糠裰卫砘ハ嗟饔谩5窍衷诖嬖谝桓鑫侍?,这四个模块通过服务注册与订阅的方式互相依赖,如果一个??槌鱿止收匣岬贾乱览邓哪?橐卜⑸收洗佣⑸收下?,进而导致整个服务的瘫痪。比如说这里的登录??橐览涤谑菘饽??,如果数据库模块发生故障,那么当登录??槿サ饔檬菘饽?榈氖焙蚩赡艿貌坏较煊?,这个调用的线程被挂起,如果处于高并发的环境下,就会导致登录??橐脖览?。当一个系统划分的??樵蕉啵庵止收戏⑸钠德示突嵩礁?,对于这个问题,Spring Cloud中最重要的解决方案就是断路。

2. 何时需要?;?/h3>

??对于一个系统而言,它往往承担着2层角色,服务提供者与服务消费者。对于服务消费者而言最大的痛苦就是如何“明哲保身”,做过网关项目的同学肯定感同身受

??上面是一个常见的系统依赖关系,底层的依赖往往很多,通信协议包括 socket、HTTP、Dubbo、WebService等等。当通信层发生网络抖动以及所依赖的系统发生业务响应异常时,我们业务本身所提供的服务能力也直接会受到影响。
??这种效果传递下去就很有可能造成雪崩效应,即整个业务联调发生异常,比如业务整体超时,或者订单数据不一致。
??那么核心问题就来了,如何检测业务处于异常状态?
??成功率!成功率直接反映了业务的数据流转状态,是最直接的业务表现。
??当然,也可以根据超时时间做判断,比如 Sentinel 的实现。其实这里概念上可以做一个转化,用时间做超时控制,超时=失败,这依然是一个成功率的概念。

3. 如何?;?/h3>

??如同豪猪一样,“刺”就是他的?;すぞ撸械墓セ鞫蓟岜淮涛耷榈捻』厝?。在 Hystrix 的实现中,这就出现了“熔断器”的概念,即当前的系统是否处于需要?;さ淖刺?。当熔断器处于开启的状态时,所有的请求都不会真正的走之前的业务逻辑,而是直接返回一个约定的信息,即 FallBack。通过这种快速失败原则?;の颐堑南低?。 但是,系统不应该永远处于“有刺”的状态,当危险过后需要恢复正常。于是对熔断器的核心操作就是如下几个功能:如果成功率过低,就打开熔断器,阻止正常业务,随着时间的流动,熔断器处于半打开状态,尝试性放入一笔请求,熔断器的核心 API 如下图:

4. 限流、熔断、隔离、降级

这四个概念是我们谈起微服务会经常谈到的概念,这里我们讨论的是 Hystrix 的实现方式。
4.1 限流
??这里的限流与 Guava 的 RateLimiter 的限流差异比较大,一个是为了“保护自我”,一个是“?;は掠巍?br> ??当对服务进行限流时,超过的流量将直接 Fallback,即熔断。而 RateLimiter 关心的其实是“流量整形”,将不规整流量在一定速度内规整。

4.2 熔断
??当我的应用无法提供服务时,我要对上游请求熔断,避免上游把我压垮,当我的下游依赖成功率过低时,我要对下游请求熔断,避免下游把我拖垮。

4.3 降级
??降级与熔断紧密相关,熔断后业务如何表现,约定一个快速失败的 Fallback,即为服务降级。

4.3 隔离
业务之间不可互相影响,不同业务需要有独立的运行空间,最彻底的,可以采用物理隔离,不同的机器部,次之,采用进程隔离,一个机器多个 Tomcat,次之,请求隔离,由于 Hystrix 框架所属的层级为代码层,所以实现的是请求隔离,线程池或信号量。

二、断路器Hystrix简单使用

1. 开始前准备

在之前的文章中我们已经成功的搭建出服务注册中心、服务提供者和服务消费者三个微服务,本文的案例我们依然在这三个案例的基础上来实现。
首先我们分别启动服务注册中心,再启动两个服务提供者的实例,端口号分别是8080和8081,然后再启动一个服务消费者,服务消费者的端口号为9000,这几个都启动成功之后,我们访问http://localhost:9000/ribbon-consumer这个地址,可以看到如下效果:

此时我们关闭掉任意一个服务提供者,再去访问这个地址,会看到如下效果:

2. Hystrix引入

下列的所有操作都是在ribbon-consumer项目中进行

2.1 服务消费者中加入断路器

<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-hystrix</artifactId>
</dependency>

2.2 修改服务消费者启动入口类
引入hystrix之后,我们需要在入口类上通过@EnableCircuitBreaker开启断路器功能,如下:

@EnableCircuitBreaker
@SpringBootApplication
@EnableDiscoveryClient
public class RibbonConsumerApplication {

    public static void main(String[] args) {
        SpringApplication.run(RibbonConsumerApplication.class, args);
    }
    @LoadBalanced
    @Bean
    RestTemplate restTemplate() {
        return new RestTemplate();
    }
}

我们也可以使用一个名为@SpringBootApplication的注解代替这三个注解,@SpringBootApplication注解的定义如下:

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@SpringBootApplication
@EnableDiscoveryClient
@EnableCircuitBreaker
public @interface SpringCloudApplication {
}

实际上就是这三个注解的一个整合

2.3 修改Controller
我们创建一个HelloService类,如下:

@Service
public class HelloService {
    @Autowired
    private RestTemplate restTemplate;

    @HystrixCommand(fallbackMethod = "error")
    public String hello() {
        ResponseEntity<String> responseEntity = restTemplate.getForEntity("http://HELLO-SERVICE/hello", String.class);
        return responseEntity.getBody();
    }

    public String error() {
        return "error";
    }
}

关于这个HelloService类我说如下几点:
(1)RestTemplate执行网络请求的操作我们放在HelloService中来完成。
(2)error方法是一个请求失败时回调的方法。
(3)在hello方法上通过@HystrixCommand注解来指定请求失败时回调的方法。

最后我们将ConsumerController的逻辑修改成下面这样:

@RestController
public class ConsumerController {
    @Autowired
    private HelloService helloService;
    @RequestMapping(value = "/ribbon-consumer",method = RequestMethod.GET)
    public String helloController() {
        return helloService.hello();
    }
}

此时我们就开启了断路器功能

2.4 启动项目并测试断路器
启动项目后,我们的访问是正常的

现在我们将其中一个服务提供者干掉

重新访问http://localhost:9000/ribbon-consumer

OK,小伙伴们看到,此时如果服务调用失败,就会调用失败的那个回调方法。事实上,不仅仅是服务提供者被关闭时我们需要断路器,如果请求超时也会触发熔断请求,调用回调方法返回数据。

三、自定义Hystrix请求命令

1. 简介

在上面内容中,我们介绍了断路器Hystrix的一个简单使用,主要是通过注解来实现断路器的功能的,不过对于Hystrix的使用,除了注解,我们也可以使用继承类的方式来实现,本文我们就来看看另一种Hystrix的使用方式。

2. 自定义断路器Hystrix

我们除了使用@HystrixCommand注解,也可以自定义类继承自HystrixCommand,创建BookCommand.java

package test.custom;

import com.netflix.hystrix.HystrixCommand;
import org.springframework.web.client.RestTemplate;

import java.awt.print.Book;

/**
 * 自定义断路器Hystrix
 *
 * 在BookCommand中注入RestTemplate,然后重写两个方法:一个是getFallback,这个方法将在服务调用失败时回调;
 * 另一个是run方法,执行请求时调用。构造方法的第一个参数主要用来保存一些分组信息。
 */
public class BookCommand extends HystrixCommand<String> {

    private RestTemplate restTemplate;

    @Override
    protected String getFallback() {
        return  "测试出问题了";
    }

    public BookCommand(Setter setter, RestTemplate restTemplate) {
        super(setter);
        this.restTemplate = restTemplate;
    }

    @Override
    protected String run() throws Exception {
        return restTemplate.getForEntity("http://HELLO-SERVICE/hello", String.class).getBody();
    }
}

在BookCommand中注入RestTemplate,然后重写两个方法:一个是getFallback,这个方法将在服务调用失败时回调;另一个是run方法,执行请求时调用。构造方法的第一个参数主要用来保存一些分组信息。

3. 同步调用和异步调用

当BookCommand创建成功之后,我们就可以在我们的Controller中调用它了,创建BookCommandController.java

package test.controller;

import com.netflix.hystrix.HystrixCommand;
import com.netflix.hystrix.HystrixCommandGroupKey;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.client.RestTemplate;
import test.custom.BookCommand;

import java.awt.print.Book;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;

/**
 * 测试自定义断路器Hystrix
 */
@RestController
public class BookCommandController {

    @Autowired
    RestTemplate restTemplate;

    @RequestMapping(value = "/test1",method = RequestMethod.GET)
    public String test1() throws ExecutionException, InterruptedException {

//        1.获取到BookCommand对象之后,我们有两种方式来执行请求,一种是调用execute方法发起一个同步请求,另一种是调用queue方法发起一个异步请求。
//        2.同步请求中可以直接返回请求结果。
//        3.异步请求中我们需要通过get方法来获取请求结果,在调用get方法的时候也可以传入超时时长。
        BookCommand bookCommand = new BookCommand(HystrixCommand.Setter.withGroupKey(HystrixCommandGroupKey.Factory.asKey("")), restTemplate);
        //同步调用
        //Book book1 = bookCommand.execute();
        //异步调用
        Future<String> queue = bookCommand.queue();
        String string = queue.get();
        return string;
    }
}

温馨提示:
(1)获取到BookCommand对象之后,我们有两种方式来执行请求,一种是调用execute方法发起一个同步请求,另一种是调用queue方法发起一个异步请求。
(2)同步请求中可以直接返回请求结果。
(3)异步请求中我们需要通过get方法来获取请求结果,在调用get方法的时候也可以传入超时时长。

4. 项目运行与访问

保持原本项目启动情况,即关闭一个服务提供者,之后重新运行ribbon-cunsumer,再访问http://localhost:9000/test1,出现以下结果

四、Hystrix的服务降级与异常处理

1. 服务降级

fallbackMethod所描述的函数实际上就是一个备胎,用来实现服务的降级处理,在注解中我们可以通过fallbackMethod属性来指定降级处理的方法名称,在自定义Hystrix请求命令时我们可以通过重写getFallback函数来处理服务降级之后的逻辑。使用注解来定义服务降级逻辑时,服务降级函数和@HystrixCommand注解要处于同一个类中,同时,服务降级函数在执行过程中也有可能发生异常,所以也可以给服务降级函数添加‘备胎’

2. 异常处理

我们在调用服务提供者时有可能会抛异常,默认情况下方法抛了异常会自动进行服务降级,交给服务降级中的方法去处理,在自定义Hystrix请求命令的方式下,我们可以在getFallback方法中调用getExecutionException方法来获取抛出的异常,举个简单的例子:

package test.controller;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.client.RestTemplate;
import test.service.HelloService;
import test.service.HelloServiceImpl;

@RestController
public class ConsumerController {

    @Autowired
    RestTemplate restTemplate;

//    @RequestMapping(value = "/ribbon-consumer",method = RequestMethod.GET)
//    public String helloController() {
//        return restTemplate.getForEntity("http://HELLO-SERVICE/hello", String.class).getBody();
//    }

        @Autowired
        private HelloService helloService;

        //测试短路访问
        @RequestMapping(value = "/ribbon-consumer",method = RequestMethod.GET)
        public String helloController() {
            return helloService.hello();
        }
}

重新启动ribbon-cunsumer项目,此时访问http://localhost:9000/test1,控制台显示结果如下:

五、Hystrix的请求缓存与请求合并

1. 请求缓存

??高并发环境下如果能处理好缓存就可以有效的减小服务器的压力,Java中有许多非常好用的缓存工具,比如Redis、EHCache等,当然在Spring Cloud的Hystrix中也提供了请求缓存的功能,我们可以通过一个注解或者一个方法来开启缓存,进而减轻高并发环境下系统的压力。OK,本文我们就来看看Hystrix中请求缓存的使用。
??请求缓存具体内容可参考https://mp.weixin.qq.com/s/YpWODLrwzFXUQRtIAHLF3Q?

2. 请求合并

??我们将一个项目拆分成很多个独立的???,这些独立的??橥ü冻痰饔美椿ハ嗯浜瞎ぷ鳎?,在高并发情况下,通信次数的增加会导致总的通信时间增加,同时,线程池的资源也是有限的,高并发环境会导致有大量的线程处于等待状态,进而导致响应延迟,为了解决这些问题,我们需要来了解Hystrix的请求合并。
??请求合并具体内容可参考https://mp.weixin.qq.com/s/0QSKVLaDjBAscRaeccaXuA?

六、Hystrix仪表盘与Turbine集群监控

1. 简介

Hystrix仪表盘,就像汽车的仪表盘实时显示汽车的各项数据一样,Hystrix仪表盘主要用来监控Hystrix的实时运行状态,通过它我们可以看到Hystrix的各项指标信息,从而快速发现系统中存在的问题进而解决它,OK,本文我们就来看看Hystrix仪表盘要怎么使用。

2. 创建最基本的Hystrix项目

2.1 新建springboot项目,项目名称为hystrix-dashboard

创建后项目结构如下:

2.2 pom.xml文件添加依赖

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.wxc</groupId>
    <artifactId>hystrix-dashboard</artifactId>
    <version>1.0-SNAPSHOT</version>

    <parent>
        <groupId>org.springframework.cloud</groupId>
        <artifactId>spring-cloud-starter-parent</artifactId>
        <version>Dalston.SR3</version>
        <relativePath/>
    </parent>

    <dependencies>
        <!-- 其他默认依赖 -->
        <!-- 我们需要添加的依赖 -->
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-hystrix</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-hystrix-dashboard</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-actuator</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-hystrix</artifactId>
        </dependency>
        <dependency>

            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-actuator</artifactId>
        </dependency>

    </dependencies>


</project>

2.3 创建项目启动类
com.wxc.test包下新建HystrixDashboardApplication.java

package com.wxc.test;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.hystrix.dashboard.EnableHystrixDashboard;

@SpringBootApplication
@EnableHystrixDashboard
public class HystrixDashboardApplication {

    public static void main(String[] args) {
        SpringApplication.run(HystrixDashboardApplication.class, args);
    }
}

2.4 项目启动与访问

浏览器输入http://localhost:2001/进行访问,访问成功后结果如下:

三个参数的含义我已在图中标注出来了。
OK,现在我们的仪表盘工程已经创建成功了,但是还不能用来监控某一个服务,要监控某一个服务,需要该服务提供一个/hystrix.stream接口,so,我们需要对我们的服务消费者工程稍加改造。

3.改造要监控的服务

3.1 改造服务消费者工程ribbon-consumer
在pom.xml文件中添加以下依赖

<dependency>
   <groupId>org.springframework.cloud</groupId>
   <artifactId>spring-cloud-starter-hystrix</artifactId>
</dependency>
<dependency>
   <groupId>org.springframework.boot</groupId>
   <artifactId>spring-boot-starter-actuator</artifactId>
</dependency>

在确保服务注册中心和服务提供者开启情况下,在服务消费者工程的入口类上添加@EnableCircuitBreaker注解,表示开启断路器功能。此时项目启动情况如下:

这个信息表明我们的consumer工程目前已经具备了/hystrix.stream接口,我们可以直接访问这个接口了。但是这里有一个细节需要小伙伴们注意:要访问/hystrix.stream接口,得先访问consumer工程中的任意一个其他接口,否则如果直接访问/hystrix.stream接口的话,会打印出一连串的ping: ping: …。 OK,我先访问consumer中的任意一个其他接口,然后在访问/hystrix.stream接口,访问地址如下:http://localhost:9000/hystrix.stream,访问结果如下:

3.2 hystrix-dashboard服务监控
在hystrix仪表盘中输入监控地址,如下:

然后点击Monitor Stream按钮,我们就可以看到监控画面了,如下:

3.3 监控参数详解

4. Turbine集群监控

OK,上文我们看了一个监控单体应用的例子,在实际应用中,我们要监控的应用往往是一个集群,这个时候我们就得采取Turbine集群监控了。Turbine有一个重要的功能就是汇聚监控信息,并将汇聚到的监控信息提供给Hystrix Dashboard来集中展示和监控。那我们就来看看Turbine集群监控如何使用。

搭建监控环境
监控环境的搭建也是分为四个步骤:

第一步:创建一个普通的Spring Boot工程
第一步创建一个名叫turbine的普通Spring Boot工程。

第二步:添加依赖
工程创建完成之后,我们需要添加一个依赖,如下:

<parent>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-parent</artifactId>
    <version>Dalston.SR3</version>
    <relativePath/> 
</parent>
<dependencies>
    <!-- 其他默认的依赖 -->
    <!-- 我们要添加的依赖 -->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-actuator</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.cloud</groupId>
        <artifactId>spring-cloud-starter-turbine</artifactId>
    </dependency>
</dependencies>

第三步:添加注解
在入口类上添加@EnableTurbine注解表示开启Turbine,如下:

@SpringBootApplication
@EnableDiscoveryClient
@EnableTurbine
public class TurbineApplication {

    public static void main(String[] args) {
        SpringApplication.run(TurbineApplication.class, args);
    }
}

第四步:修改配置
在application.properties配置文件中加入eureka和turbine的相关配置,如下:

spring.application.name=turbine
server.port=2002
management.port=2003
eureka.client.service-url.defaultZone=http://localhost:1111/eureka/
turbine.app-config=ribbon-consumer
turbine.cluster-name-expression="default"
turbine.combine-host-port=true

关于这个配置文件,我说如下几点:

1.turbine.app-config=ribbon-consumer指定了要监控的应用名字为ribbon-consumer
2.turbine.cluster-name-expression=”default”,表示集群的名字为default
3.turbine.combine-host-port=true表示同一主机上的服务通过host和port的组合来进行区分,默认情况下是使用host来区分,这样会使本地调试有问题

查看监控图
OK,监控服务创建成功之后,我们再次依次启动eureka-server、provider和consumer,其中consumer启动两个实例,两个实例的端口不一致,再分别启动hystrix-dashboard和turbine,然后在hystrix监控地址栏输入如下地址(监控之前要记得先访问一下服务中的任意一个接口):http://localhost:2002/turbine.stream,访问结果如下:

七、项目源码与参考资料下载

链接:https://pan.baidu.com/s/120z7NZZ_-Z_BHySQHHazZw
提取码:flic

八、参考文章

  1. https://www.cnblogs.com/liuys635/p/10551253.html
  2. https://www.cnblogs.com/lenve/p/7985943.html
最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,100评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,308评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,718评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,275评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,376评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,454评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,464评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,248评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,686评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,974评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,150评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,817评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,484评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,140评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,374评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,012评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,041评论 2 351

推荐阅读更多精彩内容