7个设置/获取接口了解Linux时间管理

7个设置/获取接口了解Linux时间管理

[TOC]

引言

  最近的项目开发中,频繁遇到了时间戳相关的问题,如时间回退至1970年、时区错误及时间同步不准确等。鉴于此前仅对时间接口的使用有所了解而未深入探究其原理,本篇文章进行一次系统性整理,以便后续参考。文章若存在一些错误,可在留言区明确指出。

<span style="font-size: 12px;">
<span style="color: blue;"> 注:文末提供本文源码获取方式。文章不定时更新,喜欢本公众号系列文章,可以星标公众号,避免遗漏干货文章。源码开源,如果对您有帮助,帮忙分享、点赞加收藏喔!</span>
</span>

基础概念

Linux 中的时间形式主要以两种形式呈现:

  • 相对时间 指相对于某个基准点来衡量时间流逝。通常用于描述进程运行的时间或两个事件之间的时间差。
    • 进程时间
      即进程消耗的时间,包含用户空间代码运行的时间和在内核在该进程消耗的时间(不包括进程被挂起或停止的时间)。
    • 单调时间
      是一种始终递增的时间计数器,不受系统时钟调整的影响,常用于计算程序内部的持续时间。
  • 绝对时间 指具体的日期和时刻,它与地球上的特定时间标准相关联。
    • GMT(Greenwich Mean Time 格林威治时间)
      基于英国伦敦附近的格林尼治天文台的本初子午线的标准时间
    • UTC(Universal Time Coordinated 世界标准时间)
      一种国际标准时间,与GMT几乎相同,但更精确,用于避免地球自转速度变化带来的影响
    • 本地时间
      根据用户所在地理位置所采用的时间,会随地理位置的不同而有所差异,同时也会受到夏令时等因素的影响

相关结构体

  时间编程中常用要用到的时间结构体有time_ttimevaltimespectm。《Unix环境高级编程》中一张图准确的反应出time_ttm之间的关系:

时间函数之间的关系.png
  • time_t:最简单的数据湖结构,表示从1970年1月1日00:00:00 UTC到现在的秒数。
  • tm:包含日期和时间的具体组成部分(年、月、日、时、分、秒等),通常由 time_t 转换而来,用于显示或解析时间。
  • timeval:微秒级精度,包含秒(tv_sec)和微秒(tv_usec)。
  • timespec:纳秒级精度,包含秒(tv_sec)和纳秒(tv_nsec)。
  • clock_t:表示程序执行过程中消耗的CPU时间,单位是 CLOCKS_PER_SEC

相关函数

时间获取

  • time:
    • 函数原型:time_t time(time_t *tloc);
    • 功能描述:该函数返回从1970年1月1日00:00:00 UTC以来的秒数。如果tloc不是NULL,则返回的时间值也会存储在tloc指向的位置。
    • 返回值:成功时返回当前时间(以秒为单位),失败时返回(time_t)(-1)
  • gettimeofday:
    • 函数原型:int gettimeofday(struct timeval *tv, struct timezone *tz);
    • 功能描述:这个函数提供了比time()更高的精度,可以获取当前时间精确到微秒。struct timeval包含两个成员:tv_sec(秒数)和tv_usec(微秒数)。struct timezone已经废弃,通常传入NULL
    • 返回值:成功时返回0,出错时返回-1,并设置errno
  • clock_gettime:
    • 函数原型:int clock_gettime(clockid_t clk_id, struct timespec *tp);
    • 功能描述:此函数提供了更高的时间分辨率,可以获取纳秒级别的精度。struct timespec包含两个成员:tv_sec(秒数)和tv_nsec(纳秒数)。clk_id参数指定了要查询的时间源(带有“可选”指并非所有系统都必须支持):
      • CLOCK_REALTIME
        描述:系统实时钟,反映当前的实际时间。
        特点:受系统时间调整的影响。
      • CLOCK_MONOTONIC
        描述:单调时钟,从某个未指定的起点开始计时。
        特点:不受系统时间调整的影响,适合用于测量时间间隔。
      • CLOCK_PROCESS_CPUTIME_ID
        描述:当前进程的CPU时间。
        特点:包括用户态和内核态的CPU时间。
      • CLOCK_THREAD_CPUTIME_ID
        描述:当前线程的CPU时间。
        特点:仅包括当前线程的CPU时间。
      • CLOCK_MONOTONIC_RAW (可选)
        描述:高精度单调时钟,不受系统时间调整的影响。
        特点:提供更高的时间分辨率。
      • CLOCK_REALTIME_COARSE (可选)
        描述:较低精度的系统实时钟。
        特点:速度快,但精度较低。
      • CLOCK_MONOTONIC_COARSE (可选)
        描述:较低精度的单调时钟。
        特点:速度快,但精度较低。
    • 返回值:成功时返回0,出错时返回-1,并设置errno
  • times:
    • 函数原型:clock_t times(struct tms *buf);
    • 功能描述:此函数用于获取进程所使用的时间信息,包括用户态和内核态下的运行时间。struct tms包含四个成员:tms_utime(用户态运行时间)、tms_stime(内核态运行时间)、tms_cutime(子进程用户态运行时间)、tms_cstime(子进程内核态运行时间),所有时间都以时钟滴答数(clock ticks)表示。
    • 返回值:成功时返回进程自开始执行以来所使用的时钟滴答数,若出错则返回-1L

时间设置

  • stime:
    • 函数原型:int stime(const time_t *t);
    • 功能描述:此函数用于将系统的实时钟设置为指定的时间。t是一个指向time_t类型变量的指针,该变量包含了自1970年1月1日00:00:00 UTC以来的秒数。
    • 返回值:成功时返回0,失败时返回-1,并设置errno
    • 注意事项:stime()函数通常需要root权限才能执行,且至 Linux 2.6.x之后版本不推荐使用,本地glibc 2.35实测已无法编译此函数。
  • settimeofday:
    • 函数原型:int settimeofday(const struct timeval *tv, const struct timezone *tz);
    • 功能描述:此函数允许设置系统的实时时间和时区信息。tv指向一个struct timeval结构,该结构包含了秒数和微秒数,用来表示新的系统时间。tz指向一个struct timezone结构,该结构包含了分钟偏移量和夏令时标志位,不过在现代系统中,通常不需要设置时区信息,因此可以传递NULL
    • 返回值:成功时返回0,失败时返回-1,并设置errno
    • 注意事项:与stime()类似,settimeofday()也需要适当的权限才能改变系统时间。
  • clock_settime:
    • 函数原型:int clock_settime(clockid_t clk_id, const struct timespec *tp);
    • 功能描述:此函数用于设置由clk_id标识的时钟。tp指向一个struct timespec结构,该结构包含了秒数和纳秒数,可以用来非常精确地设置时间。通常只允许设置时间源CLOCK_REALTIME(系统实时钟)。
    • 返回值:成功时返回0,失败时返回-1,并设置errno
    • 注意事项:修改系统实时钟通常需要root权限,而其他类型的时钟通常不允许设置。

时间转换

  • asctime / asctime_r(tm -> char*)
    • 函数原型:char *asctime(const struct tm *timeptr); / char *asctime_r(const struct tm *timeptr, char *buf);
    • 功能描述:将 struct tm 结构转换为字符串格式,格式为 "Sun Sep 16 01:03:52 1979\n"。asctime_r 是线程安全版本。
    • 返回值:返回指向字符串的指针。
    • 注意事项:asctime 返回的字符串是静态分配的,多次调用会覆盖前一次的结果。
  • mktime (tm -> time_t)
    • 函数原型:time_t mktime(struct tm *timeptr);
    • 功能描述:将 struct tm 结构转换为 time_t 类型的时间值。
    • 返回值:成功时返回 time_t 类型的时间值,失败时返回 (time_t)(-1)
    • 注意事项:mktime 可能会修改传入的 struct tm 结构中的某些字段。
  • ctime / ctime_r (time_t -> char*)
    • 函数原型:char *ctime(const time_t *timep); / char *ctime_r(const time_t *timep, char *buf);
    • 功能描述:将 time_t 类型的时间值转换为字符串格式,格式为 "Sun Sep 16 01:03:52 1979\n"。ctime_r 是线程安全版本。
    • 返回值:返回指向字符串的指针。
    • 注意事项:ctime 返回的字符串是静态分配的,多次调用会覆盖前一次的结果。
  • gmtime / gmtime_r (time_t -> tm ) UTC
    • 函数原型:struct tm *gmtime(const time_t *timep); / struct tm *gmtime_r(const time_t *timep, struct tm *result);
    • 功能描述:将 time_t 类型的时间值转换为 UTC 时间的 struct tm 结构。gmtime_r 是线程安全版本。
    • 返回值:成功时返回指向 struct tm 结构的指针,失败时返回 NULL
    • 注意事项:gmtime 返回的 struct tm 结构是静态分配的,多次调用会覆盖前一次的结果。
  • localtime / localtime_r (time_t -> tm) 本地时间
    • 函数原型:struct tm *localtime(const time_t *timep); / struct tm *localtime_r(const time_t *timep, struct tm *result);
    • 功能描述:将 time_t 类型的时间值转换为本地时间的 struct tm 结构。localtime_r 是线程安全版本。
    • 返回值:成功时返回指向 struct tm 结构的指针,失败时返回 NULL
    • 注意事项:localtime 返回的 struct tm 结构是静态分配的,多次调用会覆盖前一次的结果。
  • difftime (time_t -> double)
    • 函数原型:double difftime(time_t time1, time_t time0);
    • 功能描述:计算两个 time_t 类型的时间值之间的差值,以秒为单位。
    • 返回值:返回两个时间值之间的差值,以秒为单位。

时间格式化

  • strftime (tm -> char*)
    • 函数原型:size_t strftime(char *str, size_t maxsize, const char *format, const struct tm *timeptr);
    • 功能描述:根据指定的格式字符串 formatstruct tm 结构转换为字符串,并存储在 str 中。最多写入 maxsize 个字符(包括终止符 \0)。
    • 返回值:成功时返回实际写入的字符数(不包括终止符 \0),如果缓冲区太小无法容纳结果,则返回 0
    • 注意事项:确保提供的缓冲区 str 足够大,以避免溢出。

时区设置

  时区会影响到本地时间与UTC时间之间的转换(即本地时间 = UTC + 时区)。
  查阅了一些文档,目前Ubuntu上时区记录在路径/etc/localtime,其通常为软链接,指向具体的时区文件,例如 /etc/localtime -> /usr/share/zoneinfo/Asia/Shanghai。通过修改/etc/localtime指向即可修改为对应的时区(/etc/timezone也会记录当前时区,但似乎仅用于显示)。

实例测试

测试time/stime

time

void TestGetTime()
{
    // time UTC时间戳
    time_t tmt1 = time(NULL);
    printf("timestamp  : %ld\n", tmt1);

    // ctime_r UTC时间戳转换为本地时间字符串
    char cbuf[50] = {0};
    ctime_r(&tmt1, cbuf);
    printf("ctime_r    : %ld(%6d) %s", tmt1, 0, cbuf);

    // gmtime_r UTC时间戳转换为UTC时间字符串
    tm gtm;
    time_t tmt2;
    char gbuf[50] = {0};
    gmtime_r(&tmt1, &gtm);
    asctime_r(&gtm, gbuf);
    tmt2 = mktime(&gtm);    // mktime 会自动减时区
    printf("gmtime_r   : %ld(%6ld) %s %s", tmt2, tmt2-tmt1, gtm.tm_zone, gbuf);

    // 将时间戳转换为本地时间
    tm ltm;
    time_t tmt3;
    char lbuf[50] = {0};
    localtime_r(&tmt1, &ltm);
    asctime_r(&ltm, lbuf);
    tmt3 = mktime(&ltm);
    printf("localtime_r: %ld(%6ld) %s %s", tmt3, tmt3-tmt1, ltm.tm_zone, lbuf);

    char buf3[50] = {0};
    strftime(buf3, 50, "%Z %a %b %d %H:%M:%S %Y", &ltm);
    printf("strftime   : %ld(%6ld) %s\n", tmt3, tmt3-tmt1, buf3);
}

测试结果

timestamp  : 1732450363
ctime_r    : 1732450363(     0) Sun Nov 24 20:12:43 2024
gmtime_r   : 1732421563(-28800) CST Sun Nov 24 12:12:43 2024
localtime_r: 1732450363(     0) CST Sun Nov 24 20:12:43 2024
strftime   : 1732450363(     0) CST Sun Nov 24 20:12:43 2024

gmtime_r 打印的是UTC时间戳,与本地时间相差28800s (8h),即本地与UTC时间相差8h

测试gettimeofday/settimeofday

void Testgettimeofday()
{
    struct timeval tv;
    gettimeofday(&tv, NULL);
    printf("tv_sec: %ld, tv_usec: %ld\n", (long)tv.tv_sec, (long)tv.tv_usec);
}

void Testsettimeofday()
{
    Testgettimeofday();
    struct timeval tv1;
    tv1.tv_sec = 1731985300;
    tv1.tv_usec = 100;
    int ret = settimeofday(&tv1, NULL);
    if (ret == -1) {
        perror("settimeofday");
    }
    Testgettimeofday();
}

测试结果

tv_sec: 1732450828, tv_usec: 890873
tv_sec: 1731985300, tv_usec: 150

注意在调用设置时间接口时,需要root权限执行,否则会设置失败。

测试clock_gettime/clock_settime

void Testclock_gettime()
{
    std::string name[] = {
        "CLOCK_REALTIME",
        "CLOCK_MONOTONIC",
        "CLOCK_PROCESS_CPUTIME_ID",
        "CLOCK_THREAD_CPUTIME_ID",
        "CLOCK_MONOTONIC_RAW",
        "CLOCK_REALTIME_COARSE",
        "CLOCK_MONOTONIC_COARSE",
        "CLOCK_BOOTTIME",
        "CLOCK_REALTIME_ALARM",
        "CLOCK_BOOTTIME_ALARM",
    };

    // printf("Test clock_gettime\n");
    printf("%-25s  %10s  %10s\n", "CLOCK TYPE", "SEC", "NSEC");
    printf("-----------------------------------------------------------------------------\n");
    for (int i = 0; i <= CLOCK_BOOTTIME_ALARM; i++) {
        struct timespec ts;
        clock_gettime(i, &ts);
        printf("%-25s: %10ld, %10ld\n", name[i].c_str(), (long)ts.tv_sec, (long)ts.tv_nsec);
    }
    printf("-----------------------------------------------------------------------------\n");
}

void Testclock_settime()
{
    Testclock_gettime();

    // Only CLOCK_REALTIME is allowed to be set
    struct timespec ts1;
    ts1.tv_sec = 1731985300;
    ts1.tv_nsec = 100;
    int ret = clock_settime(CLOCK_REALTIME, &ts1);
    if (ret == -1) {
        perror("clock_settime");
    }

    Testclock_gettime();
}

测试结果

CLOCK TYPE                        SEC        NSEC
-----------------------------------------------------------------------------
CLOCK_REALTIME           : 1732451153,  160842537
CLOCK_MONOTONIC          :      45250,  516265743
CLOCK_PROCESS_CPUTIME_ID :          0,     908800
CLOCK_THREAD_CPUTIME_ID  :          0,     910400
CLOCK_MONOTONIC_RAW      :      45249,   35729391
CLOCK_REALTIME_COARSE    : 1732451153,  145187465
CLOCK_MONOTONIC_COARSE   :      45250,  500594052
CLOCK_BOOTTIME           :      45250,  516287258
CLOCK_REALTIME_ALARM     : 1732451153,  160881972
CLOCK_BOOTTIME_ALARM     :      45250,  516289642
-----------------------------------------------------------------------------
CLOCK TYPE                        SEC        NSEC
-----------------------------------------------------------------------------
CLOCK_REALTIME           : 1731985300,      32053
CLOCK_MONOTONIC          :      45250,  516347493
CLOCK_PROCESS_CPUTIME_ID :          0,     988400
CLOCK_THREAD_CPUTIME_ID  :          0,     989400
CLOCK_MONOTONIC_RAW      :      45249,   35795309
CLOCK_REALTIME_COARSE    : 1731985300,        100
CLOCK_MONOTONIC_COARSE   :      45250,  516314680
CLOCK_BOOTTIME           :      45250,  516352354
CLOCK_REALTIME_ALARM     : 1731985300,      38528
CLOCK_BOOTTIME_ALARM     :      45250,  516353885
-----------------------------------------------------------------------------

从测试结果看,更改系统时间时,仅有时间源CLOCK_REALTIMECLOCK_REALTIME_ALARM会随之修改而跳变,其他时间源不会随着系统时间的修改而跳变。在了解这些特性后,在编写应用程序时选择合适的时间源,以满足不同的需求。

测试sleep后,时间的变化

void TestTimeWithSleep(int sec)
{
    std::string name[] = {
        "CLOCK_REALTIME",
        "CLOCK_MONOTONIC",
        "CLOCK_PROCESS_CPUTIME_ID",
        "CLOCK_THREAD_CPUTIME_ID",
        "CLOCK_MONOTONIC_RAW",
        "CLOCK_REALTIME_COARSE",
        "CLOCK_MONOTONIC_COARSE",
        "CLOCK_BOOTTIME",
        "CLOCK_REALTIME_ALARM",
        "CLOCK_BOOTTIME_ALARM",
    };

    struct timespec ots[10];
    for (int i = 0; i < 10; i++) {
        clock_gettime(i, &ots[i]);
    }

    sleep(sec);
    struct  timespec nts[10];
    for (int j = 0; j < 10; j++) {
        clock_gettime(j, &nts[j]);
    }

    printf("%-25s  %10s %10s %10s %10s %7s %8s\n", "CLOCK TYPE", "OLDSEC", "OLDNSEC", "NEWSEC", "NEWNSEC", "DIFFSEC", "DIFFNSEC");
    printf("-------------------------------------------------------------------------------------------\n");
    for (int i = 0; i <= CLOCK_BOOTTIME_ALARM; i++) {
        printf("%-25s: %10ld %10ld %10ld %10ld %7ld %8ld\n",
        name[i].c_str(), (long)ots[i].tv_sec, (long)ots[i].tv_nsec,
        (long)nts[i].tv_sec, (long)nts[i].tv_nsec, (long)(nts[i].tv_sec - ots[i].tv_sec), (long)(nts[i].tv_nsec - ots[i].tv_nsec));
    }
}

测试结果

sleep 5s 结果如下:

CLOCK TYPE                     OLDSEC    OLDNSEC     NEWSEC    NEWNSEC DIFFSEC DIFFNSEC
-------------------------------------------------------------------------------------------
CLOCK_REALTIME           : 1732451618  944834581 1732451623  945683524       5   848943
CLOCK_MONOTONIC          :      45716  300258307      45721  301107230       5   848923
CLOCK_PROCESS_CPUTIME_ID :          0    1010700          0    1048800       0    38100
CLOCK_THREAD_CPUTIME_ID  :          0    1011000          0    1050000       0    39000
CLOCK_MONOTONIC_RAW      :      45714  819871428      45719  820723025       5   851597
CLOCK_REALTIME_COARSE    : 1732451618  935986823 1732451623  935984705       5    -2118
CLOCK_MONOTONIC_COARSE   :      45716  291410495      45721  291408377       5    -2118
CLOCK_BOOTTIME           :      45716  300260726      45721  301110251       5   849525
CLOCK_REALTIME_ALARM     : 1732451618  944837451 1732451623  945687049       5   849598
CLOCK_BOOTTIME_ALARM     :      45716  300287520      45721  301111127       5   823607

  从上述结果看,CLOCK_PROCESS_CPUTIME_IDCLOCK_THREAD_CPUTIME_ID没有记录sleep 5s的时间,也应征了上述所描述的进程挂起或停止时,进程时间不会记录。
  用times接口验证会更明显,sleep前后times获取的时间值基本没有变化。

测试修改时区

void TestSetTimeZone(const std::string& tz)
{
    int ret = 0;
    std::string target = "/usr/share/zoneinfo/" + tz;

    ret = unlink("/etc/localtime");
    if (ret == -1) {
        perror("unlink");
    }

    ret = symlink(target.c_str(), "/etc/localtime");
    if (ret == -1) {
        perror("symlink");
        return;
    }

    tzset();
    TestGetTimeZone();
    TestGetTime();
}

测试结果

设置时区America/New_York

timestamp  : 1732452775
ctime_r    : 1732452775(     0) Sun Nov 24 07:52:55 2024
gmtime_r   : 1732470775( 18000) EST Sun Nov 24 12:52:55 2024
localtime_r: 1732452775(     0) EST Sun Nov 24 07:52:55 2024
strftime   : 1732452775(     0) EST Sun Nov 24 07:52:55 2024

通过打印可看出时区已经显示EST,与Asia/Shanghai时区相差了13h。

总结

  • Linux 时间相关接口比较简单,之前没有系统了解过,一直使用的比较混乱,其实主要就是根据实际的精度需求选择对应的接口即可。
  • 在嵌入式开发项目中,时区管理是一项不可忽视的任务。通常情况下,通过GPS基站或网络时间协议(NTP)服务器进行时间同步以确保设备时区的准确性。在调整时区时,推荐仅更新系统的时区配置文件,而不是直接对系统时间进行增减操作,以此避免可能的时间计算错误。
  • 在实际项目中,推荐使用协调世界时(UTC)作为时间基准,而非依赖于本地时间。这是因为本地时间会因时区变更而发生变化,而UTC提供了一个全球统一的标准,不受地理位置的影响。
  • 时间服务是操作系统中的基础组成部分之一,因此在进行时间校准时,需要仔细规划校准的时间点。不恰当的时间跳跃可能导致依赖于系统时间的应用程序和服务出现故障。
  • 在过去的经验中,wait_for会随着时间跳变而异常。尽管印象中,不应该这样,其依赖的应该是相对时间即单调时间。经过查阅相关资料,发现gcc版本和glibc版本对wait_for都有影响,gcc >=10 且 glibc >= 2.30 才会对程序行为没有影响。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,347评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,435评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,509评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,611评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,837评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,987评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,730评论 0 267
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,194评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,525评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,664评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,334评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,944评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,764评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,997评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,389评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,554评论 2 349

推荐阅读更多精彩内容