手寫LRU緩存淘汰算法

手寫LRU緩存淘汰算法

背景

在我們這個(gè)日益追求高效的世界费薄,我們對(duì)任何事情的等待都顯得十分的浮躁硝全,網(wǎng)頁頁面刷新不出來,好煩楞抡,電腦打開運(yùn)行程序慢伟众,又是好煩!那怎么辦召廷,技術(shù)的產(chǎn)生不就是我們所服務(wù)么凳厢,今天我們就聊一聊緩存這個(gè)技術(shù),并使用我們熟知的數(shù)據(jù)結(jié)構(gòu)--用鏈表實(shí)現(xiàn)LRU緩存淘汰算法竞慢。

在學(xué)習(xí)如何使用鏈表實(shí)現(xiàn)LRU緩存淘汰算法前先紫,我們先提出幾個(gè)問題,大家好好思考下筹煮,問題如下:

  • 什么是緩存遮精,緩存的作用?
  • 緩存的淘汰策略有哪些败潦?
  • 如何使用鏈表實(shí)現(xiàn)LRU緩存淘汰算法本冲,有什么特點(diǎn),如何優(yōu)化劫扒?

好了檬洞,我們帶著上面的問題來學(xué)進(jìn)行下面的學(xué)習(xí)。

1粟关、什么是緩存疮胖,緩存的作用是什么?

緩存可以簡單的理解為保存數(shù)據(jù)的一個(gè)副本,以便于后續(xù)能夠快速的進(jìn)行訪問闷板。以計(jì)算機(jī)的使用場景為例澎灸,當(dāng)cpu要訪問內(nèi)存中的一條數(shù)據(jù)時(shí),它會(huì)先在緩存里查找遮晚,如果能夠找到則直接使用性昭,如果沒找到,則需要去內(nèi)存里查找县遣;

同樣的糜颠,在數(shù)據(jù)庫的訪問場景中,當(dāng)項(xiàng)目系統(tǒng)需要查詢數(shù)據(jù)庫中的某條數(shù)據(jù)時(shí)萧求,可以先讓請(qǐng)求查詢緩存其兴,如果命中,就直接返回緩存的結(jié)果夸政,如果沒有命中元旬,就查詢數(shù)據(jù)庫, 并將查詢結(jié)果放入緩存,下次請(qǐng)求時(shí)查詢緩存命中匀归,直接返回結(jié)果坑资,就不用再次查詢數(shù)據(jù)庫。

通過以上兩個(gè)例子穆端,我們發(fā)現(xiàn)無論在哪種場景下袱贮,都存在這樣一個(gè)順序:先緩存,后內(nèi)存体啰;先緩存攒巍,后數(shù)據(jù)庫。但是緩存的存在也占用了一部分內(nèi)存空間狡赐,所以緩存是典型的以空間換時(shí)間窑业,犧牲數(shù)據(jù)的實(shí)時(shí)性,卻滿足計(jì)算機(jī)運(yùn)行的高效性枕屉。

仔細(xì)想一下常柄,我們?nèi)粘i_發(fā)中遇到緩存的例子還挺多的。

  • 操作系統(tǒng)的緩存

減少與磁盤的交互

  • 數(shù)據(jù)庫緩存

減少對(duì)數(shù)據(jù)庫的查詢

  • Web服務(wù)器緩存

減少對(duì)應(yīng)用服務(wù)器的請(qǐng)求

  • 客戶瀏覽器的緩存

減少對(duì)網(wǎng)站的訪問

2搀擂、緩存有哪些淘汰策略?

緩存的本質(zhì)是以空間換時(shí)間西潘,那么緩存的容量大小肯定是有限的,當(dāng)緩存被占滿時(shí)哨颂,緩存中的那些數(shù)據(jù)應(yīng)該被清理出去喷市,那些數(shù)據(jù)應(yīng)該被保留呢?這就需要緩存的淘汰策略來決定威恼。

事實(shí)上品姓,常用的緩存的淘汰策略有三種:先進(jìn)先出算法(First in First out FIFO);淘汰一定時(shí)期內(nèi)被訪問次數(shù)最少的頁面(Least Frequently Used LFU)箫措;淘汰最長時(shí)間未被使用的頁面(Least Recently Used LRU)

這些算法在不同層次的緩存上執(zhí)行時(shí)具有不同的效率腹备,需要結(jié)合具體的場景來選擇。

2.1 FIFO算法

FIFO算法即先進(jìn)先出算法斤蔓,常采用隊(duì)列實(shí)現(xiàn)植酥。在緩存中,它的設(shè)計(jì)原則是:如果一個(gè)數(shù)據(jù)最先進(jìn)入緩存中弦牡,則應(yīng)該最早淘汰掉友驮。

image-20210227115124480
  • 新訪問的數(shù)據(jù)插入FIFO隊(duì)列的尾部,隊(duì)列中數(shù)據(jù)由隊(duì)到隊(duì)頭按順序順序移動(dòng)
  • 隊(duì)列滿時(shí)驾锰,刪除隊(duì)頭的數(shù)據(jù)

2.2 LRU算法

LRU算法是根據(jù)對(duì)數(shù)據(jù)的歷史訪問次數(shù)來進(jìn)行淘汰數(shù)據(jù)的卸留,通常使用鏈表來實(shí)現(xiàn)。在緩存中椭豫,它的設(shè)計(jì)原則是:如果數(shù)據(jù)最近被訪問過耻瑟,那么將來它被訪問的幾率也很高买喧。

image-20210227120433527
  • 新加入的數(shù)據(jù)插入到鏈表的頭部
  • 每當(dāng)緩存命中時(shí)(即緩存數(shù)據(jù)被訪問),則將數(shù)據(jù)移到鏈表頭部
  • 當(dāng)來鏈表已滿的時(shí)候匆赃,將鏈表尾部的數(shù)據(jù)丟棄

2.3 LFU算法

LFU算法是根據(jù)數(shù)據(jù)的歷史訪問頻率來淘汰數(shù)據(jù),因此今缚,LFU算法中的每個(gè)數(shù)據(jù)塊都有一個(gè)引用計(jì)數(shù)算柳,所有數(shù)據(jù)塊按照引用計(jì)數(shù)排序,具有相同引用計(jì)數(shù)的數(shù)據(jù)塊則按照時(shí)間排序姓言。在緩存中瞬项,它的設(shè)計(jì)原則是:如果數(shù)據(jù)被訪問多次,那么將來它的訪問頻率也會(huì)更高何荚。

image-20210227121952816
  • 新加入數(shù)據(jù)插入到隊(duì)列尾部(引用計(jì)數(shù)為1囱淋;
  • 隊(duì)列中的數(shù)據(jù)被訪問后,引用計(jì)數(shù)增加餐塘,隊(duì)列重新排序妥衣;
  • 當(dāng)需要淘汰數(shù)據(jù)時(shí),將已經(jīng)排序的列表最后的數(shù)據(jù)塊刪除戒傻。

3税手、如何使用鏈表實(shí)現(xiàn)緩存淘汰,有什么特點(diǎn)需纳,如何優(yōu)化芦倒?

在上面的文章中我們理解了緩存的概念淘汰策略,其中LRU算法是筆試/面試中考察比較頻繁的不翩,我秋招的時(shí)候兵扬,很多公司都讓我手寫了這個(gè)算法,為了避免大家采坑口蝠,下面器钟,我們就手寫一個(gè)LRU緩存淘汰算法。

我們都知道鏈表的形式不止一種亚皂,我們應(yīng)該選擇哪一種呢俱箱?

思考三分鐘........

好了,公布答案灭必!

事實(shí)上狞谱,鏈表按照不同的連接結(jié)構(gòu)可以劃分為單鏈表循環(huán)鏈表雙向鏈表禁漓。

  • 單鏈表
  • 每個(gè)節(jié)點(diǎn)只包含一個(gè)指針跟衅,即后繼指針。
  • 單鏈表有兩個(gè)特殊的節(jié)點(diǎn)播歼,即首節(jié)點(diǎn)和尾節(jié)點(diǎn)伶跷,用首節(jié)點(diǎn)地址表示整條鏈表掰读,尾節(jié)點(diǎn)的后繼指針指向空地址null。
  • 性能特點(diǎn):插入和刪除節(jié)點(diǎn)的時(shí)間復(fù)雜度為O(1)叭莫,查找的時(shí)間復(fù)雜度為O(n)蹈集。
  • 循環(huán)鏈表
  • 除了尾節(jié)點(diǎn)的后繼指針指向首節(jié)點(diǎn)的地址外均與單鏈表一致。
  • 適用于存儲(chǔ)有循環(huán)特點(diǎn)的數(shù)據(jù)雇初,比如約瑟夫問題拢肆。
  • 雙向鏈表
  • 節(jié)點(diǎn)除了存儲(chǔ)數(shù)據(jù)外,還有兩個(gè)指針分別指向前一個(gè)節(jié)點(diǎn)地址(前驅(qū)指針prev)和下一個(gè)節(jié)點(diǎn)地址(后繼指針next)

  • 首節(jié)點(diǎn)的前驅(qū)指針prev和尾節(jié)點(diǎn)的后繼指針均指向空地址靖诗。

雙向鏈表相較于單鏈表的一大優(yōu)勢在于:找到前驅(qū)節(jié)點(diǎn)的時(shí)間復(fù)雜度為O(1)郭怪,而單鏈表只能從頭節(jié)點(diǎn)慢慢往下找,所以仍然是O(n).而且刊橘,對(duì)于插入和刪除也是有優(yōu)化的鄙才。

我們可能會(huì)有問題:單鏈表的插入刪除不是O(1)嗎?

是的促绵,但是一般情況下攒庵,我們想要進(jìn)行插入刪除操作,很多時(shí)候還是得先進(jìn)行查找绞愚,再插入或者刪除叙甸,可見其實(shí)是先O(n),再O(1)。

不熟悉鏈表解題的同學(xué)可以先看看我的上一篇算法解析文章刷了LeetCode鏈表專題位衩,我發(fā)現(xiàn)了一個(gè)秘密裆蒸。

因?yàn)槲覀冃枰獎(jiǎng)h除操作,刪除一個(gè)節(jié)點(diǎn)不僅要得到該節(jié)點(diǎn)本身的指針糖驴,也需要操作其它前驅(qū)節(jié)點(diǎn)的指針僚祷,而雙向鏈表能夠直接找到前驅(qū),保證了操作時(shí)間復(fù)雜度為O(1)贮缕,因此使用雙向鏈表作為實(shí)現(xiàn)LRU緩存淘汰算法的結(jié)構(gòu)會(huì)更高效辙谜。

算法思路

維護(hù)一個(gè)雙向鏈表,保存所有緩存的值感昼,并且最老的值放在鏈表最后面装哆。

  • 當(dāng)訪問的值在鏈表中時(shí): 將找到鏈表中值將其刪除,并重新在鏈表頭添加該值(保證鏈表中 數(shù)值的順序是從新到舊)
  • 當(dāng)訪問的值不在鏈表中時(shí): 當(dāng)鏈表已滿:刪除鏈表最后一個(gè)值定嗓,將要添加的值放在鏈表頭 當(dāng)鏈表未滿:直接在鏈表頭添加

3.1 LRU緩存淘汰算法

極客時(shí)間王爭的《數(shù)據(jù)結(jié)構(gòu)與算法之美》給出了一個(gè)使用有序單鏈表實(shí)現(xiàn)LRU緩存淘汰算法蜕琴,代碼如下:

public class LRUBaseLinkedList<T> {

    /**
     * 默認(rèn)鏈表容量
     */
    private final static Integer DEFAULT_CAPACITY = 10;

    /**
     * 頭結(jié)點(diǎn)
     */
    private SNode<T> headNode;

    /**
     * 鏈表長度
     */
    private Integer length;

    /**
     * 鏈表容量
     */
    private Integer capacity;

    public LRUBaseLinkedList() {
        this.headNode = new SNode<>();
        this.capacity = DEFAULT_CAPACITY;
        this.length = 0;
    }

    public LRUBaseLinkedList(Integer capacity) {
        this.headNode = new SNode<>();
        this.capacity = capacity;
        this.length = 0;
    }

    public void add(T data) {
        SNode preNode = findPreNode(data);

        // 鏈表中存在,刪除原數(shù)據(jù)宵溅,再插入到鏈表的頭部
        if (preNode != null) {
            deleteElemOptim(preNode);
            intsertElemAtBegin(data);
        } else {
            if (length >= this.capacity) {
                //刪除尾結(jié)點(diǎn)
                deleteElemAtEnd();
            }
            intsertElemAtBegin(data);
        }
    }

    /**
     * 刪除preNode結(jié)點(diǎn)下一個(gè)元素
     *
     * @param preNode
     */
    private void deleteElemOptim(SNode preNode) {
        SNode temp = preNode.getNext();
        preNode.setNext(temp.getNext());
        temp = null;
        length--;
    }

    /**
     * 鏈表頭部插入節(jié)點(diǎn)
     *
     * @param data
     */
    private void intsertElemAtBegin(T data) {
        SNode next = headNode.getNext();
        headNode.setNext(new SNode(data, next));
        length++;
    }

    /**
     * 獲取查找到元素的前一個(gè)結(jié)點(diǎn)
     *
     * @param data
     * @return
     */
    private SNode findPreNode(T data) {
        SNode node = headNode;
        while (node.getNext() != null) {
            if (data.equals(node.getNext().getElement())) {
                return node;
            }
            node = node.getNext();
        }
        return null;
    }

    /**
     * 刪除尾結(jié)點(diǎn)
     */
    private void deleteElemAtEnd() {
        SNode ptr = headNode;
        // 空鏈表直接返回
        if (ptr.getNext() == null) {
            return;
        }

        // 倒數(shù)第二個(gè)結(jié)點(diǎn)
        while (ptr.getNext().getNext() != null) {
            ptr = ptr.getNext();
        }

        SNode tmp = ptr.getNext();
        ptr.setNext(null);
        tmp = null;
        length--;
    }

    private void printAll() {
        SNode node = headNode.getNext();
        while (node != null) {
            System.out.print(node.getElement() + ",");
            node = node.getNext();
        }
        System.out.println();
    }

    public class SNode<T> {

        private T element;

        private SNode next;

        public SNode(T element) {
            this.element = element;
        }

        public SNode(T element, SNode next) {
            this.element = element;
            this.next = next;
        }

        public SNode() {
            this.next = null;
        }

        public T getElement() {
            return element;
        }

        public void setElement(T element) {
            this.element = element;
        }

        public SNode getNext() {
            return next;
        }

        public void setNext(SNode next) {
            this.next = next;
        }
    }

    public static void main(String[] args) {
        LRUBaseLinkedList list = new LRUBaseLinkedList();
        Scanner sc = new Scanner(System.in);
        while (true) {
            list.add(sc.nextInt());
            list.printAll();
        }
    }
}

這段代碼不管緩存有沒有滿凌简,都需要遍歷一遍鏈表,所以這種基于鏈表的實(shí)現(xiàn)思路恃逻,緩存訪問的時(shí)間復(fù)雜度為 O(n)雏搂。

3.2使用哈希表優(yōu)化LRU

事實(shí)上藕施,這個(gè)思路還可以繼續(xù)優(yōu)化,我們可以把單鏈表換成雙向鏈表凸郑,并引入散列表裳食。

  • 雙向鏈表支持查找前驅(qū),保證操作的時(shí)間復(fù)雜度為O(1)
  • 引入散列表記錄每個(gè)數(shù)據(jù)的位置芙沥,將緩存訪問的時(shí)間復(fù)雜度降到O(1)

哈希表查找較快胞谈,但是數(shù)據(jù)無固定的順序;鏈表倒是有順序之分憨愉。插入、刪除較快卿捎,但是查找較慢配紫。將它們結(jié)合,就可以形成一種新的數(shù)據(jù)結(jié)構(gòu)--哈希鏈表(LinkedHashMap)

image-20210227203448255

力扣上146題-LRU緩存機(jī)制剛好可以拿來練手午阵,題圖如下:

題目:

運(yùn)用你所掌握的數(shù)據(jù)結(jié)構(gòu)躺孝,設(shè)計(jì)和實(shí)現(xiàn)一個(gè) LRU (最近最少使用) 緩存機(jī)制 。

  • 實(shí)現(xiàn) LRUCache 類:

LRUCache(int capacity) 以正整數(shù)作為容量 capacity 初始化 LRU 緩存
int get(int key) 如果關(guān)鍵字 key 存在于緩存中底桂,則返回關(guān)鍵字的值植袍,否則返回 -1 。
void put(int key, int value) 如果關(guān)鍵字已經(jīng)存在籽懦,則變更其數(shù)據(jù)值于个;如果關(guān)鍵字不存在,則插入該組「關(guān)鍵字-值」暮顺。當(dāng)緩存容量達(dá)到上限時(shí)厅篓,它應(yīng)該在寫入新數(shù)據(jù)之前刪除最久未使用的數(shù)據(jù)值,從而為新的數(shù)據(jù)值留出空間捶码。

思路:

我們的思路就是哈希表+雙向鏈表

  • 哈希表用于滿足題目時(shí)間復(fù)雜度O(1)的要求羽氮,雙向鏈表用于存儲(chǔ)順序
  • 哈希表鍵值類型:<Integer, ListNode>,哈希表的鍵用于存儲(chǔ)輸入的key惫恼,哈希表的值用于存儲(chǔ)雙向鏈表的節(jié)點(diǎn)
  • 雙向鏈表的節(jié)點(diǎn)中除了value外還需要包含key档押,因?yàn)樵趧h除最久未使用的數(shù)據(jù)時(shí),需要通過鏈表來定位hashmap中應(yīng)當(dāng)刪除的鍵值對(duì)
  • 一些操作:雙向鏈表中祈纯,在后面的節(jié)點(diǎn)表示被最近訪問
  • 新加入的節(jié)點(diǎn)放在鏈表末尾令宿,addNodeToLast(node)
  • 若容量達(dá)到上限,去除最久未使用的數(shù)據(jù)盆繁,removeNode(head.next)
  • 若數(shù)據(jù)新被訪問過掀淘,比如被get了或被put了新值,把該節(jié)點(diǎn)挪到鏈表末尾油昂,moveNodeToLast(node)
  • 為了操作的方便革娄,在雙向鏈表頭和尾分別定義一個(gè)head和tail節(jié)點(diǎn)倾贰。

代碼

class LRUCache {
    private int capacity;
    private HashMap<Integer, ListNode> hashmap; 
    private ListNode head;
    private ListNode tail;

    private class ListNode{
        int key;
        int val;
        ListNode prev;
        ListNode next;
        public ListNode(){  
        }
        public ListNode(int key, int val){
            this.key = key;
            this.val = val;
        }
    }

    public LRUCache(int capacity) {
        this.capacity = capacity;
        hashmap = new HashMap<>();
        head = new ListNode();
        tail = new ListNode();
        head.next = tail;
        tail.prev = head;
    }

    private void removeNode(ListNode node){
        node.prev.next = node.next;
        node.next.prev = node.prev;
    }

    private void addNodeToLast(ListNode node){
        node.prev = tail.prev;
        node.prev.next = node;
        node.next = tail;
        tail.prev= node;
    }

    private void moveNodeToLast(ListNode node){
        removeNode(node);
        addNodeToLast(node);
    }
    
    public int get(int key) {   
        if(hashmap.containsKey(key)){
            ListNode node = hashmap.get(key);
            moveNodeToLast(node);
            return node.val;
        }else{
            return -1;
        }
    }
    
    public void put(int key, int value) {
        if(hashmap.containsKey(key)){
            ListNode node = hashmap.get(key);
            node.val = value;
            moveNodeToLast(node);
            return;
        }
        if(hashmap.size() == capacity){
            hashmap.remove(head.next.key);
            removeNode(head.next);
        }

        ListNode node = new ListNode(key, value);
        hashmap.put(key, node);
        addNodeToLast(node);
    }
}

巨人的肩膀

[1]數(shù)據(jù)結(jié)構(gòu)與算法之美-王爭

[2]力扣-LRU緩存機(jī)制(146題)

[3]https://blog.csdn.net/yangpl_tale/article/details/44998423

[4]https://leetcode-cn.com/problems/lru-cache/solution/146-lru-huan-cun-ji-zhi-ha-xi-biao-shuan-l3um/

本文由博客群發(fā)一文多發(fā)等運(yùn)營工具平臺(tái) OpenWrite 發(fā)布

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市拦惋,隨后出現(xiàn)的幾起案子匆浙,更是在濱河造成了極大的恐慌,老刑警劉巖厕妖,帶你破解...
    沈念sama閱讀 206,968評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件首尼,死亡現(xiàn)場離奇詭異,居然都是意外死亡言秸,警方通過查閱死者的電腦和手機(jī)软能,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來举畸,“玉大人查排,你說我怎么就攤上這事〕冢” “怎么了跋核?”我有些...
    開封第一講書人閱讀 153,220評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長叛买。 經(jīng)常有香客問我砂代,道長,這世上最難降的妖魔是什么率挣? 我笑而不...
    開封第一講書人閱讀 55,416評(píng)論 1 279
  • 正文 為了忘掉前任刻伊,我火速辦了婚禮,結(jié)果婚禮上椒功,老公的妹妹穿的比我還像新娘娃圆。我一直安慰自己,他們只是感情好蛾茉,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,425評(píng)論 5 374
  • 文/花漫 我一把揭開白布讼呢。 她就那樣靜靜地躺著,像睡著了一般谦炬。 火紅的嫁衣襯著肌膚如雪悦屏。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,144評(píng)論 1 285
  • 那天键思,我揣著相機(jī)與錄音础爬,去河邊找鬼。 笑死吼鳞,一個(gè)胖子當(dāng)著我的面吹牛看蚜,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播赔桌,決...
    沈念sama閱讀 38,432評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼供炎,長吁一口氣:“原來是場噩夢啊……” “哼渴逻!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起音诫,我...
    開封第一講書人閱讀 37,088評(píng)論 0 261
  • 序言:老撾萬榮一對(duì)情侶失蹤惨奕,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后竭钝,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體梨撞,經(jīng)...
    沈念sama閱讀 43,586評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,028評(píng)論 2 325
  • 正文 我和宋清朗相戀三年香罐,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了卧波。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,137評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡庇茫,死狀恐怖幽勒,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情港令,我是刑警寧澤,帶...
    沈念sama閱讀 33,783評(píng)論 4 324
  • 正文 年R本政府宣布锈颗,位于F島的核電站顷霹,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏击吱。R本人自食惡果不足惜淋淀,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,343評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望覆醇。 院中可真熱鬧朵纷,春花似錦、人聲如沸永脓。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽常摧。三九已至搅吁,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間落午,已是汗流浹背谎懦。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評(píng)論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留溃斋,地道東北人界拦。 一個(gè)月前我還...
    沈念sama閱讀 45,595評(píng)論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像梗劫,于是被迫代替她去往敵國和親享甸。 傳聞我的和親對(duì)象是個(gè)殘疾皇子截碴,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,901評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容