Spring Boot 2.x基礎(chǔ)教程:配置文件詳解

快速入門一節(jié)中,我們輕松的實現(xiàn)了一個簡單的RESTful API應(yīng)用品抽,體驗了一下Spring Boot給我們帶來的諸多優(yōu)點储笑,我們用非常少的代碼量就成功的實現(xiàn)了一個Web應(yīng)用,這是傳統(tǒng)的Spring應(yīng)用無法辦到的圆恤,雖然我們在實現(xiàn)Controller時用到的代碼是一樣的突倍,但是在配置方面,相信大家也注意到了盆昙,在上面的例子中羽历,除了Maven的配置之后,就沒有引入任何的配置淡喜。

這就是之前我們所提到的秕磷,Spring Boot針對我們常用的開發(fā)場景提供了一系列自動化配置來減少原本復(fù)雜而又幾乎很少改動的模板化配置內(nèi)容。但是炼团,我們還是需要去了解如何在Spring Boot中修改這些自動化的配置內(nèi)容澎嚣,以應(yīng)對一些特殊的場景需求疏尿,比如:我們在同一臺主機上需要啟動多個基于Spring Boot的web應(yīng)用,若我們不為每個應(yīng)用指定特別的端口號易桃,那么默認的8080端口必將導(dǎo)致沖突褥琐。

如果您還有在讀我的Spring Cloud系列教程,其實有大量的工作都會是針對配置文件的晤郑。所以我們有必要深入的了解一些關(guān)于Spring Boot中的配置文件的知識敌呈,比如:它的配置方式、如何實現(xiàn)多環(huán)境配置贩汉,配置信息的加載順序等驱富。

配置基礎(chǔ)

快速入門示例中,我們介紹Spring Boot的工程結(jié)構(gòu)時匹舞,有提到過 src/main/resources目錄是Spring Boot的配置目錄褐鸥,所以我們要為應(yīng)用創(chuàng)建配置個性化配置時,就是在該目錄之下赐稽。

Spring Boot的默認配置文件位置為: src/main/resources/application.properties叫榕。關(guān)于Spring Boot應(yīng)用的配置內(nèi)容都可以集中在該文件中了,根據(jù)我們引入的不同Starter模塊姊舵,可以在這里定義諸如:容器端口名晰绎、數(shù)據(jù)庫鏈接信息、日志級別等各種配置信息括丁。比如荞下,我們需要自定義web模塊的服務(wù)端口號,可以在application.properties中添加server.port=8888來指定服務(wù)端口為8888史飞,也可以通過spring.application.name=hello來指定應(yīng)用名(該名字在Spring Cloud應(yīng)用中會被注冊為服務(wù)名)尖昏。

Spring Boot的配置文件除了可以使用傳統(tǒng)的properties文件之外笋颤,還支持現(xiàn)在被廣泛推薦使用的YAML文件室琢。

YAML(英語發(fā)音:/?j?m?l/,尾音類似camel駱駝)是一個可讀性高妒牙,用來表達資料序列的格式吐绵。YAML參考了其他多種語言迹淌,包括:C語言、Python己单、Perl唉窃,并從XML、電子郵件的數(shù)據(jù)格式(RFC 2822)中獲得靈感荷鼠。Clark Evans在2001年首次發(fā)表了這種語言句携,另外Ingy d?t Net與Oren Ben-Kiki也是這語言的共同設(shè)計者。目前已經(jīng)有數(shù)種編程語言或腳本語言支援(或者說解析)這種語言允乐。YAML是"YAML Ain't a Markup Language"(YAML不是一種標記語言)的遞回縮寫矮嫉。在開發(fā)的這種語言時削咆,YAML 的意思其實是:"Yet Another Markup Language"(仍是一種標記語言),但為了強調(diào)這種語言以數(shù)據(jù)做為中心蠢笋,而不是以標記語言為重點拨齐,而用反向縮略語重新命名。AML的語法和其他高階語言類似昨寞,并且可以簡單表達清單瞻惋、散列表,標量等資料形態(tài)援岩。它使用空白符號縮排和大量依賴外觀的特色歼狼,特別適合用來表達或編輯數(shù)據(jù)結(jié)構(gòu)、各種設(shè)定檔享怀、傾印除錯內(nèi)容羽峰、文件大綱(例如:許多電子郵件標題格式和YAML非常接近)。盡管它比較適合用來表達階層式(hierarchical model)的數(shù)據(jù)結(jié)構(gòu)添瓷,不過也有精致的語法可以表示關(guān)聯(lián)性(relational model)的資料梅屉。由于YAML使用空白字元和分行來分隔資料,使得它特別適合用grep/Python/Perl/Ruby操作鳞贷。其讓人最容易上手的特色是巧妙避開各種封閉符號坯汤,如:引號、各種括號等搀愧,這些符號在巢狀結(jié)構(gòu)時會變得復(fù)雜而難以辨認惰聂。 —— 維基百科

YAML采用的配置格式不像properties的配置那樣以單純的鍵值對形式來表示,而是以類似大綱的縮進形式來表示咱筛。比如:下面的一段YAML配置信息

environments:
    dev:
        url: http://dev.bar.com
        name: Developer Setup
    prod:
        url: http://foo.bar.com
        name: My Cool App

與其等價的properties配置如下庶近。

environments.dev.url=http://dev.bar.com
environments.dev.name=Developer Setup
environments.prod.url=http://foo.bar.com
environments.prod.name=My Cool App

通過YAML的配置方式,我們可以看到配置信息利用階梯化縮進的方式眷蚓,其結(jié)構(gòu)顯得更為清晰易讀,同時配置內(nèi)容的字符量也得到顯著的減少反番。除此之外沙热,YAML還可以在一個單個文件中通過使用spring.profiles屬性來定義多個不同的環(huán)境配置。例如下面的內(nèi)容罢缸,在指定為test環(huán)境時篙贸,server.port將使用8882端口;而在prod環(huán)境枫疆,server.port將使用8883端口爵川;如果沒有指定環(huán)境,server.port將使用8881端口息楔。

server:
    port: 8881
---
spring:
    profiles: test
server:
    port: 8882
---
spring:
    profiles: prod
server:
    port: 8883

注意:YAML目前還有一些不足寝贡,它無法通過@PropertySource注解來加載配置扒披。但是,YAML加載屬性到內(nèi)存中保存的時候是有序的圃泡,所以當配置文件中的信息需要具備順序含義時碟案,YAML的配置方式比起properties配置文件更有優(yōu)勢。

自定義參數(shù)

我們除了可以在Spring Boot的配置文件中設(shè)置各個Starter模塊中預(yù)定義的配置屬性颇蜡,也可以在配置文件中定義一些我們需要的自定義屬性价说。比如在application.properties中添加:

book.name=SpringCloudInAction
book.author=ZhaiYongchao

然后,在應(yīng)用中我們可以通過@Value注解來加載這些自定義的參數(shù)风秤,比如:

@Component
public class Book {

    @Value("${book.name}")
    private String name;
    @Value("${book.author}")
    private String author;

    // 省略getter和setter
}

@Value注解加載屬性值的時候可以支持兩種表達式來進行配置:

  • 一種是我們上面介紹的PlaceHolder方式鳖目,格式為 ${...},大括號內(nèi)為PlaceHolder
  • 另外還可以使用SpEL表達式(Spring Expression Language)缤弦, 格式為 #{...}领迈,大括號內(nèi)為SpEL表達式

參數(shù)引用

application.properties中的各個參數(shù)之間,我們也可以直接通過使用PlaceHolder的方式來進行引用甸鸟,就像下面的設(shè)置:

book.name=SpringCloud
book.author=ZhaiYongchao
book.desc=${book.author}  is writing《${book.name}》

book.desc參數(shù)引用了上文中定義的book.namebook.author屬性惦费,最后該屬性的值就是ZhaiYongchao is writing《SpringCloud》

使用隨機數(shù)

在一些特殊情況下抢韭,有些參數(shù)我們希望它每次加載的時候不是一個固定的值薪贫,比如:密鑰、服務(wù)端口等刻恭。在Spring Boot的屬性配置文件中瞧省,我們可以通過使用${random}配置來產(chǎn)生隨機的int值、long值或者string字符串鳍贾,這樣我們就可以容易的通過配置來屬性的隨機生成鞍匾,而不是在程序中通過編碼來實現(xiàn)這些邏輯。

${random}的配置方式主要有一下幾種骑科,讀者可作為參考使用橡淑。

# 隨機字符串
com.didispace.blog.value=${random.value}
# 隨機int
com.didispace.blog.number=${random.int}
# 隨機long
com.didispace.blog.bignumber=${random.long}
# 10以內(nèi)的隨機數(shù)
com.didispace.blog.test1=${random.int(10)}
# 10-20的隨機數(shù)
com.didispace.blog.test2=${random.int[10,20]}

該配置方式可以用于設(shè)置應(yīng)用端口等場景,避免在本地調(diào)試時出現(xiàn)端口沖突的麻煩

命令行參數(shù)

回顧一下在本章的快速入門中咆爽,我們還介紹了如何啟動Spring Boot應(yīng)用梁棠,其中提到了使用命令java -jar命令來啟動的方式。該命令除了啟動應(yīng)用之外斗埂,還可以在命令行中來指定應(yīng)用的參數(shù)符糊,比如:java -jar xxx.jar --server.port=8888,直接以命令行的方式呛凶,來設(shè)置server.port屬性男娄,另啟動應(yīng)用的端口設(shè)為8888。

在命令行方式啟動Spring Boot應(yīng)用時,連續(xù)的兩個減號--就是對application.properties中的屬性值進行賦值的標識模闲。所以建瘫,java -jar xxx.jar --server.port=8888命令,等價于我們在application.properties中添加屬性server.port=8888围橡。

通過命令行來修改屬性值是Spring Boot非常重要的一個特性暖混,通過此特性,理論上已經(jīng)使得我們應(yīng)用的屬性在啟動前是可變的翁授,所以其中端口號也好拣播、數(shù)據(jù)庫連接也好,都是可以在應(yīng)用啟動時發(fā)生改變收擦,而不同于以往的Spring應(yīng)用通過Maven的Profile在編譯器進行不同環(huán)境的構(gòu)建贮配。其最大的區(qū)別就是,Spring Boot的這種方式塞赂,可以讓應(yīng)用程序的打包內(nèi)容泪勒,貫穿開發(fā)、測試以及線上部署宴猾,而Maven不同Profile的方案每個環(huán)境所構(gòu)建的包圆存,其內(nèi)容本質(zhì)上是不同的。但是仇哆,如果每個參數(shù)都需要通過命令行來指定沦辙,這顯然也不是一個好的方案,所以下面我們看看如果在Spring Boot中實現(xiàn)多環(huán)境的配置讹剔。

多環(huán)境配置

我們在開發(fā)任何應(yīng)用的時候油讯,通常同一套程序會被應(yīng)用和安裝到幾個不同的環(huán)境,比如:開發(fā)延欠、測試陌兑、生產(chǎn)等。其中每個環(huán)境的數(shù)據(jù)庫地址由捎、服務(wù)器端口等等配置都會不同兔综,如果在為不同環(huán)境打包時都要頻繁修改配置文件的話,那必將是個非常繁瑣且容易發(fā)生錯誤的事狞玛。

對于多環(huán)境的配置邻奠,各種項目構(gòu)建工具或是框架的基本思路是一致的,通過配置多份不同環(huán)境的配置文件为居,再通過打包命令指定需要打包的內(nèi)容之后進行區(qū)分打包,Spring Boot也不例外杀狡,或者說更加簡單蒙畴。

在Spring Boot中多環(huán)境配置文件名需要滿足application-{profile}.properties的格式,其中{profile}對應(yīng)你的環(huán)境標識,比如:

  • application-dev.properties:開發(fā)環(huán)境
  • application-test.properties:測試環(huán)境
  • application-prod.properties:生產(chǎn)環(huán)境

至于哪個具體的配置文件會被加載膳凝,需要在application.properties文件中通過spring.profiles.active屬性來設(shè)置碑隆,其值對應(yīng)配置文件中的{profile}值。如:spring.profiles.active=test就會加載application-test.properties配置文件內(nèi)容蹬音。

下面上煤,以不同環(huán)境配置不同的服務(wù)端口為例,進行樣例實驗著淆。

  • 針對各環(huán)境新建不同的配置文件application-dev.properties劫狠、application-test.propertiesapplication-prod.properties

  • 在這三個文件均都設(shè)置不同的server.port屬性永部,如:dev環(huán)境設(shè)置為1111独泞,test環(huán)境設(shè)置為2222,prod環(huán)境設(shè)置為3333

  • application.properties中設(shè)置spring.profiles.active=dev苔埋,就是說默認以dev環(huán)境設(shè)置

  • 測試不同配置的加載

  • 執(zhí)行java -jar xxx.jar懦砂,可以觀察到服務(wù)端口被設(shè)置為1111,也就是默認的開發(fā)環(huán)境(dev)

  • 執(zhí)行java -jar xxx.jar --spring.profiles.active=test组橄,可以觀察到服務(wù)端口被設(shè)置為2222荞膘,也就是測試環(huán)境的配置(test)

  • 執(zhí)行java -jar xxx.jar --spring.profiles.active=prod,可以觀察到服務(wù)端口被設(shè)置為3333玉工,也就是生產(chǎn)環(huán)境的配置(prod)

按照上面的實驗羽资,可以如下總結(jié)多環(huán)境的配置思路:

  • application.properties中配置通用內(nèi)容,并設(shè)置spring.profiles.active=dev瓮栗,以開發(fā)環(huán)境為默認配置
  • application-{profile}.properties中配置各個環(huán)境不同的內(nèi)容
  • 通過命令行方式去激活不同環(huán)境的配置

加載順序

在上面的例子中削罩,我們將Spring Boot應(yīng)用需要的配置內(nèi)容都放在了項目工程中,雖然我們已經(jīng)能夠通過spring.profiles.active或是通過Maven來實現(xiàn)多環(huán)境的支持费奸。但是弥激,當我們的團隊逐漸壯大,分工越來越細致之后愿阐,往往我們不需要讓開發(fā)人員知道測試或是生成環(huán)境的細節(jié)微服,而是希望由每個環(huán)境各自的負責人(QA或是運維)來集中維護這些信息。那么如果還是以這樣的方式存儲配置內(nèi)容缨历,對于不同環(huán)境配置的修改就不得不去獲取工程內(nèi)容來修改這些配置內(nèi)容以蕴,當應(yīng)用非常多的時候就變得非常不方便。同時辛孵,配置內(nèi)容都對開發(fā)人員可見丛肮,本身這也是一種安全隱患。對此魄缚,現(xiàn)在出現(xiàn)了很多將配置內(nèi)容外部化的框架和工具宝与,后續(xù)將要介紹的Spring Cloud Config就是其中之一焚廊,為了后續(xù)能更好的理解Spring Cloud Config的加載機制,我們需要對Spring Boot對數(shù)據(jù)文件的加載機制有一定的了解习劫。

Spring Boot為了能夠更合理的重寫各屬性的值咆瘟,使用了下面這種較為特別的屬性加載順序:

  1. 命令行中傳入的參數(shù)。
  2. SPRING_APPLICATION_JSON中的屬性诽里。SPRING_APPLICATION_JSON是以JSON格式配置在系統(tǒng)環(huán)境變量中的內(nèi)容袒餐。
  3. java:comp/env中的JNDI屬性。
  4. Java的系統(tǒng)屬性谤狡,可以通過System.getProperties()獲得的內(nèi)容灸眼。
  5. 操作系統(tǒng)的環(huán)境變量
  6. 通過random.*配置的隨機屬性
  7. 位于當前應(yīng)用jar包之外,針對不同{profile}環(huán)境的配置文件內(nèi)容豌汇,例如:application-{profile}.properties或是YAML定義的配置文件
  8. 位于當前應(yīng)用jar包之內(nèi)幢炸,針對不同{profile}環(huán)境的配置文件內(nèi)容,例如:application-{profile}.properties或是YAML定義的配置文件
  9. 位于當前應(yīng)用jar包之外的application.propertiesYAML配置內(nèi)容
  10. 位于當前應(yīng)用jar包之內(nèi)的application.propertiesYAML配置內(nèi)容
  11. @Configuration注解修改的類中拒贱,通過@PropertySource注解定義的屬性
  12. 應(yīng)用默認屬性宛徊,使用SpringApplication.setDefaultProperties定義的內(nèi)容

優(yōu)先級按上面的順序有高到低,數(shù)字越小優(yōu)先級越高逻澳。

可以看到闸天,其中第7項和第9項都是從應(yīng)用jar包之外讀取配置文件,所以斜做,實現(xiàn)外部化配置的原理就是從此切入苞氮,為其指定外部配置文件的加載位置來取代jar包之內(nèi)的配置內(nèi)容。通過這樣的實現(xiàn)瓤逼,我們的工程在配置中就變的非常干凈笼吟,我們只需要在本地放置開發(fā)需要的配置即可,而其他環(huán)境的配置就可以不用關(guān)心霸旗,由其對應(yīng)環(huán)境的負責人去維護即可贷帮。

2.x 新特性

在Spring Boot 2.0中推出了Relaxed Binding 2.0,對原有的屬性綁定功能做了非常多的改進以幫助我們更容易的在Spring應(yīng)用中加載和讀取配置信息诱告。下面本文就來說說Spring Boot 2.0中對配置的改進撵枢。

配置文件綁定

簡單類型

在Spring Boot 2.0中對配置屬性加載的時候會除了像1.x版本時候那樣移除特殊字符外,還會將配置均以全小寫的方式進行匹配和加載精居。所以锄禽,下面的4種配置方式都是等價的:

  • properties格式:
spring.jpa.databaseplatform=mysql
spring.jpa.database-platform=mysql
spring.jpa.databasePlatform=mysql
spring.JPA.database_platform=mysql
  • yaml格式:
spring:
  jpa:
    databaseplatform: mysql
    database-platform: mysql
    databasePlatform: mysql
    database_platform: mysql

Tips:推薦使用全小寫配合-分隔符的方式來配置,比如:spring.jpa.database-platform=mysql

List類型

在properties文件中使用[]來定位列表類型靴姿,比如:

spring.my-example.url[0]=http://example.com
spring.my-example.url[1]=http://spring.io

也支持使用逗號分割的配置方式沃但,上面與下面的配置是等價的:

spring.my-example.url=http://example.com,http://spring.io

而在yaml文件中使用可以使用如下配置:

spring:
  my-example:
    url:
      - http://example.com
      - http://spring.io

也支持逗號分割的方式:

spring:
  my-example:
    url: http://example.com, http://spring.io

注意:在Spring Boot 2.0中對于List類型的配置必須是連續(xù)的,不然會拋出UnboundConfigurationPropertiesException異常佛吓,所以如下配置是不允許的:

foo[0]=a
foo[2]=b

在Spring Boot 1.x中上述配置是可以的宵晚,foo[1]由于沒有配置恨旱,它的值會是null

Map類型

Map類型在properties和yaml中的標準配置方式如下:

  • properties格式:
spring.my-example.foo=bar
spring.my-example.hello=world
  • yaml格式:
spring:
  my-example:
    foo: bar
    hello: world

注意:如果Map類型的key包含非字母數(shù)字和-的字符,需要用[]括起來坝疼,比如:

spring:
  my-example:
    '[foo.baz]': bar

環(huán)境屬性綁定

簡單類型

在環(huán)境變量中通過小寫轉(zhuǎn)換與.替換_來映射配置文件中的內(nèi)容,比如:環(huán)境變量SPRING_JPA_DATABASEPLATFORM=mysql的配置會產(chǎn)生與在配置文件中設(shè)置spring.jpa.databaseplatform=mysql一樣的效果谆沃。

List類型

由于環(huán)境變量中無法使用[]符號钝凶,所以使用_來替代。任何由下劃線包圍的數(shù)字都會被認為是[]的數(shù)組形式唁影。比如:

MY_FOO_1_ = my.foo[1]
MY_FOO_1_BAR = my.foo[1].bar
MY_FOO_1_2_ = my.foo[1][2]

另外耕陷,最后環(huán)境變量最后是以數(shù)字和下劃線結(jié)尾的話,最后的下劃線可以省略据沈,比如上面例子中的第一條和第三條等價于下面的配置:

MY_FOO_1 = my.foo[1]
MY_FOO_1_2 = my.foo[1][2]

系統(tǒng)屬性綁定

簡單類型

系統(tǒng)屬性與文件配置中的類似哟沫,都以移除特殊字符并轉(zhuǎn)化小寫后實現(xiàn)綁定,比如下面的命令行參數(shù)都會實現(xiàn)配置spring.jpa.databaseplatform=mysql的效果:

-Dspring.jpa.database-platform=mysql
-Dspring.jpa.databasePlatform=mysql
-Dspring.JPA.database_platform=mysql

List類型

系統(tǒng)屬性的綁定也與文件屬性的綁定類似锌介,通過[]來標示嗜诀,比如:

-D"spring.my-example.url[0]=http://example.com"
-D"spring.my-example.url[1]=http://spring.io"

同樣的,他也支持逗號分割的方式孔祸,比如:

-Dspring.my-example.url=http://example.com,http://spring.io

屬性的讀取

上文介紹了Spring Boot 2.0中對屬性綁定的內(nèi)容隆敢,可以看到對于一個屬性我們可以有多種不同的表達,但是如果我們要在Spring應(yīng)用程序的environment中讀取屬性的時候崔慧,每個屬性的唯一名稱符合如下規(guī)則:

  • 通過.分離各個元素
  • 最后一個.將前綴與屬性名稱分開
  • 必須是字母(a-z)和數(shù)字(0-9)
  • 必須是小寫字母
  • 用連字符-來分隔單詞
  • 唯一允許的其他字符是[]拂蝎,用于List的索引
  • 不能以數(shù)字開頭

所以,如果我們要讀取配置文件中spring.jpa.database-platform的配置惶室,可以這樣寫:

this.environment.containsProperty("spring.jpa.database-platform")

而下面的方式是無法獲取到spring.jpa.database-platform配置內(nèi)容的:

this.environment.containsProperty("spring.jpa.databasePlatform")

注意:使用@Value獲取配置內(nèi)容的時候也需要這樣的特點

全新的綁定API

在Spring Boot 2.0中增加了新的綁定API來幫助我們更容易的獲取配置信息温自。下面舉個例子來幫助大家更容易的理解:

例子一:簡單類型

假設(shè)在propertes配置中有這樣一個配置:com.didispace.foo=bar

我們?yōu)樗鼊?chuàng)建對應(yīng)的配置類:

@Data
@ConfigurationProperties(prefix = "com.didispace")
public class FooProperties {

    private String foo;

}

接下來,通過最新的Binder就可以這樣來拿配置信息了:

@SpringBootApplication
public class Application {

    public static void main(String[] args) {
        ApplicationContext context = SpringApplication.run(Application.class, args);

        Binder binder = Binder.get(context.getEnvironment());

        // 綁定簡單配置
        FooProperties foo = binder.bind("com.didispace", Bindable.of(FooProperties.class)).get();
        System.out.println(foo.getFoo());
    }
}

例子二:List類型

如果配置內(nèi)容是List類型呢皇钞?比如:

com.didispace.post[0]=Why Spring Boot
com.didispace.post[1]=Why Spring Cloud

com.didispace.posts[0].title=Why Spring Boot
com.didispace.posts[0].content=It is perfect!
com.didispace.posts[1].title=Why Spring Cloud
com.didispace.posts[1].content=It is perfect too!

要獲取這些配置依然很簡單悼泌,可以這樣實現(xiàn):

ApplicationContext context = SpringApplication.run(Application.class, args);

Binder binder = Binder.get(context.getEnvironment());

// 綁定List配置
List<String> post = binder.bind("com.didispace.post", Bindable.listOf(String.class)).get();
System.out.println(post);

List<PostInfo> posts = binder.bind("com.didispace.posts", Bindable.listOf(PostInfo.class)).get();
System.out.println(posts);

代碼示例

本教程配套倉庫:

如果您覺得本文不錯,歡迎Star鹅士、Follow支持券躁!您的關(guān)注是我堅持的動力!

相關(guān)閱讀

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末掉盅,一起剝皮案震驚了整個濱河市也拜,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌趾痘,老刑警劉巖慢哈,帶你破解...
    沈念sama閱讀 219,539評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異永票,居然都是意外死亡卵贱,警方通過查閱死者的電腦和手機滥沫,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,594評論 3 396
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來键俱,“玉大人兰绣,你說我怎么就攤上這事”嗾瘢” “怎么了缀辩?”我有些...
    開封第一講書人閱讀 165,871評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長踪央。 經(jīng)常有香客問我臀玄,道長,這世上最難降的妖魔是什么畅蹂? 我笑而不...
    開封第一講書人閱讀 58,963評論 1 295
  • 正文 為了忘掉前任健无,我火速辦了婚禮,結(jié)果婚禮上液斜,老公的妹妹穿的比我還像新娘累贤。我一直安慰自己,他們只是感情好旗唁,可當我...
    茶點故事閱讀 67,984評論 6 393
  • 文/花漫 我一把揭開白布畦浓。 她就那樣靜靜地躺著,像睡著了一般检疫。 火紅的嫁衣襯著肌膚如雪讶请。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,763評論 1 307
  • 那天屎媳,我揣著相機與錄音夺溢,去河邊找鬼。 笑死烛谊,一個胖子當著我的面吹牛风响,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播丹禀,決...
    沈念sama閱讀 40,468評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼状勤,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了双泪?” 一聲冷哼從身側(cè)響起持搜,我...
    開封第一講書人閱讀 39,357評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎焙矛,沒想到半個月后葫盼,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,850評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡村斟,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,002評論 3 338
  • 正文 我和宋清朗相戀三年贫导,在試婚紗的時候發(fā)現(xiàn)自己被綠了抛猫。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,144評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡孩灯,死狀恐怖闺金,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情峰档,我是刑警寧澤掖看,帶...
    沈念sama閱讀 35,823評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站面哥,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏毅待。R本人自食惡果不足惜尚卫,卻給世界環(huán)境...
    茶點故事閱讀 41,483評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望尸红。 院中可真熱鬧吱涉,春花似錦、人聲如沸外里。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,026評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽盅蝗。三九已至鳖链,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間墩莫,已是汗流浹背芙委。 一陣腳步聲響...
    開封第一講書人閱讀 33,150評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留狂秦,地道東北人灌侣。 一個月前我還...
    沈念sama閱讀 48,415評論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像裂问,于是被迫代替她去往敵國和親侧啼。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,092評論 2 355

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