23. 從零開始學(xué)springboot-excel導(dǎo)入導(dǎo)出

前言

上一章作者分享了mybatis通用mapper的實(shí)用覆旭。在實(shí)際開發(fā)中埠褪,我們常常會用到excel導(dǎo)入導(dǎo)出相關(guān)的功能,本章宫仗,作者將分享一個實(shí)用的excel操作案例够挂,實(shí)現(xiàn)excel的導(dǎo)入導(dǎo)出功能。

亮點(diǎn):

  • excel導(dǎo)入反射java實(shí)體
  • excel導(dǎo)出使用模版

這邊簡單解釋下藕夫。
excel導(dǎo)入反射java實(shí)體:
首先我們講下正常實(shí)現(xiàn)excel導(dǎo)入的方法孽糖,如下偽代碼

public List<Student> importExcel(File excelFile, Student student){
    List<Student> list = new ArrayList<Student>();
    for (excelFile) {
        Student s = new Student();
        //excel的第一行的第一列為name
        student.name = excelFile.row[0].column[0];
        //excel的第一行的第二列為age
        student.age = excelFile.row[0].column[1];
        list.add(s);
    }
    return list;
}

可以看到,我們實(shí)現(xiàn)了將excel數(shù)據(jù)列轉(zhuǎn)換為student實(shí)體的邏輯毅贮。那么办悟,問題來了,以后我想轉(zhuǎn)為teacher實(shí)體怎么辦滩褥?在寫一個類似的方法病蛉?可以看出,這種方式在方法體內(nèi)部實(shí)現(xiàn)實(shí)體類屬性與excel列的綁定,很不靈活铺然。那么俗孝,能不能傳入一個實(shí)體類,在運(yùn)行時魄健,動態(tài)的獲取該實(shí)體的屬性實(shí)現(xiàn)綁定呢赋铝?。此時就很有必要了解java的反射機(jī)制了沽瘦。
JAVA反射機(jī)制是在運(yùn)行狀態(tài)中革骨,對于任意一個實(shí)體類,都能夠知道這個類的所有屬性和方法其垄;對于任意一個對象苛蒲,都能夠調(diào)用它的任意方法和屬性卤橄。
我們再看看利用反射機(jī)制實(shí)現(xiàn)excel導(dǎo)入

public  <T extends Object> List<T> importExcel(File excelFile, Class<T> clazz) {
    List<T> list = new ArrayList<>();
    //獲取類的所有屬性
    Field[] fields = clazz.getDeclaredFields();
    T obj = null;
    for (excelFile){
        // 創(chuàng)建實(shí)體
        obj = clazz.newInstance();
        for (Field f : fields) {
            //屬性賦值绿满,實(shí)現(xiàn)屬性和excel列的綁定關(guān)系
        }
        list.add(obj);
    }
    return list;
}

可以看到,利用反射機(jī)制實(shí)現(xiàn)的導(dǎo)入方法更加靈活窟扑,這樣就可以使其更加通用喇颁。

excel導(dǎo)出使用模版
實(shí)現(xiàn)過excel導(dǎo)出功能的可能都知道,excel最麻煩的就是編寫樣式嚎货,尤其是樣式復(fù)雜的表格橘霎。仔細(xì)觀察表格,你會發(fā)現(xiàn)殖属,正常復(fù)雜的都是表頭樣式姐叁,往往是這樣的(舉個簡單的例子)


4.png

然后下半部分可能就是正常的列表展示。
那么我們能不能把這復(fù)雜的部分抽出來洗显,單獨(dú)由人工來做(畢竟office建立這么一個表格so easy外潜,用代碼去畫簡直抓狂。)挠唆,而后处窥,我們代碼讀取這個表格,動態(tài)填充數(shù)據(jù)呢玄组?答案滔驾,是肯定的。在接下來的案例中你會看到這樣的實(shí)現(xiàn)俄讹。

目錄結(jié)構(gòu)

1.png

添加依賴

 <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.mybatis.spring.boot</groupId>
            <artifactId>mybatis-spring-boot-starter</artifactId>
            <version>2.0.0</version>
        </dependency>

        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>8.0.15</version>
        </dependency>

        <!-- SpringBoot通用Mapper依賴 -->
        <dependency>
            <groupId>tk.mybatis</groupId>
            <artifactId>mapper-spring-boot-starter</artifactId>
            <version>2.1.5</version>
        </dependency>

        <!-- excel依賴 -->
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi</artifactId>
            <version>3.17</version>
        </dependency>
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-ooxml</artifactId>
            <version>3.17</version>
        </dependency>

        <!-- 實(shí)用工具 -->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <scope>provided</scope>
        </dependency>
    </dependencies>  
2.png

這里我們除了使用了poi excel的依賴哆致,還使用了上章用過的通用mapper依賴,以及l(fā)ombok依賴患膛,關(guān)于lombok依賴摊阀,你可以簡單理解為一個工具包,這樣我們編寫實(shí)體類時只需要使用

@Setter
@Getter

兩個注解即可自動實(shí)現(xiàn)類屬性的set和get方法,省了不少事驹溃。

添加配置

spring:
  datasource:
    url: jdbc:mysql://127.0.0.1:3306/test?useUnicode=true&characterEncoding=UTF-8&useSSL=false&autoReconnect=true&failOverReadOnly=false&serverTimezone=GMT%2B8
    username: root
    password: 123456
    driver-class-name: com.mysql.cj.jdbc.Driver

完善代碼

我們先實(shí)現(xiàn)excel的工具類

utils/excel/ExcelAnnotation

package com.mrcoder.sbmexcel.utils.excel;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
@Retention(RetentionPolicy.RUNTIME)
public @interface ExcelAnnotation {

    /**
     * 列索引
     *
     * @return
     */
    public int columnIndex() default 0;

    /**
     * 列名
     *
     * @return
     */
    public String columnName() default "";

}

ExcelAnnotation這個類實(shí)現(xiàn)了一個自定義的注解城丧,定義了兩個屬性,這樣我們可以在實(shí)體類中使用

//columnIndex定義excel的列豌鹤,columnName定義表列名稱
@ExcelAnnotation(columnIndex = 1, columnName = "姓名")
private String name;

意思就是導(dǎo)入的excel第二列對應(yīng)數(shù)據(jù)表的name列亡哄。
此處注意,excel索引是從0開始的布疙。

我們看下需要導(dǎo)入的excel表格結(jié)構(gòu)


3.png

第一列需要是我們無需導(dǎo)入的蚊惯,所以,我們定義實(shí)體類時灵临,excel注解從1開始截型。

接下來我們定義實(shí)體類
model/Excel

package com.mrcoder.sbmexcel.model;

import com.mrcoder.sbmexcel.utils.excel.ExcelAnnotation;
import lombok.Getter;
import lombok.Setter;
import lombok.experimental.Accessors;

import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.Table;
import java.util.Date;

@Setter
@Getter
@Table(name = "excel")
@Accessors(chain = true)
public class Excel {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
  
    @ExcelAnnotation(columnIndex = 1, columnName = "姓名")
    private String name;

    @ExcelAnnotation(columnIndex = 2, columnName = "年齡")
    private Integer age;

    @ExcelAnnotation(columnIndex = 3, columnName = "身高")
    private Double height;

    @ExcelAnnotation(columnIndex = 4, columnName = "體重")
    private Double weight;

    @ExcelAnnotation(columnIndex = 5, columnName = "學(xué)歷")
    private String edu;
    private Date createTime;
    private Date updateTime;
    private Integer status;
}

從實(shí)體類中可以看出,我們使用了

@Setter
@Getter

lombok注解儒溉,所以在該實(shí)體中無需手寫get和set方法宦焦。

然后我們實(shí)現(xiàn)excel讀取反射成實(shí)體的工具類
utils/excel/ExcelUtil

package com.mrcoder.sbmexcel.utils.excel;
import java.io.*;
import java.lang.reflect.Field;
import java.math.BigDecimal;
import java.util.ArrayList;
import java.util.List;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.ss.usermodel.*;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import org.springframework.web.multipart.MultipartFile;
/**
 * Excel 工具類
 *
 * @author mrcoder
 * @version 1.0 2019.05.29
 */
public class ExcelUtil {

    /**
     * 讀取excel反射實(shí)體
     *
     * @param file  MultipartFile
     * @param clazz entity
     * @return
     * @throws RuntimeException
     */
    public static <T extends Object> List<T> readExcelObject(MultipartFile file, Class<T> clazz) {

        // 存儲excel數(shù)據(jù)
        List<T> list = new ArrayList<>();
        Workbook wookbook = null;
        InputStream inputStream = null;

        try {
            inputStream = file.getInputStream();
        } catch (IOException e) {
            throw new RuntimeException("文件讀取異常");
        }

        // 根據(jù)excel文件版本獲取工作簿
        if (file.getOriginalFilename().endsWith(".xls")) {
            wookbook = xls(inputStream);
        } else if (file.getOriginalFilename().endsWith(".xlsx")) {
            wookbook = xlsx(inputStream);
        } else {
            throw new RuntimeException("非excel文件");
        }

        // 得到一個工作表
        Sheet sheet = wookbook.getSheetAt(0);

        // 獲取行總數(shù)
        int rows = sheet.getLastRowNum() + 1;
        Row row;

        // 獲取類所有屬性
        Field[] fields = clazz.getDeclaredFields();

        T obj = null;
        int coumnIndex = 0;
        Cell cell = null;
        ExcelAnnotation excelAnnotation = null;
        for (int i = 1; i < rows; i++) {
            // 獲取excel行
            row = sheet.getRow(i);
            //此處用來過濾空行
            Cell cell0 = row.getCell(0);
            cell0.setCellType(CellType.STRING);
            Cell cell1 = row.getCell(1);
            cell1.setCellType(CellType.STRING);
            if (cell0.getStringCellValue() == "" && cell1.getStringCellValue() == "") {
                continue;
            }
            try {
                // 創(chuàng)建實(shí)體
                obj = clazz.newInstance();
                for (Field f : fields) {
                    // 設(shè)置屬性可訪問
                    f.setAccessible(true);
                    // 判斷是否是注解
                    if (f.isAnnotationPresent(ExcelAnnotation.class)) {
                        // 獲取注解
                        excelAnnotation = f.getAnnotation(ExcelAnnotation.class);
                        // 獲取列索引
                        coumnIndex = excelAnnotation.columnIndex();
                        // 獲取單元格
                        cell = row.getCell(coumnIndex);
                        // 設(shè)置屬性
                        setFieldValue(obj, f, wookbook, cell);
                    }
                }
                System.out.println(obj);
                // 添加到集合中
                list.add(obj);
            } catch (InstantiationException e1) {
                e1.printStackTrace();
            } catch (IllegalAccessException e1) {
                e1.printStackTrace();
            } catch (Exception e) {
                e.printStackTrace();
                throw new RuntimeException("excel文件內(nèi)容出錯");
            }
        }
        try {
            //釋放資源
            wookbook.close();
            inputStream.close();

        } catch (IOException e) {
            e.printStackTrace();
        }
        return list;
    }

    /**
     * 綁定實(shí)體值
     *
     * @param obj      Object
     * @param f        Field
     * @param wookbook Workbook
     * @param cell     Cell
     * @return
     * @throws RuntimeException
     */
    private static void setFieldValue(Object obj, Field f, Workbook wookbook, Cell cell) {
        try {

            cell.setCellType(CellType.STRING);

            if (f.getType() == byte.class || f.getType() == Byte.class) {

                f.set(obj, Byte.parseByte(cell.getStringCellValue()));

            } else if (f.getType() == int.class || f.getType() == Integer.class) {

                f.set(obj, Integer.parseInt(cell.getStringCellValue()));

            } else if (f.getType() == Double.class || f.getType() == double.class) {

                f.set(obj, Double.parseDouble(cell.getStringCellValue()));

            } else if (f.getType() == BigDecimal.class) {

                f.set(obj, new BigDecimal(cell.getStringCellValue()));

            } else {

                f.set(obj, cell.getStringCellValue());

            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    /**
     * 對excel 2003處理
     */
    private static Workbook xls(InputStream is) {
        try {
            // 得到工作簿
            return new HSSFWorkbook(is);
        } catch (IOException e) {
            e.printStackTrace();
        }
        return null;
    }

    /**
     * 對excel 2007處理
     */
    private static Workbook xlsx(InputStream is) {
        try {
            // 得到工作簿
            return new XSSFWorkbook(is);
        } catch (IOException e) {
            e.printStackTrace();
        }
        return null;
    }
}

該工具類接受上傳excel文件,以及實(shí)體類參數(shù)顿涣,主要就是將每一行的excel數(shù)據(jù)轉(zhuǎn)換成一個實(shí)體波闹,這樣方便我們后續(xù)批量入庫操作。

mapper/ExcelMapper

package com.mrcoder.sbmexcel.mapper;
import com.mrcoder.sbmexcel.model.Excel;
import tk.mybatis.mapper.additional.insert.InsertListMapper;
import tk.mybatis.mapper.common.ExampleMapper;
import tk.mybatis.mapper.common.Mapper;

public interface ExcelMapper extends Mapper<Excel>, InsertListMapper<Excel>, ExampleMapper<Excel> {
}

service/ExcelService

package com.mrcoder.sbmexcel.service;


import com.mrcoder.sbmexcel.mapper.ExcelMapper;
import com.mrcoder.sbmexcel.model.Excel;
import com.mrcoder.sbmexcel.utils.excel.ExcelUtil;
import org.apache.poi.ss.usermodel.BorderStyle;
import org.apache.poi.ss.usermodel.CellStyle;
import org.apache.poi.ss.usermodel.FillPatternType;
import org.apache.poi.ss.usermodel.IndexedColors;
import org.apache.poi.ss.util.CellRangeAddress;
import org.apache.poi.xssf.usermodel.XSSFCellStyle;
import org.apache.poi.xssf.usermodel.XSSFRow;
import org.apache.poi.xssf.usermodel.XSSFSheet;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.util.ResourceUtils;
import org.springframework.web.multipart.MultipartFile;
import org.springframework.web.multipart.MultipartHttpServletRequest;

import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.List;

@Service
public class ExcelService {


    @Autowired
    private ExcelMapper excelMapper;


    //excel導(dǎo)入
    public int importExcel(HttpServletRequest request) {
        MultipartHttpServletRequest multipartRequest = (MultipartHttpServletRequest) request;
        MultipartFile excel = multipartRequest.getFile("filename");

        try {
            List<Excel> excelData = ExcelUtil.readExcelObject(excel, Excel.class);
            //檢查每列數(shù)據(jù)
            for (int i = 0; i < excelData.size(); i++) {
                excelData.get(i).setStatus(1);
                Date time = new Date();
                excelData.get(i).setCreateTime(time);
                excelData.get(i).setUpdateTime(time);
            }
            //批量導(dǎo)入
            return excelMapper.insertList(excelData);

        } catch (Exception e) {
            throw new RuntimeException(e.getMessage());
        }
    }


    //excel導(dǎo)入
    public void exportExcel(HttpServletResponse response) {
        try {
            // 1.讀取Excel模板
            File file = ResourceUtils.getFile(ResourceUtils.CLASSPATH_URL_PREFIX + "excel/export.xlsx");
            InputStream in = new FileInputStream(file);
            XSSFWorkbook wb = new XSSFWorkbook(in);
            // 2.讀取模板里面的所有Sheet
            XSSFSheet sheet = wb.getSheetAt(0);
            // 3.設(shè)置公式自動讀取
            sheet.setForceFormulaRecalculation(true);
            // 4.向相應(yīng)的單元格里面設(shè)置值
            XSSFRow row;

            // 5.得到第二行
            row = sheet.getRow(1);
            // 6.設(shè)置單元格屬性值和樣式
            row.getCell(1).setCellValue("張三");
            row.getCell(3).setCellValue("18");
            row.getCell(6).setCellValue("本科");
            row.getCell(8).setCellValue(new Date());

            row = sheet.getRow(2);
            row.getCell(1).setCellValue("1511xxxx234");
            row.getCell(3).setCellValue("廣東");
            row.getCell(6).setCellValue("本科");

            row = sheet.getRow(3);
            row.getCell(1).setCellValue("180");
            row.getCell(3).setCellValue("已婚");
            row.getCell(6).setCellValue("深圳");
            row.getCell(8).setCellValue("2");

            row = sheet.getRow(4);
            row.getCell(1).setCellValue("60");
            row.getCell(3).setCellValue("中國");
            row.getCell(6).setCellValue("其它");
            row.getCell(8).setCellValue("備注");

            //單元格合并
            row = sheet.getRow(6);
            row.getCell(0).setCellValue("合并列");
            CellRangeAddress region = new CellRangeAddress(6, 6, 0, 5);
            sheet.addMergedRegion(region);

            //單元格設(shè)置背景色
            CellStyle style = wb.createCellStyle();
            style.setFillForegroundColor(IndexedColors.SKY_BLUE.getIndex());
            style.setFillPattern(FillPatternType.SOLID_FOREGROUND);
            row.getCell(0).setCellStyle(style);

            //設(shè)置單元格邊框
            row = sheet.getRow(8);
            XSSFCellStyle style2 = wb.createCellStyle();
            style2.setBorderBottom(BorderStyle.DOUBLE);
            style2.setBorderRight(BorderStyle.DOUBLE);
            style2.setBorderLeft(BorderStyle.DOUBLE);
            style2.setBorderTop(BorderStyle.DOUBLE);
            style2.setBottomBorderColor(IndexedColors.SKY_BLUE.getIndex());
            style2.setRightBorderColor(IndexedColors.SKY_BLUE.getIndex());
            row.getCell(0).setCellStyle(style2);

            // 7.設(shè)置sheet名稱和單元格內(nèi)容
            wb.setSheetName(0, "測試");

            String fileName = new String(("export-" + new SimpleDateFormat("yyyy-MM-dd").format(new Date()))
                    .getBytes(), "UTF-8");
            response.setContentType("multipart/form-data");
            response.setHeader("Content-disposition", "attachment;filename=" + fileName + ".xlsx");
            response.setHeader("Pragma", "No-cache");
            response.setCharacterEncoding("utf-8");
            ServletOutputStream out = response.getOutputStream();
            wb.write(out);
            out.flush();
            out.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }


}

ExcelService實(shí)現(xiàn)了導(dǎo)入導(dǎo)出功能的封裝涛碑。導(dǎo)入功能無需多說精堕,導(dǎo)出實(shí)現(xiàn)了讀取模版,填充數(shù)據(jù)蒲障,以及對Excel 表格的一些合并歹篓,樣式修改等功能~

最后,我們完成controller
ExcelController

package com.mrcoder.sbmexcel.controller;
import com.mrcoder.sbmexcel.service.ExcelService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

@RestController
public class ExcelController {
    @Autowired
    private ExcelService excelService;
    @PostMapping("/importExcel")
    public int importExcel(HttpServletRequest request) {
        return excelService.importExcel(request);
    }
    @GetMapping("/exportExcel")
    public void exportExcel(HttpServletResponse response) {
        excelService.exportExcel(response);
    }
}

運(yùn)行

http://localhost:8080/importExcel 導(dǎo)入

3.png

http://localhost:8080/exportExcel 導(dǎo)出

4.png

項目地址

https://github.com/MrCoderStack/SpringBootDemo/tree/master/sbm-excel

請關(guān)注我的訂閱號

訂閱號.png
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末揉阎,一起剝皮案震驚了整個濱河市庄撮,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌余黎,老刑警劉巖重窟,帶你破解...
    沈念sama閱讀 211,423評論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異惧财,居然都是意外死亡巡扇,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,147評論 2 385
  • 文/潘曉璐 我一進(jìn)店門垮衷,熙熙樓的掌柜王于貴愁眉苦臉地迎上來厅翔,“玉大人,你說我怎么就攤上這事搀突〉睹疲” “怎么了?”我有些...
    開封第一講書人閱讀 157,019評論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長甸昏。 經(jīng)常有香客問我顽分,道長,這世上最難降的妖魔是什么施蜜? 我笑而不...
    開封第一講書人閱讀 56,443評論 1 283
  • 正文 為了忘掉前任卒蘸,我火速辦了婚禮,結(jié)果婚禮上翻默,老公的妹妹穿的比我還像新娘缸沃。我一直安慰自己,他們只是感情好修械,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,535評論 6 385
  • 文/花漫 我一把揭開白布趾牧。 她就那樣靜靜地躺著,像睡著了一般肯污。 火紅的嫁衣襯著肌膚如雪翘单。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,798評論 1 290
  • 那天仇箱,我揣著相機(jī)與錄音县恕,去河邊找鬼。 笑死剂桥,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的属提。 我是一名探鬼主播权逗,決...
    沈念sama閱讀 38,941評論 3 407
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼冤议!你這毒婦竟也來了斟薇?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,704評論 0 266
  • 序言:老撾萬榮一對情侶失蹤恕酸,失蹤者是張志新(化名)和其女友劉穎堪滨,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蕊温,經(jīng)...
    沈念sama閱讀 44,152評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡袱箱,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,494評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了义矛。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片发笔。...
    茶點(diǎn)故事閱讀 38,629評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖凉翻,靈堂內(nèi)的尸體忽然破棺而出了讨,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 34,295評論 4 329
  • 正文 年R本政府宣布前计,位于F島的核電站胞谭,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏男杈。R本人自食惡果不足惜韭赘,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,901評論 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望势就。 院中可真熱鬧泉瞻,春花似錦、人聲如沸苞冯。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,742評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽舅锄。三九已至鞭达,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間皇忿,已是汗流浹背畴蹭。 一陣腳步聲響...
    開封第一講書人閱讀 31,978評論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留鳍烁,地道東北人叨襟。 一個月前我還...
    沈念sama閱讀 46,333評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像幔荒,于是被迫代替她去往敵國和親糊闽。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,499評論 2 348