ARM架構(gòu)下的Docker環(huán)境校坑,OpenJDK官方?jīng)]有8版本鏡像拣技,如何完美解決千诬?

為什么需要ARM架構(gòu)下的OpenJDK8的Docker鏡像耍目?

對現(xiàn)有的Java應用,之前一直運行在x86處理器環(huán)境下徐绑,編譯和運行都是JDK8邪驮,如今在樹莓派的Docker環(huán)境運行(或者其他ARM架構(gòu)電腦,例如華為泰山服務器)傲茄,需要JDK8鏡像作為基礎(chǔ)鏡像毅访。

現(xiàn)在有什么問題?

《ARM64架構(gòu)下盘榨,OpenJDK的官方Docker鏡像為何沒有8版本喻粹?》一文中,已經(jīng)確定了OpenJDK官方并未提供8版本的Docker鏡像草巡,因此守呜,原有的Java應用,如果是基于JDK8編譯和運行的,現(xiàn)在從X86架構(gòu)轉(zhuǎn)戰(zhàn)到ARM架構(gòu)的Docker環(huán)境下查乒,就會面臨沒有JDK基礎(chǔ)鏡像的問題弥喉;

應對之道

《ARM64架構(gòu)下,OpenJDK的官方Docker鏡像為何沒有8版本玛迄?》一文曾經(jīng)提到應對之道:

  1. 自己編譯一個8版本的OpenJDK安裝包由境,以此來做Docker鏡像;
  2. Oracle提供了ARM版本的JDKD安裝包蓖议,以此包來做Docker鏡像虏杰;
  3. 用OpenJDK的11版本,但是11和8的差異要自行處理拒担;

對于第一種方式嘹屯,自己編譯8版本的OpenJDK,難度太大(對我自己而言)从撼,因為編譯OpenJDK需要低版本的OpenJDK作為編譯工具州弟,也就是說我要找到ARM版本的OpenJDK7,才能編譯ARM版本的OpenJDK8低零,因此我覺得這樣做的難度太大…

今天要討論的是第二種和第三種婆翔,

環(huán)境信息

  1. 硬件:樹莓派4B
  2. 操作系統(tǒng):openfans的64為Debian
  3. Docker:19.03.1
  4. docker-compose:1.24.1

參考文檔

  1. 在樹莓派4B安裝64位Debian和Docker的方法,請參考《樹莓派4B安裝64位Linux(不用顯示器鍵盤鼠標)》
  2. 在樹莓派4B安裝docker-compose的方法掏婶,請參考《樹莓派4B安裝docker-compose(64位Linux)》
  3. 將Java應用制作成Docker鏡像啃奴,請參考《Docker與Jib(maven插件版)實戰(zhàn)》

Java應用的源碼

本文要解決的問題是ARM架構(gòu)的電腦上,如何在Docker環(huán)境運行Java應用雄妥,因此需要有個Java應用來驗證最蕾,這里找了個最普通的SpringBoot應用,提供一個hello world的http接口老厌,通過jib插件構(gòu)建成Docker鏡像瘟则,整個應用的源碼可以從GitHub上下載,地址和鏈接信息如下表所示:

名稱 鏈接 備注
項目主頁 https://github.com/zq2599/blog_demos 該項目在GitHub上的主頁
git倉庫地址(https) https://github.com/zq2599/blog_demos.git 該項目源碼的倉庫地址枝秤,https協(xié)議
git倉庫地址(ssh) git@github.com:zq2599/blog_demos.git 該項目源碼的倉庫地址醋拧,ssh協(xié)議

</br>

這個git項目中有多個文件夾,本章的源碼在hellojib文件夾下淀弹,如下圖紅框所示:


在這里插入圖片描述

操作步驟簡介

接下來的操作步驟丹壕,如下圖所示:


在這里插入圖片描述

ARM機器上安裝JDK

要想在ARM機器上編譯構(gòu)建hellojib工程,就要把JDK和Maven裝好薇溃,先裝JDK菌赖;

  1. 去Oracle網(wǎng)站下載ARM版本的JDK8,地址是:https://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.html 沐序,如下圖紅框所示:
    在這里插入圖片描述
  2. 上述JDK文件解壓后是個名為<font color="blue">jdk1.8.0_221</font>的文件夾琉用,將此文件夾放在ARM電腦的<font color="blue">/usr/lib/jvm</font>目錄下忿峻;
  3. 打開文件<font color="blue">~/.bashrc</font>,增加以下內(nèi)容:
export JAVA_HOME=/usr/lib/jvm/jdk1.8.0_221
export JRE_HOME=${JAVA_HOME}/jre
export CLASSPATH=.:${JAVA_HOME}/lib:${JRE_HOME}/lib
export PATH=${JAVA_HOME}/bin:$PATH
  1. 執(zhí)行<font color="blue">source ~/.bashrc</font>辕羽,使得配置立即生效逛尚;
  2. 執(zhí)行命令<font color="blue">java -version</font>試試JDK是否已經(jīng)可用:
root@raspbian:~# java -version
java version "1.8.0_221"
Java(TM) SE Runtime Environment (build 1.8.0_221-b11)
Java HotSpot(TM) 64-Bit Server VM (build 25.221-b11, mixed mode)

ARM機器上安裝Maven

  1. 去Maven官網(wǎng)下載安裝包,我這里下載的是3.6.2版本刁愿,地址:https://www-eu.apache.org/dist/maven/maven-3/3.6.2/binaries/apache-maven-3.6.2-bin.tar.gz
  2. 安裝包解壓之后是個名為<font color="blue">apache-maven-3.6.2</font>的文件夾绰寞,將此文件夾放在ARM電腦的<font color="blue">/usr/local</font>目錄下;
  3. 打開文件<font color="blue">~/.bashrc</font>铣口,增加以下內(nèi)容:
export MAVEN_HOME=/usr/local/apache-maven-3.6.2
export PATH=$MAVEN_HOME/bin:$PATH
  1. 執(zhí)行<font color="blue">source ~/.bashrc</font>滤钱,使得配置立即生效;
  2. 執(zhí)行命令<font color="blue">mvn -version</font>試試maven是否已經(jīng)可用:
root@raspbian:~# mvn -version
Apache Maven 3.6.2 (40f52333136460af0dc0d7232c0dc0bcf0d9e117; 2019-08-27T23:06:16+08:00)
Maven home: /usr/local/apache-maven-3.6.2
Java version: 1.8.0_221, vendor: Oracle Corporation, runtime: /usr/lib/jvm/jdk1.8.0_221/jre
Default locale: en_US, platform encoding: UTF-8
OS name: "linux", version: "4.19.69-openfans+20190906-v8", arch: "aarch64", family: "unix"

折騰了這么久脑题,總算是完成了準備工作件缸,接下來開始做作鏡像了;

要把咱們自己的Java應用做成Docker鏡像叔遂,需要有個JDK8鏡像作為基礎(chǔ)鏡像他炊,接下來我們來做這個基礎(chǔ)鏡像;

自己動手已艰,做一個JDK8鏡像

  1. 在ARM電腦上新建一個文件夾痊末,里面新建名為<font color="blue">Dockerfile</font>的文件,內(nèi)容如下:
#Docker image of JDK8 in ARM64
# VERSION 8
# Author: bolingcavalry

#基礎(chǔ)鏡像使用的是OpenJDK官方鏡像公用的
FROM buildpack-deps:stretch-scm

#作者
MAINTAINER BolingCavalry <zq2599@gmail.com>

# Default to UTF-8 file.encoding
ENV LANG C.UTF-8

ENV JAVA_HOME /usr/local/jdk8

ENV PATH $JAVA_HOME/bin:$PATH

ENV JDK_FILE jdk-8u221-linux-arm64-vfp-hflt.tar.gz

COPY $JDK_FILE /usr/local/

RUN mkdir -p "$JAVA_HOME"; \
    tar --extract \
        --file /usr/local/$JDK_FILE \
        --directory "$JAVA_HOME" \
        --strip-components 1 \
        --no-same-owner; \
    rm /usr/local/$JDK_FILE
  1. 將之前下載的<font color="blue">jdk-8u221-linux-arm64-vfp-hflt.tar.gz</font>文件復制到Dockerfile文件所在目錄哩掺;
  2. 在Dockerfile文件所在目錄執(zhí)行命令<font color="blue">docker build -t bolingcavalry/arm64jdk:8 .</font>(注意命令的末尾有個小數(shù)點凿叠,不要漏了)
  3. 執(zhí)行成功后控制臺輸出以下信息:
root@raspbian:~/test# docker build -t bolingcavalry/arm64jdk:8 .
Sending build context to Docker daemon   73.2MB
Step 1/8 : FROM buildpack-deps:stretch-scm
 ---> 1838b930d30a
Step 2/8 : MAINTAINER BolingCavalry <zq2599@gmail.com>
 ---> Using cache
 ---> ce7488aef612
Step 3/8 : ENV LANG C.UTF-8
 ---> Using cache
 ---> 0bdb9ce285a9
Step 4/8 : ENV JAVA_HOME /usr/local/jdk8
 ---> Using cache
 ---> 39a412f0d874
Step 5/8 : ENV PATH $JAVA_HOME/bin:$PATH
 ---> Using cache
 ---> 1a3077d36d4e
Step 6/8 : ENV JDK_FILE jdk-8u221-linux-arm64-vfp-hflt.tar.gz
 ---> Using cache
 ---> c39af69e10f7
Step 7/8 : COPY $JDK_FILE /usr/local/
 ---> Using cache
 ---> 5fc704c5b9ff
Step 8/8 : RUN mkdir -p "$JAVA_HOME";     tar --extract         --file /usr/local/$JDK_FILE         --directory "$JAVA_HOME"        --strip-components 1        --no-same-owner;     rm /usr/local/$JDK_FILE
 ---> Running in a63663306adc
Removing intermediate container a63663306adc
 ---> ddc652d5dec0
Successfully built ddc652d5dec0
Successfully tagged bolingcavalry/arm64jdk:8
  1. 驗證一下效果:
root@raspbian:~/test# docker run --rm bolingcavalry/arm64jdk:8 java -version
java version "1.8.0_221"
Java(TM) SE Runtime Environment (build 1.8.0_221-b11)
Java HotSpot(TM) 64-Bit Server VM (build 25.221-b11, mixed mode)

把鏡像推送到鏡像倉庫

將剛剛做好的鏡像推送到倉庫,這么做的原因如下:

  1. 可以讓大家使用到此鏡像嚼吞;
  2. 接下來要用Jib插件將Java應用制作成鏡像盒件,Jib插件一定要從鏡像倉庫下載<font color="blue">bolingcavalry/arm64jdk:8</font>作為Java應用的基礎(chǔ)鏡像;

這里我將<font color="blue">bolingcavalry/arm64jdk:8</font>推送到了hub.docker.com舱禽,如果您沒有hub.docker.com的賬號炒刁,也可以選擇推送到私有鏡像倉庫,只要是鏡像倉庫呢蔫,Jib插件都支持切心;

將Java應用構(gòu)建成鏡像

  1. 以前面提到的hellojib為例飒筑,打開pom.xml文件片吊,將jib插件的配置改為如下內(nèi)容:
<!--使用jib插件-->
            <plugin>
                <groupId>com.google.cloud.tools</groupId>
                <artifactId>jib-maven-plugin</artifactId>
                <version>1.3.0</version>
                <configuration>
                    <!--from節(jié)點用來設(shè)置鏡像的基礎(chǔ)鏡像,相當于Docerkfile中的FROM關(guān)鍵字-->
                    <from>
                        <!--使用openjdk官方鏡像协屡,tag是8-jdk-stretch,表示鏡像的操作系統(tǒng)是debian9,裝好了jdk8-->
                <image>bolingcavalry/arm64jdk:8</image>
                    </from>
                    <to>
                        <!--鏡像名稱和tag,使用了mvn內(nèi)置變量${project.version}寨典,表示當前工程的version-->
                        <image>bolingcavalry/hellojib:${project.version}</image>
                    </to>
                    <!--容器相關(guān)的屬性-->
                    <container>
                        <!--jvm內(nèi)存參數(shù)-->
                        <jvmFlags>
                            <jvmFlag>-Xms1g</jvmFlag>
                            <jvmFlag>-Xmx1g</jvmFlag>
                        </jvmFlags>
                        <!--要暴露的端口-->
                        <ports>
                            <port>8080</port>
                        </ports>
                    </container>
                </configuration>
            </plugin>
  1. 執(zhí)行命令<font color="blue">mvn clean compile jib:dockerBui-U</font>即可構(gòu)建鏡像调鬓,控制臺輸出如下信息(友情提示认然,這可能是個漫長的等待過程,我這等了9分多鐘):
root@raspbian:~/test/hellojib# mvn clean compile jib:dockerBuild -U
[INFO] Scanning for projects...
[INFO] 
[INFO] ---------------------< com.bolingcavalry:hellojib >---------------------
[INFO] Building hellojib 0.0.1-SNAPSHOT
[INFO] --------------------------------[ jar ]---------------------------------
[INFO] 
[INFO] --- maven-clean-plugin:3.1.0:clean (default-clean) @ hellojib ---
[INFO] Deleting /root/test/hellojib/target
[INFO] 
[INFO] --- maven-resources-plugin:3.1.0:resources (default-resources) @ hellojib ---
[INFO] Using 'UTF-8' encoding to copy filtered resources.
[INFO] Copying 1 resource
[INFO] Copying 0 resource
[INFO] 
[INFO] --- maven-compiler-plugin:3.8.1:compile (default-compile) @ hellojib ---
[INFO] Changes detected - recompiling the module!
[INFO] Compiling 2 source files to /root/test/hellojib/target/classes
[INFO] 
[INFO] --- jib-maven-plugin:1.3.0:dockerBuild (default-cli) @ hellojib ---
[INFO] 
[INFO] Containerizing application to Docker daemon as bolingcavalry/hellojib:0.0.1-SNAPSHOT...
[INFO] The base image requires auth. Trying again for bolingcavalry/arm64jdk:8...
[INFO] 
[INFO] Container entrypoint set to [java, -Xms1g, -Xmx1g, -cp, /app/resources:/app/classes:/app/libs/*, com.bolingcavalry.hellojib.HellojibApplication]
[INFO] 
[INFO] Built image to Docker daemon as bolingcavalry/hellojib:0.0.1-SNAPSHOT
[INFO] Executing tasks:
[INFO] [==============================] 100.0% complete
[INFO] 
[INFO] ------------------------------------------------------------------------
[INFO] BUILD SUCCESS
[INFO] ------------------------------------------------------------------------
[INFO] Total time:  09:59 min
[INFO] Finished at: 2019-10-02T19:42:42+08:00
[INFO] ------------------------------------------------------------------------
  1. 驗證hellojib工程的鏡像是否正常漫萄,執(zhí)行命令<font color="blue">docker run --rm -p 8080:8080 bolingcavalry/hellojib:0.0.1-SNAPSHOT</font>卷员,控制臺顯示SpringBoot應用啟動成功:
root@raspbian:~/test/hellojib# docker run --rm -p 8080:8080 bolingcavalry/hellojib:0.0.1-SNAPSHOT

  .   ____          _            __ _ _
 /\\ / ___'_ __ _ _(_)_ __  __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
 \\/  ___)| |_)| | | | | || (_| |  ) ) ) )
  '  |____| .__|_| |_|_| |_\__, | / / / /
 =========|_|==============|___/=/_/_/_/
 :: Spring Boot ::        (v2.1.6.RELEASE)

2019-10-02 11:45:02.425  INFO 1 --- [           main] c.b.hellojib.HellojibApplication         : Starting HellojibApplication on d2d856d3c623 with PID 1 (/app/classes started by root in /)
2019-10-02 11:45:02.440  INFO 1 --- [           main] c.b.hellojib.HellojibApplication         : No active profile set, falling back to default profiles: default
2019-10-02 11:45:07.203  INFO 1 --- [           main] o.s.b.w.embedded.tomcat.TomcatWebServer  : Tomcat initialized with port(s): 8080 (http)
2019-10-02 11:45:07.342  INFO 1 --- [           main] o.apache.catalina.core.StandardService   : Starting service [Tomcat]
2019-10-02 11:45:07.344  INFO 1 --- [           main] org.apache.catalina.core.StandardEngine  : Starting Servlet engine: [Apache Tomcat/9.0.21]
2019-10-02 11:45:07.761  INFO 1 --- [           main] o.a.c.c.C.[Tomcat].[localhost].[/]       : Initializing Spring embedded WebApplicationContext
2019-10-02 11:45:07.762  INFO 1 --- [           main] o.s.web.context.ContextLoader            : Root WebApplicationContext: initialization completed in 5108 ms
2019-10-02 11:45:08.863  INFO 1 --- [           main] o.s.s.concurrent.ThreadPoolTaskExecutor  : Initializing ExecutorService 'applicationTaskExecutor'
2019-10-02 11:45:10.027  INFO 1 --- [           main] o.s.b.w.embedded.tomcat.TomcatWebServer  : Tomcat started on port(s): 8080 (http) with context path ''
2019-10-02 11:45:10.037  INFO 1 --- [           main] c.b.hellojib.HellojibApplication         : Started HellojibApplication in 8.932 seconds (JVM running for 9.876)
2019-10-02 11:45:52.965  INFO 1 --- [nio-8080-exec-1] o.a.c.c.C.[Tomcat].[localhost].[/]       : Initializing Spring DispatcherServlet 'dispatcherServlet'
2019-10-02 11:45:52.966  INFO 1 --- [nio-8080-exec-1] o.s.web.servlet.DispatcherServlet        : Initializing Servlet 'dispatcherServlet'
2019-10-02 11:45:52.993  INFO 1 --- [nio-8080-exec-1] o.s.web.servlet.DispatcherServlet        : Completed initialization in 27 ms
  1. ARM電腦的IP地址是<font color="blue">192.168.50.118</font>,因此在瀏覽器訪問:http://192.168.50.118:8080/hello 腾务,如下圖毕骡,可見hellojib工程的容器可以正常工作,成功返回了數(shù)據(jù):
    在這里插入圖片描述

    SpringBoot工程終于在ARM機器的Docker環(huán)境下成功運行了岩瘦,這里采用的是自制JDK8鏡像的方式未巫,還有一種方法也是可行的,即:使用OpenJDK官方的JDK11鏡像启昧;

使用OpenJDK官方的JDK11鏡像

使用JDK11鏡像叙凡,意味著Java工程所用的JDK從8升級到11,這個操作和Docker的關(guān)系不大密末,您只要驗證應用在升級JDK后是否能運行正常即可握爷,本文就不贅述了,我把自己在升級過程中遇到的問題列出來严里,幫您跳過小坑:

  1. 從JDK9開始引入了module的概念饼拍,JDK8自帶的一些jar包不再默認提供,您需要在應用的pom.xml中添加以下依賴田炭,否則SpringBoot啟動時會因為某些lass找不到導致啟動失斒Τ:
<dependency>
    <groupId>javax.xml.bind</groupId>
    <artifactId>jaxb-api</artifactId>
    <version>2.3.1</version>
</dependency>
<dependency>
    <groupId>com.sun.xml.bind</groupId>
    <artifactId>jaxb-impl</artifactId>
    <version>2.3.1</version>
</dependency>
<dependency>
    <groupId>org.glassfish.jaxb</groupId>
    <artifactId>jaxb-runtime</artifactId>
    <version>2.3.1</version>
</dependency>
  1. 如果您的不想修改pom.xml,此時我的做法:
    a. 將上述三個依賴對應的jar包全部找出來(注意這里絕不止三個jar包教硫,還有它們的間接依賴)叨吮,放在ARM電腦的某個文件夾下面,例如/usr/local/extendJar;
    b. 修改Jib插件的配置瞬矩,增加一個classpath茶鉴,例如:/usr/local/extendJar(注意這里的路徑是容器內(nèi)的);
    c. 在啟動容器的時候景用,增加一個數(shù)據(jù)卷映射涵叮,將宿主機的/usr/local/extendJar映射到容器的/usr/local/extendJar;

至此伞插,OpenJDK官方在ARM架構(gòu)不提供8版本鏡像的問題已完美解決割粮,如果您正在使用ARM服務器做Docker+Java開發(fā),希望此文能給您一些參考媚污。

https://github.com/zq2599/blog_demos

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末舀瓢,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子耗美,更是在濱河造成了極大的恐慌京髓,老刑警劉巖航缀,帶你破解...
    沈念sama閱讀 218,451評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異堰怨,居然都是意外死亡芥玉,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,172評論 3 394
  • 文/潘曉璐 我一進店門备图,熙熙樓的掌柜王于貴愁眉苦臉地迎上來飞傀,“玉大人,你說我怎么就攤上這事诬烹≡曳常” “怎么了?”我有些...
    開封第一講書人閱讀 164,782評論 0 354
  • 文/不壞的土叔 我叫張陵绞吁,是天一觀的道長幢痘。 經(jīng)常有香客問我,道長家破,這世上最難降的妖魔是什么颜说? 我笑而不...
    開封第一講書人閱讀 58,709評論 1 294
  • 正文 為了忘掉前任,我火速辦了婚禮汰聋,結(jié)果婚禮上门粪,老公的妹妹穿的比我還像新娘。我一直安慰自己烹困,他們只是感情好玄妈,可當我...
    茶點故事閱讀 67,733評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著髓梅,像睡著了一般拟蜻。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上枯饿,一...
    開封第一講書人閱讀 51,578評論 1 305
  • 那天酝锅,我揣著相機與錄音,去河邊找鬼奢方。 笑死搔扁,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的蟋字。 我是一名探鬼主播稿蹲,決...
    沈念sama閱讀 40,320評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼愉老!你這毒婦竟也來了场绿?” 一聲冷哼從身側(cè)響起剖效,我...
    開封第一講書人閱讀 39,241評論 0 276
  • 序言:老撾萬榮一對情侶失蹤嫉入,失蹤者是張志新(化名)和其女友劉穎焰盗,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體咒林,經(jīng)...
    沈念sama閱讀 45,686評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡熬拒,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,878評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了垫竞。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片澎粟。...
    茶點故事閱讀 39,992評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖欢瞪,靈堂內(nèi)的尸體忽然破棺而出活烙,到底是詐尸還是另有隱情,我是刑警寧澤遣鼓,帶...
    沈念sama閱讀 35,715評論 5 346
  • 正文 年R本政府宣布啸盏,位于F島的核電站,受9級特大地震影響骑祟,放射性物質(zhì)發(fā)生泄漏回懦。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,336評論 3 330
  • 文/蒙蒙 一次企、第九天 我趴在偏房一處隱蔽的房頂上張望怯晕。 院中可真熱鬧,春花似錦缸棵、人聲如沸舟茶。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,912評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽稚晚。三九已至,卻和暖如春型诚,著一層夾襖步出監(jiān)牢的瞬間客燕,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,040評論 1 270
  • 我被黑心中介騙來泰國打工狰贯, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留也搓,地道東北人。 一個月前我還...
    沈念sama閱讀 48,173評論 3 370
  • 正文 我出身青樓涵紊,卻偏偏與公主長得像傍妒,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子摸柄,可洞房花燭夜當晚...
    茶點故事閱讀 44,947評論 2 355