99%的程序員都在用Lombok景馁,原理竟然這么簡單?我也手擼了一個陵且!

以下文章來源于Java中文社群 裁僧,作者老王

世界上只有一種英雄主義,就是看清生活的真相之后依然熱愛生活慕购。

對于 Lombok 我相信大部分人都不陌生,但對于它的實現(xiàn)原理以及缺點卻鮮為人知茬底,而本文將會從 Lombok 的原理出發(fā)沪悲,手擼一個簡易版的 Lombok,讓你理解這個熱門技術(shù)背后的執(zhí)行原理阱表,以及它的優(yōu)缺點殿如。

簡介

在講原理之前,我們先來復習一下 Lombok (老司機可直接跳過本段)最爬。

Lombok 是一個非常熱門的開源項目 (https://github.com/rzwitserloot/lombok)涉馁,使用它可以有效的解決 Java 工程中那些繁瑣又重復代碼,例如 Setter爱致、Getter烤送、toString、equals糠悯、hashCode 以及非空判斷等帮坚,都可以使用 Lombok 有效的解決。

使用

1.添加 Lombok 插件

在 IDE 中必須安裝 Lombok 插件互艾,才能正常調(diào)用被 Lombok 修飾的代碼试和,以 Idea 為例,添加的步驟如下:

  • 點擊 File > Settings > Plugins 進入插件管理頁面
  • 點擊 Browse repositories...
  • 搜索 Lombok Plugin
  • 點擊 Install plugin 安裝插件
  • 重啟 IntelliJ IDEA

安裝完成纫普,如下圖所示:
image

2.添加 Lombok 庫

接下來我們需要在項目中添加最新的 Lombok 庫阅悍,如果是 Maven 項目,直接在 pom.xml 中添加如下配置:

  <!-- https://mvnrepository.com/artifact/org.projectlombok/lombok -->
    <dependency>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
        <version>1.18.12</version>
        <scope>provided</scope>
    </dependency>
</dependencies>

如果是 JDK 9+ 可使用模塊的方式添加,配置如下:

    <path>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
        <version>1.18.12</version>
    </path>
</annotationProcessorPaths>

3.使用 Lombok

接下來到了前半部分中最重要的 Lombok 使用環(huán)節(jié)了节视,我們先來看在沒有使用 Lombok 之前的代碼:

public class Person {
    private Integer id;
    private String name;
    public Integer getId() {
        return id;
    }
    public void setId(Integer id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
}

這是使用 Lombok 之后的代碼:

@Getter
@Setter
public class Person {
    private Integer id;
    private String name;
}

可以看出在 Lombok 之后晦墙,用一個注解就搞定了之前所有 Getter/Setter 的代碼,讓代碼瞬間優(yōu)雅了很多肴茄。

Lombok 所有注解如下:

  • val:用在局部變量前面晌畅,相當于將變量聲明為 final;

  • @NonNull:給方法參數(shù)增加這個注解會自動在方法內(nèi)對該參數(shù)進行是否為空的校驗寡痰,如果為空抗楔,則拋出 NPE(NullPointerException);

  • @Cleanup:自動管理資源拦坠,用在局部變量之前连躏,在當前變量范圍內(nèi)即將執(zhí)行完畢退出之前會自動清理資源,自動生成 try-finally 這樣的代碼來關(guān)閉流贞滨;

  • @Getter/@Setter:用在屬性上入热,再也不用自己手寫 setter 和 getter 方法了,還可以指定訪問范圍晓铆;

  • @ToString:用在類上可以自動覆寫 toString 方法勺良,當然還可以加其他參數(shù),例如 @ToString(exclude=”id”) 排除 id 屬性骄噪,或者 @ToString(callSuper=true, includeFieldNames=true) 調(diào)用父類的 toString 方法尚困,包含所有屬性;

  • @EqualsAndHashCode:用在類上自動生成 equals 方法和 hashCode 方法链蕊;

  • @NoArgsConstructor, @RequiredArgsConstructor and @AllArgsConstructor:用在類上事甜,自動生成無參構(gòu)造和使用所有參數(shù)的構(gòu)造函數(shù)以及把所有 @NonNull 屬性作為參數(shù)的構(gòu)造函數(shù),如果指定 staticName="of" 參數(shù)滔韵,同時還會生成一個返回類對象的靜態(tài)工廠方法逻谦,比使用構(gòu)造函數(shù)方便很多;

  • @Data:注解在類上陪蜻,相當于同時使用了 @ToString邦马、@EqualsAndHashCode、@Getter囱皿、@Setter 和 @RequiredArgsConstrutor 這些注解勇婴,對于 POJO 類十分有用;

  • @Value:用在類上嘱腥,是 @Data 的不可變形式耕渴,相當于為屬性添加 final 聲明,只提供 getter 方法齿兔,而不提供 setter 方法橱脸;

  • @Builder:用在類础米、構(gòu)造器、方法上添诉,為你提供復雜的 builder APIs屁桑,讓你可以像如下方式一樣調(diào)用Person.builder().name("xxx").city("xxx").build();

  • @SneakyThrows:自動拋受檢異常栏赴,而無需顯式在方法上使用 throws 語句蘑斧;

  • @Synchronized:用在方法上,將方法聲明為同步的须眷,并自動加鎖竖瘾,而鎖對象是一個私有的屬性lock或者LOCK,而 Java 中的 synchronized 關(guān)鍵字鎖對象是 this花颗,鎖在 this 或者自己的類對象上存在副作用捕传,就是你不能阻止非受控代碼去鎖 this 或者類對象,這可能會導致競爭條件或者其它線程錯誤扩劝;

  • @Getter(lazy=true):可以替代經(jīng)典的 Double Check Lock 樣板代碼庸论;

  • @Log:根據(jù)不同的注解生成不同類型的 log 對象,但是實例名稱都是 log棒呛,有六種可選實現(xiàn)類

  • @CommonsLog Creates log = org.apache.commons.logging.LogFactory.getLog(LogExample.class);

  • @Log Creates log = java.util.logging.Logger.getLogger(LogExample.class.getName());

  • @Log4j Creates log = org.apache.log4j.Logger.getLogger(LogExample.class);

  • @Log4j2 Creates log = org.apache.logging.log4j.LogManager.getLogger(LogExample.class);

  • @Slf4j Creates log = org.slf4j.LoggerFactory.getLogger(LogExample.class);

  • @XSlf4j Creates log = org.slf4j.ext.XLoggerFactory.getXLogger(LogExample.class);

它們的具體使用如下:

① val 使用

val sets = new HashSet<String>();
// 相當于
final Set<String> sets = new HashSet<>();

② NonNull 使用

 public void notNullExample(@NonNull String string) {
    string.length();
}
// 相當于
public void notNullExample(String string) {
    if (string != null) {
        string.length();
    } else {
        throw new NullPointerException("null");
    }
}

③ Cleanup 使用

public static void main(String[] args) {
    try {
        @Cleanup InputStream inputStream = new FileInputStream(args[0]);
    } catch (FileNotFoundException e) {
        e.printStackTrace();
    }
    // 相當于
    InputStream inputStream = null;
    try {
        inputStream = new FileInputStream(args[0]);
    } catch (FileNotFoundException e) {
        e.printStackTrace();
    } finally {
        if (inputStream != null) {
            try {
                inputStream.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}

④ Getter/Setter 使用

@Setter(AccessLevel.PUBLIC)
@Getter(AccessLevel.PROTECTED)
private int id;
private String shap;

⑤ ToString 使用

@ToString(exclude = "id", callSuper = true, includeFieldNames = true)
public class LombokDemo {
    private int id;
    private String name;
    private int age;
    public static void main(String[] args) {
        // 輸出 LombokDemo(super=LombokDemo@48524010, name=null, age=0)
        System.out.println(new LombokDemo());
    }
}

⑥ EqualsAndHashCode 使用

@EqualsAndHashCode(exclude = {"id", "shape"}, callSuper = false)
public class LombokDemo {
    private int id;
    private String shap;
}

⑦ NoArgsConstructor聂示、RequiredArgsConstructor、AllArgsConstructor 使用

@NoArgsConstructor
@RequiredArgsConstructor(staticName = "of")
@AllArgsConstructor
public class LombokDemo {
    @NonNull
    private int id;
    @NonNull
    private String shap;
    private int age;
    public static void main(String[] args) {
        new LombokDemo(1, "Java");
        // 使用靜態(tài)工廠方法
        LombokDemo.of(2, "Java");
        // 無參構(gòu)造
        new LombokDemo();
        // 包含所有參數(shù)
        new LombokDemo(1, "Java", 2);
    }
}

⑧ Builder 使用

@Builder
public class BuilderExample {
    private String name;
    private int age;
    @Singular
    private Set<String> occupations;
    public static void main(String[] args) {
        BuilderExample test = BuilderExample.builder().age(11).name("Java").build();
    }
}

⑨ SneakyThrows 使用

public class ThrowsTest {
    @SneakyThrows()
    public void read() {
        InputStream inputStream = new FileInputStream("");
    }
    @SneakyThrows
    public void write() {
        throw new UnsupportedEncodingException();
    }
    // 相當于
    public void read() throws FileNotFoundException {
        InputStream inputStream = new FileInputStream("");
    }
    public void write() throws UnsupportedEncodingException {
        throw new UnsupportedEncodingException();
    }
}

⑩ Synchronized 使用

public class SynchronizedDemo {
    @Synchronized
    public static void hello() {
        System.out.println("world");
    }
    // 相當于
    private static final Object $LOCK = new Object[0];
    public static void hello() {
        synchronized ($LOCK) {
            System.out.println("world");
        }
    }
}

? Getter(lazy = true) 使用

public class GetterLazyExample {
    @Getter(lazy = true)
    private final double[] cached = expensive();
    private double[] expensive() {
        double[] result = new double[1000000];
        for (int i = 0; i < result.length; i++) {
            result[i] = Math.asin(i);
        }
        return result;
    }
}
// 相當于
import java.util.concurrent.atomic.AtomicReference;
public class GetterLazyExample {
    private final AtomicReference<java.lang.Object> cached = new AtomicReference<>();
    public double[] getCached() {
        java.lang.Object value = this.cached.get();
        if (value == null) {
            synchronized (this.cached) {
                value = this.cached.get();
                if (value == null) {
                    final double[] actualValue = expensive();
                    value = actualValue == null ? this.cached : actualValue;
                    this.cached.set(value);
                }
            }
        }
        return (double[]) (value == this.cached ? null : value);
    }
    private double[] expensive() {
        double[] result = new double[1000000];
        for (int i = 0; i < result.length; i++) {
            result[i] = Math.asin(i);
        }
        return result;
    }
}

原理分析

我們知道 Java 的編譯過程大致可以分為三個階段:

  1. 解析與填充符號表
  2. 注解處理
  3. 分析與字節(jié)碼生成

編譯過程如下圖所示:
image

而 Lombok 正是利用「注解處理」這一步進行實現(xiàn)的条霜,Lombok 使用的是 JDK 6 實現(xiàn)的 JSR 269: Pluggable Annotation Processing API (編譯期的注解處理器) 催什,它是在編譯期時把 Lombok 的注解代碼,轉(zhuǎn)換為常規(guī)的 Java 方法而實現(xiàn)優(yōu)雅地編程的宰睡。

這一點可以在程序中得到驗證,比如本文剛開始用 @Data 實現(xiàn)的代碼:

image

在我們編譯之后气筋,查看 Person 類的編譯源碼發(fā)現(xiàn)拆内,代碼竟然是這樣的:

image

可以看出 Person 類在編譯期被注解翻譯器修改成了常規(guī)的 Java 方法,添加 Getter宠默、Setter麸恍、equals、hashCode 等方法搀矫。

Lombok 的執(zhí)行流程如下:

image

可以看出抹沪,在編譯期階段,當 Java 源碼被抽象成語法樹 (AST) 之后瓤球,Lombok 會根據(jù)自己的注解處理器動態(tài)的修改 AST融欧,增加新的代碼 (節(jié)點),在這一切執(zhí)行之后卦羡,再通過分析生成了最終的字節(jié)碼 (.class) 文件噪馏,這就是 Lombok 的執(zhí)行原理麦到。

手擼一個 Lombok

我們實現(xiàn)一個簡易版的 Lombok 自定義一個 Getter 方法,我們的實現(xiàn)步驟是:

  1. 自定義一個注解標簽接口欠肾,并實現(xiàn)一個自定義的注解處理器瓶颠;
  2. 利用 tools.jar 的 javac api 處理 AST (抽象語法樹)
  3. 使用自定義的注解處理器編譯代碼。

這樣就可以實現(xiàn)一個簡易版的 Lombok 了刺桃。

1.定義自定義注解和注解處理器

首先創(chuàng)建一個 MyGetter.java 自定義一個注解粹淋,代碼如下:

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.SOURCE) // 注解只在源碼中保留
@Target(ElementType.TYPE) // 用于修飾類
public @interface MyGetter { // 定義 Getter

}

再實現(xiàn)一個自定義的注解處理器,代碼如下:

import com.sun.source.tree.Tree;
import com.sun.tools.javac.api.JavacTrees;
import com.sun.tools.javac.code.Flags;
import com.sun.tools.javac.code.Type;
import com.sun.tools.javac.processing.JavacProcessingEnvironment;
import com.sun.tools.javac.tree.JCTree;
import com.sun.tools.javac.tree.TreeMaker;
import com.sun.tools.javac.tree.TreeTranslator;
import com.sun.tools.javac.util.*;

import javax.annotation.processing.*;
import javax.lang.model.SourceVersion;
import javax.lang.model.element.Element;
import javax.lang.model.element.TypeElement;
import javax.tools.Diagnostic;
import java.util.Set;

@SupportedSourceVersion(SourceVersion.RELEASE_8)
@SupportedAnnotationTypes("com.example.lombok.MyGetter")
public class MyGetterProcessor extends AbstractProcessor {

    private Messager messager; // 編譯時期輸入日志的
    private JavacTrees javacTrees; // 提供了待處理的抽象語法樹
    private TreeMaker treeMaker; // 封裝了創(chuàng)建AST節(jié)點的一些方法
    private Names names; // 提供了創(chuàng)建標識符的方法

    @Override
    public synchronized void init(ProcessingEnvironment processingEnv) {
        super.init(processingEnv);
        this.messager = processingEnv.getMessager();
        this.javacTrees = JavacTrees.instance(processingEnv);
        Context context = ((JavacProcessingEnvironment) processingEnv).getContext();
        this.treeMaker = TreeMaker.instance(context);
        this.names = Names.instance(context);
    }

    @Override
    public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
        Set<? extends Element> elementsAnnotatedWith = roundEnv.getElementsAnnotatedWith(MyGetter.class);
        elementsAnnotatedWith.forEach(e -> {
            JCTree tree = javacTrees.getTree(e);
            tree.accept(new TreeTranslator() {
                @Override
                public void visitClassDef(JCTree.JCClassDecl jcClassDecl) {
                    List<JCTree.JCVariableDecl> jcVariableDeclList = List.nil();
                    // 在抽象樹中找出所有的變量
                    for (JCTree jcTree : jcClassDecl.defs) {
                        if (jcTree.getKind().equals(Tree.Kind.VARIABLE)) {
                            JCTree.JCVariableDecl jcVariableDecl = (JCTree.JCVariableDecl) jcTree;
                            jcVariableDeclList = jcVariableDeclList.append(jcVariableDecl);
                        }
                    }
                    // 對于變量進行生成方法的操作
                    jcVariableDeclList.forEach(jcVariableDecl -> {
                        messager.printMessage(Diagnostic.Kind.NOTE, jcVariableDecl.getName() + " has been processed");
                        jcClassDecl.defs = jcClassDecl.defs.prepend(makeGetterMethodDecl(jcVariableDecl));
                    });
                    super.visitClassDef(jcClassDecl);
                }
            });
        });
        return true;
    }

    private JCTree.JCMethodDecl makeGetterMethodDecl(JCTree.JCVariableDecl jcVariableDecl) {
        ListBuffer<JCTree.JCStatement> statements = new ListBuffer<>();
        // 生成表達式 例如 this.a = a;
        JCTree.JCExpressionStatement aThis = makeAssignment(treeMaker.Select(treeMaker.Ident(
                names.fromString("this")), jcVariableDecl.getName()), treeMaker.Ident(jcVariableDecl.getName()));
        statements.append(aThis);
        JCTree.JCBlock block = treeMaker.Block(0, statements.toList());

        // 生成入?yún)?        JCTree.JCVariableDecl param = treeMaker.VarDef(treeMaker.Modifiers(Flags.PARAMETER),
                jcVariableDecl.getName(), jcVariableDecl.vartype, null);
        List<JCTree.JCVariableDecl> parameters = List.of(param);

        // 生成返回對象
        JCTree.JCExpression methodType = treeMaker.Type(new Type.JCVoidType());
        return treeMaker.MethodDef(treeMaker.Modifiers(Flags.PUBLIC),
                getNewMethodName(jcVariableDecl.getName()), methodType, List.nil(),
                parameters, List.nil(), block, null);

    }

    private Name getNewMethodName(Name name) {
        String s = name.toString();
        return names.fromString("get" + s.substring(0, 1).toUpperCase() + s.substring(1, name.length()));
    }

    private JCTree.JCExpressionStatement makeAssignment(JCTree.JCExpression lhs, JCTree.JCExpression rhs) {
        return treeMaker.Exec(
                treeMaker.Assign(
                        lhs,
                        rhs
                )
        );
    }
}

自定義的注解處理器是我們實現(xiàn)簡易版的 Lombok 的重中之重瑟慈,我們需要繼承 AbstractProcessor 類桃移,重寫它的 init() 和 process() 方法,在 process() 方法中我們先查詢所有的變量封豪,在給變量添加對應的方法谴轮。我們使用 TreeMaker 對象和 Names 來處理 AST,如上代碼所示吹埠。

當這些代碼寫好之后第步,我們就可以新增一個 Person 類來試一下我們自定義的 @MyGetter 功能了,代碼如下:

@MyGetter
public class Person {
    private String name;
}

2.使用自定義的注解處理器編譯代碼

上面的所有流程執(zhí)行完成之后缘琅,我們就可以編譯代碼測試效果了粘都。首先,我們先進入代碼的根目錄刷袍,執(zhí)行以下三條命令翩隧。

進入的根目錄如下:

image

① 使用 tools.jar 編譯自定義的注解器

javac -cp $JAVA_HOME/lib/tools.jar MyGetter* -d .

注意:命令最后面有一個“.”表示當前文件夾。

② 使用自定義注解器呻纹,編譯 Person 類

javac -processor com.example.lombok.MyGetterProcessor Person.java

③ 查看 Person 源碼

javap -p Person.class

源碼文件如下:

image

可以看到我們自定義的 getName() 方法已經(jīng)成功生成了堆生,到這里簡易版的 Lombok 就大功告成了。

Lombok 優(yōu)缺點

Lombok 的優(yōu)點很明顯雷酪,它可以讓我們寫更少的代碼吴旋,節(jié)約了開發(fā)時間奏候,并且讓代碼看起來更優(yōu)雅扮休,它的缺點有以下幾個唇撬。

缺點1:降低了可調(diào)試性

Lombok 會幫我們自動生成很多代碼,但這些代碼是在編譯期生成的吩跋,因此在開發(fā)和調(diào)試階段這些代碼可能是“丟失的”寞射,這就給調(diào)試代碼帶來了很大的不便。

缺點2:可能會有兼容性問題

Lombok 對于代碼有很強的侵入性锌钮,加上現(xiàn)在 JDK 版本升級比較快桥温,每半年發(fā)布一個版本,而 Lombok 又屬于第三方項目轧粟,并且由開源團隊維護策治,因此就沒有辦法保證版本的兼容性和迭代的速度脓魏,進而可能會產(chǎn)生版本不兼容的情況。

缺點3:可能會坑到隊友

尤其對于組人來的新人可能影響更大通惫,假如這個之前沒用過 Lombok茂翔,當他把代碼拉下來之后,因為沒有安裝 Lombok 的插件履腋,在編譯項目時珊燎,就會提示找不到方法等錯誤信息,導致項目編譯失敗遵湖,進而影響了團結(jié)成員之間的協(xié)作悔政。

缺點4:破壞了封裝性

面向?qū)ο蠓庋b的定義是:通過訪問權(quán)限控制,隱藏內(nèi)部數(shù)據(jù)延旧,外部僅能通過類提供的有限的接口訪問和修改內(nèi)部數(shù)據(jù)谋国。

也就是說,我們不應該無腦的使用 Lombok 對外暴露所有字段的 Getter/Setter 方法迁沫,因為有些字段在某些情況下是不允許直接修改的芦瘾,比如購物車中的商品數(shù)量,它直接影響了購物詳情和總價集畅,因此在修改的時候應該提供統(tǒng)一的方法近弟,進行關(guān)聯(lián)修改,而不是給每個字段添加訪問和修改的方法挺智。

小結(jié)

本文我們介紹了 Lombok 的使用以及執(zhí)行原理祷愉,它是通過 JDK 6 實現(xiàn)的 JSR 269: Pluggable Annotation Processing API (編譯期的注解處理器) ,在編譯期時把 Lombok 的注解轉(zhuǎn)換為 Java 的常規(guī)方法的赦颇,我們可以通過繼承 AbstractProcessor 類二鳄,重寫它的 init() 和 process() 方法,實現(xiàn)一個簡易版的 Lombok媒怯。但同時 Lombok 也存在這一些使用上的缺點泥从,比如:降低了可調(diào)試性、可能會有兼容性等問題沪摄,因此我們在使用時要根據(jù)自己的業(yè)務場景和實際情況,來選擇要不要使用 Lombok纱烘,以及應該如何使用 Lombok杨拐。

最后提醒一句,再好的技術(shù)也不是萬金油擂啥,就好像再好的鞋子也得適合自己的腳才行哄陶!

轉(zhuǎn)自 & 鳴謝
博主:macrozheng

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市哺壶,隨后出現(xiàn)的幾起案子屋吨,更是在濱河造成了極大的恐慌,老刑警劉巖至扰,帶你破解...
    沈念sama閱讀 206,311評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異敢课,居然都是意外死亡,警方通過查閱死者的電腦和手機直秆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評論 2 382
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來圾结,“玉大人瑰剃,你說我怎么就攤上這事◇菀埃” “怎么了晌姚?”我有些...
    開封第一講書人閱讀 152,671評論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀的道長遗座。 經(jīng)常有香客問我舀凛,道長,這世上最難降的妖魔是什么途蒋? 我笑而不...
    開封第一講書人閱讀 55,252評論 1 279
  • 正文 為了忘掉前任猛遍,我火速辦了婚禮,結(jié)果婚禮上号坡,老公的妹妹穿的比我還像新娘懊烤。我一直安慰自己,他們只是感情好宽堆,可當我...
    茶點故事閱讀 64,253評論 5 371
  • 文/花漫 我一把揭開白布腌紧。 她就那樣靜靜地躺著,像睡著了一般畜隶。 火紅的嫁衣襯著肌膚如雪壁肋。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,031評論 1 285
  • 那天籽慢,我揣著相機與錄音浸遗,去河邊找鬼。 笑死箱亿,一個胖子當著我的面吹牛跛锌,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播届惋,決...
    沈念sama閱讀 38,340評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼髓帽,長吁一口氣:“原來是場噩夢啊……” “哼菠赚!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起郑藏,我...
    開封第一講書人閱讀 36,973評論 0 259
  • 序言:老撾萬榮一對情侶失蹤衡查,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后译秦,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體峡捡,經(jīng)...
    沈念sama閱讀 43,466評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡筑悴,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,937評論 2 323
  • 正文 我和宋清朗相戀三年砚婆,在試婚紗的時候發(fā)現(xiàn)自己被綠了突勇。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片甲馋。...
    茶點故事閱讀 38,039評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡定躏,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出垮抗,到底是詐尸還是另有隱情冒版,我是刑警寧澤辞嗡,帶...
    沈念sama閱讀 33,701評論 4 323
  • 正文 年R本政府宣布欲间,位于F島的核電站,受9級特大地震影響班缎,放射性物質(zhì)發(fā)生泄漏她渴。R本人自食惡果不足惜趁耗,卻給世界環(huán)境...
    茶點故事閱讀 39,254評論 3 307
  • 文/蒙蒙 一苛败、第九天 我趴在偏房一處隱蔽的房頂上張望罢屈。 院中可真熱鬧缠捌,春花似錦译蒂、人聲如沸柔昼。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,259評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至招盲,卻和暖如春曹货,著一層夾襖步出監(jiān)牢的瞬間讳推,已是汗流浹背银觅。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留匀伏,地道東北人够颠。 一個月前我還...
    沈念sama閱讀 45,497評論 2 354
  • 正文 我出身青樓履磨,卻偏偏與公主長得像庆尘,于是被迫代替她去往敵國和親减余。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 42,786評論 2 345

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