java实现导入导出excel数据

项目需要,要实现一个导入导出excel的功能,于是,任务驱动着我学习到了POI和JXL这2个java操作Excel的插件。

一、POI和JXL介绍

1、POI:是对所有office资源进行读写的一套工具包、属于apache开源组织。

poi操作excel:
poi是把整个文件的属性都封装在HSSFWorkbook 中;
通过HSSFWorkbook来操作单个工作薄。然后通过工作薄来操作行;
在通过行来操控单元格。这样一级一级的分拆下来;
HSSFWorkbook---->HSSFSheet----->HSSFRow---->HSSFCell;
由于是基于HSSFWorkbook对象一步步创建起来的。所以不用把创建好的单元格添加进这个对象中、
如果需要对部分表格进行设置样式什么的。就可以创立HSSFCellStyle对象来进行设定样式;

2、JXL:只能对excel进行操作的一套工具包。

jxl是把整个文件封装在Workbook相关对象中;
通过Workbook去创建sheet工作薄;但是和poi不一样的地方是
jxl是通过向sheet中使用label(单元格)来进行读取写入;
Workbook----->sheet------>label ;
jxl是先创建一个工作区域、然后区创立单元格、单元格包含这个单元格的位置、内容等信息;然后把这个单元格加入工作区;

二、实例操作

1、用POI方式实现导出数据到excel功能

实现思路:当一点击导出按钮就跳转到指定的action去执行对应方法,先调用dao对数据库数据进行查询,根据dao层的方法返回一个集合对象,然后把这个集合对象的数据交互给POI去动态生成单元格并设置进去。项目后端我用的是ssh框架搭建项目,所以逻辑代码我写在action类中。直接看以下代码示例

前端jsp页面按钮部分

<button type="button" class="btn btn-default" style="width:80px;height:30px">
  <a href="excelAction!exportExchangeSudent"><i class="icon_upload"></i>&nbsp;&nbsp;<b>导出</b></a>
</button>

交换生实体类

package com.international.model;

import java.util.Date;

import cn.afterturn.easypoi.excel.annotation.Excel;
import cn.afterturn.easypoi.excel.annotation.ExcelTarget;

public class ExchangeStudent {
    //此表按照单表来操作,只用作记录
    private int id;
    private String studentNo;
    private String studentName;
    private String sex;
    private String major;
    private String className;
    private Date startTime;
    private Date endTime;
    private String exchangeCollege;
    private String reserves1;
    private String reserves2;
    private String reserves3;
    
    public int getId() {
        return id;
    }
    public void setId(int id) {
        this.id = id;
    }
    public String getStudentNo() {
        return studentNo;
    }
    public void setStudentNo(String studentNo) {
        this.studentNo = studentNo;
    }
    public String getStudentName() {
        return studentName;
    }
    public void setStudentName(String studentName) {
        this.studentName = studentName;
    }
    public String getSex() {
        return sex;
    }
    public void setSex(String sex) {
        this.sex = sex;
    }
    public String getMajor() {
        return major;
    }
    public void setMajor(String major) {
        this.major = major;
    }
    public String getClassName() {
        return className;
    }
    public void setClassName(String className) {
        this.className = className;
    }
    public Date getStartTime() {
        return startTime;
    }
    public void setStartTime(Date startTime) {
        this.startTime = startTime;
    }
    public Date getEndTime() {
        return endTime;
    }
    public void setEndTime(Date endTime) {
        this.endTime = endTime;
    }
    public String getExchangeCollege() {
        return exchangeCollege;
    }
    public void setExchangeCollege(String exchangeCollege) {
        this.exchangeCollege = exchangeCollege;
    }
    public String getReserves1() {
        return reserves1;
    }
    public void setReserves1(String reserves1) {
        this.reserves1 = reserves1;
    }
    public String getReserves2() {
        return reserves2;
    }
    public void setReserves2(String reserves2) {
        this.reserves2 = reserves2;
    }
    public String getReserves3() {
        return reserves3;
    }
    public void setReserves3(String reserves3) {
        this.reserves3 = reserves3;
    }
    
    
}

后端action类内的业务方法代码

//导出交换生
    public String exportExchangeSudent() throws Exception{
        //获取交换生对象
        List<ExchangeStudent> studentList = esd.queryInterStudents("");
        String []tableHeader={"学号","姓名","性别","班级","专业","交换开始时间","交换结束时间","交换的院校"}; 
        short cellNumber=(short)tableHeader.length;//表的列数 
        workbook = new HSSFWorkbook(); //创建一个Excel 
        style = workbook.createCellStyle(); //设置表头的类型 
        style.setAlignment(HorizontalAlignment.CENTER); 
        style1 = workbook.createCellStyle(); //设置数据类型 
        style1.setAlignment(HorizontalAlignment.CENTER); 
        HSSFFont font = workbook.createFont(); //设置字体 
        HSSFSheet sheet = workbook.createSheet("sheet1"); //创建一个sheet 
        HSSFHeader header = sheet.getHeader();//设置sheet的头 
        try {              
            //根据是否取出数据,设置header信息 
            if(studentList.size() < 1 ){ 
                header.setCenter("查无资料"); 
            }else{ 
                header.setCenter("交换生表"); 
                row = sheet.createRow(0); 
                row.setHeight((short)400);
                //表头
                for(int k = 0;k < cellNumber;k++){
                    cell = row.createCell((short) k);//创建第0行第k列 
                    cell.setCellValue(tableHeader[k]);//设置第0行第k列的值 
                    sheet.setColumnWidth((short)k,(short)8000);//设置列的宽度 
                    font.setColor(HSSFFont.COLOR_NORMAL); // 设置单元格字体的颜色. 
                    font.setFontHeight((short)350); //设置单元字体高度 
                    style1.setFont(font);//设置字体风格 
                    cell.setCellStyle(style1); 
                } 
                // 给Excel填充数据                         
                for(int i = 0 ;i < studentList.size() ;i++){    
                    //获取InternationalStudent对象
                    ExchangeStudent student1 = (ExchangeStudent)studentList.get(i); 
                    row = sheet.createRow((short) (i + 1));//创建第i+1行 
                    row.setHeight((short)400);//设置行高 
                    
                    if(student1.getStudentNo() != null){ 
                        cell = row.createCell((short) 0);//创建第i+1行第0列 
                        cell.setCellValue(student1.getStudentNo());//设置第i+1行第0列的值 
                        cell.setCellStyle(style);//设置风格 
                    } 
                    if(student1.getStudentName() != null){ 
                        cell = row.createCell((short) 1); //创建第i+1行第1列 
                        cell.setCellValue(student1.getStudentName());//设置第i+1行第1列的值 
                        cell.setCellStyle(style); //设置风格 
                    } 
                    if(student1.getSex() != null){ 
                        cell = row.createCell((short) 2); 
                        cell.setCellValue(student1.getSex()); 
                        cell.setCellStyle(style); 
                    } 
                    if(student1.getClassName()!= null){ 
                        cell = row.createCell((short) 3); 
                        cell.setCellValue(student1.getClassName()); 
                        cell.setCellStyle(style); 
                    }
                    if(student1.getMajor()!= null){ 
                        cell = row.createCell((short) 4); 
                        cell.setCellValue(student1.getMajor()); 
                        cell.setCellStyle(style); 
                    }
                    if(student1.getStartTime() != null){ 
                        cell = row.createCell((short) 5); 
                        cell.setCellValue(student1.getStartTime().toString().substring(0,10)); 
                        cell.setCellStyle(style); 
                    }
                    if(student1.getEndTime() != null){ 
                        cell = row.createCell((short) 6); 
                        cell.setCellValue(student1.getEndTime().toString().substring(0,10)); 
                        cell.setCellStyle(style); 
                    } 
                    if(student1.getExchangeCollege() != null){ 
                        cell = row.createCell((short) 7); 
                        cell.setCellValue(student1.getExchangeCollege()); 
                        cell.setCellStyle(style); 
                    } 
                } 
            } 
        } catch (Exception e) { 
            e.printStackTrace(); 
        } 
        outputSetting("交换生表.xls");
        return null; 
    }

//固定配置
    public void outputSetting(String fileName) {
        HttpServletResponse response = null;//创建一个HttpServletResponse对象 
        OutputStream out = null;//创建一个输出流对象 
        try { 
            response = ServletActionContext.getResponse();//初始化HttpServletResponse对象 
            out = response.getOutputStream();// 得到输出流
            response.setHeader("Content-disposition","attachment; filename="+new String(fileName.getBytes(),"ISO-8859-1"));//filename是下载的xls的名 
            response.setContentType("application/msexcel;charset=UTF-8");//设置类型 
            response.setHeader("Pragma","No-cache");//设置头 
            response.setHeader("Cache-Control","no-cache");//设置头 
            response.setDateHeader("Expires", 0);//设置日期头 
            workbook.write(out); 
            out.flush(); 
            workbook.write(out); 
        } catch (IOException e) { 
            e.printStackTrace(); 
        }finally{ 
            try{ 
                if(out!=null){ 
                    out.close(); 
                } 
            }catch(IOException e){ 
                e.printStackTrace(); 
            } 
        } 
    }

一些关于ssh框架的配置代码

//struts.xml文件
<!-- 导出Excel的Action -->
<action name="excelAction" class="excelAction">

</action>

//applicationContext-actions.xml
<!-- 导出的bean -->
<bean id="excelAction" class="com.international.common.ExcelAction" scope="prototype">
  <property name="esd">
    <ref bean="exchangeStuDao"/>
  </property>
</bean>
<!-- 交换生的bean -->
<bean id="exchangeStudentAction" class="com.international.actions.student.ExchangeStudentAction">
  <property name="esd">
    <ref bean="exchangeStuDao"/>
  </property>
</bean>

编写完以上代码,通过poi方式导出excel数据就实现了,主要关注action类那部分代码。

2、用JXL方式实现导入excel数据功能

实现思路:前端界面按下导入按钮,弹出个模态框,提示选择导入excel文件,其实导入功能是先上传文件到服务器指定路径,然后通过JXL去读取上传的excel数据,然后装进集合对象里,最后再循环的把集合对象里的数据遍历插入到数据库中,相当于遍历调用添加交换生的方法。直接看下面贴出来的代码。

前端jsp页面关键代码,有用到bootstrap框架

<div class="row" style="margin:5px">
                  <div class="nav search-row" id="top_menu">
                <!--  search form start -->
                <ul class="nav top-menu">                    
                    <li>
                       <form class="navbar-form" action="exchangeStudentAction!showStudent" method="post">
                             <input class="form-control" name="loginUserName" placeholder="输入查找的关键字" type="text" required/>
                             <button type="submit" class="btn btn-default" style="width:80px;height:30px">
                                <i class=" icon_search"></i>&nbsp;&nbsp;<b>搜索</b></button>
                             <button type="button" class="btn btn-default" style="width:80px;height:30px">
                                <a href="addExchangeStudent.jsp"><i class="icon_plus_alt2"></i>&nbsp;&nbsp;<b>添加</b></a></button>
                             <button type="button" class="btn btn-default" style="width:120px;height:30px">
                                <a href="exchangeStudentAction!showStudent?pageNo=1"><i class="icon_menu"></i>&nbsp;&nbsp;<b>显示全部</b></a></button>
                             <button type="button" class="btn btn-default" style="width:80px;height:30px">
                                <a href="excelAction!exportExchangeSudent"><i class="icon_upload"></i>&nbsp;&nbsp;<b>导出</b></a>
                             </button>
                             <button class="btn btn-default" data-toggle="modal" data-target="#myModal" style="width:80px;height:30px; float:rigtht">
                                    <a href="#"><i class="icon_download"></i>&nbsp;&nbsp;<b>导入</b></a>
                             </button>
                             <a class="btn btn-default" href="excelAction!downloadExStudent" style="width:140px;height:30px"><i class="icon_upload"></i>&nbsp;&nbsp;<b>下载Excel模板</b></a>
                        </form>
                    </li>                    
                </ul>
                <!--  search form end -->                
              </div>
            </div>
<!-- 模态框(Modal) -->
            <div class="modal fade" id="myModal" tabindex="-1" role="dialog" aria-labelledby="myModalLabel" aria-hidden="true">
                <div class="modal-dialog">
                    <div class="modal-content">
                        <div class="modal-header">
                            <button type="button" class="close" data-dismiss="modal" aria-hidden="true">
                                &times;
                            </button>
                            <h4 class="modal-title" id="myModalLabel">
                                导入交换生EXCEL表
                            </h4>                   
                        </div>
                        <div class="modal-body">
                           <form enctype="multipart/form-data" id="studentExcel" method="post">
                                 <input id="file-zh" name="upload" type="file" multiple>    
                                 <br/><br/>
                                 <b><a class="btn btn-primary">导入前请先下载交换生Excel表模板,按规范导入</a></b> 
                            </form>
                        </div>
                        <div class="modal-footer">
                            <button type="button" class="btn btn-default" data-dismiss="modal">关闭
                            </button>
                            <button type="button" class="btn btn-primary" data-dismiss="modal" onclick="importExcel()">
                                提交
                            </button>
                        </div>
                    </div><!-- /.modal-content -->
                </div><!-- /.modal -->
            </div>

js代码实现跳转action

//导入excel文件
            function importExcel(){
              var formData = new FormData($("#studentExcel")[0]);  // 要求使用的html对象
              //console.log(formData);
              $.ajax(         
                  {
                      type:"post",
                      url:"http://localhost:8080/InternationalSys/background/importExcelAction!importExStudentExcel",
                      //注:如果没有文件,只是简单的表单数据则可以使用 $('#formid').serialize();
                      data:formData,
                      dataType:"json",  
                      async:false,              
                      contentType: false,  
                      processData: false, 
                      success: function(data){
                          if(data!=null && data!=""){
                              alert(data);
                          }
                       }
                  }
              );
          }
        </script>

后端action类内的逻辑方法代码

package com.international.common;

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;

import org.apache.commons.io.FileUtils;
import org.apache.commons.lang.StringUtils;

import jxl.*;
import org.apache.struts2.ServletActionContext;

import com.international.dao.ExchangeStuDao;
import com.international.model.ExchangeStudent;
import com.opensymphony.xwork2.ActionSupport;

import jxl.read.biff.BiffException;
import jxl.write.DateTime;

public class ImportExcelAction extends ActionSupport{
    
    private String uploadFileName;
    private File upload;
    private String savePath;
    private List<ExchangeStudent> exStudentList = new ArrayList<ExchangeStudent>();
    
    private ExchangeStuDao esd;
    
    public ExchangeStuDao getEsd() {
        return esd;
    }
    public void setEsd(ExchangeStuDao esd) {
        this.esd = esd;
    }
    public String getUploadFileName() {
        return uploadFileName;
    }
    public void setUploadFileName(String uploadFileName) {
        this.uploadFileName = uploadFileName;
    }
    public File getUpload() {
        return upload;
    }
    public void setUpload(File upload) {
        this.upload = upload;
    }
    public String getSavePath() {
        return savePath;
    }
    public void setSavePath(String savePath) {
        this.savePath = savePath;
    }
    public List<ExchangeStudent> getExStudentList() {
        return exStudentList;
    }
    public void setExStudentList(List<ExchangeStudent> exStudentList) {
        this.exStudentList = exStudentList;
    }
    
    //导入excel数据到交换生表  
    @SuppressWarnings("static-access")
    public void importExStudentExcel() throws IOException{  
        System.out.println("excel的值为:"+ uploadFileName);
        String message="";
        String path=null;
        Workbook book = null;   //jxl工作簿
        InputStream fileIn = null;
        if(uploadFileName!=null && !uploadFileName.equals("")){
            //拦截仅允许上传文件类型
            if(uploadFileName.substring(uploadFileName.lastIndexOf(".")).equals(".xls")) {
                //获取需要上传文件的文件路径  
                path=ServletActionContext.getServletContext().getRealPath(this.getSavePath()+ "\\" +this.uploadFileName);
                System.out.println("上传Excel的路径:"+path);
                 //判断文件是否上传,如果上传的话将会创建该目录 
                File target= new File(path); // 定义目标文件对象
                try {
                    FileUtils.copyFile(upload, target);
                } catch (Exception e) {
                    e.printStackTrace();
                }               
                // 删除临时文件
                upload.delete();
                int sum=0;//计算导入成功的条数
                try {
                    System.out.println("上传路径:"+path);
                    fileIn = new FileInputStream(path);
                    //根据指定的文件输入流导入Excel从而产生Workbook对象 
                    System.out.println("输入流:"+fileIn);
                    //获取Excel对象
                    try {
                        book = book.getWorkbook(new File(path));
                    } catch (BiffException e1) {
                        // TODO 自动生成的 catch 块
                        e1.printStackTrace();
                    }
                    //获取Excel的第一个sheet表
                    Sheet sheet = (Sheet) book.getSheet(0);
                    System.out.println("获取到的Excel数据:"+ sheet);
                    System.out.println("行数:"+sheet.getRows());
                    System.out.println("列数:"+sheet.getColumns());
                    //对Sheet中的每一行进行迭代  
                    for (int i = 1; i < sheet.getRows(); i++) {
                        //创建实体类  
                        ExchangeStudent exStudent = new ExchangeStudent();
                        // 获取第一列第二行单元格对象
                        exStudent.setStudentNo(sheet.getCell(0, i).getContents());
                        exStudent.setStudentName(sheet.getCell(1, i).getContents());
                        exStudent.setSex(sheet.getCell(2, i).getContents());
                        exStudent.setClassName(sheet.getCell(3, i).getContents());
                        exStudent.setMajor(sheet.getCell(4, i).getContents());
                        Date sTime = null;
                        //日期格式处理方式:
                        if(sheet.getCell(5, i).getType() == CellType.DATE){
                             DateCell dc = (DateCell)sheet.getCell(5, i);
                             sTime = dc.getDate();    //获取单元格的date类型
                        }
                        exStudent.setStartTime(sTime);
                        Date eTime = null;
                        //日期格式处理方式:
                        if(sheet.getCell(6, i).getType() == CellType.DATE){
                             DateCell dc = (DateCell)sheet.getCell(6, i);
                             eTime = dc.getDate();    //获取单元格的date类型
                        }
                        exStudent.setEndTime(eTime);
                        exStudent.setExchangeCollege(sheet.getCell(7, i).getContents());
                        //不是空对象才加入集合中
                        if(exStudent!=null) {
                            exStudentList.add(exStudent);
                        }
                    }  
                    System.out.println(exStudentList);
                    for(int i=0; i<exStudentList.size(); i++){
                        if(!esd.queryStudentNo(exStudentList.get(i).getStudentNo())) {
                            if(esd.addExStudent(exStudentList.get(i))){
                                sum++;
                            }
                        }else {
                            message = "此条学生记录已存在";
                            System.out.println("此条学生记录已存在");
                            continue;
                        }
                        
                    }
                    System.out.println("导入的学生人数:"+sum);
                    if(sum>0){
                        System.out.println("导入成功");
                        message="导入成功";
                    }else{
                        System.out.println("导入失败");
                        message="导入失败";
                    }
                    fileIn.close();
                } catch (IOException ex) {
                    ex.printStackTrace();
                }
            }else {
                System.out.println("上传的只能是后缀为.xls的Excel文件");
                message = "上传的只能是后缀为.xls的Excel文件";
            }
        }else{
            message="请先上传交换生信息的Excel文件";
            System.out.println("请先上传交换生信息的Excel文件");
        }
        ajaxAction.toJson(ServletActionContext.getResponse(),message);
    }
}

dao层的代码

//根据学号查询数据库是否有这个交换生
    public boolean queryStudentNo(String stuNo) {
        Session session = null;
        try {
            session = sessionFactory.openSession();
            String hql = "from ExchangeStudent where studentNo="+stuNo;
            Query query = session.createQuery(hql);
            List list = query.list();
            if(list.size()>0) {
                return true;
            }else {
                return false;
            }
        }catch (Exception e) {
            e.printStackTrace();
            return false;
        }finally {
            session.close();
        }
    }

//添加交换生信息    
    public boolean addExStudent(ExchangeStudent exchangeStudent) {
        Session session = null;
        try{
            session = sessionFactory.openSession();
            Transaction tran = session.beginTransaction();
            session.save(exchangeStudent);
            tran.commit();
            return true;
        }catch(Exception e){
            e.printStackTrace();
            return false;
        }finally{
            session.close();
        }
    }

一些ssh的配置文件

//applicationContext-actions.xml
<!-- 导入的bean -->
<bean id="importExcelAction" class="com.international.common.ImportExcelAction">
  <property name="esd">
    <ref bean="exchangeStuDao"/>
  </property>
</bean>

//struts.xml
<!-- 导入Excel的Action -->
<action name="importExcelAction" class="importExcelAction">
  <param name="savePath">/upload</param>
</action>

根据以上的代码,通过jxl方法的导入功能也实现了,主要关注action类的那部分代码。


图片.png

三、期间遭遇问题(PS:我认为这才是我写这篇文章的初心和这篇文章的重点,差不多的功能对于每个人都可能出现许多不同的问题,我们要学会自己解决问题,我把这分享出来。)

注意:以上导入导出的实例代码要成功,都得基于导入相应的POI和JXL的jar包,否则是空谈。

  • 1、关于poi3.17版本设置单元格样式居中字段
  • 2、报错:Caused by: com.microsoft.sqlserver.jdbc.SQLServerException: 违反了 UNIQUE KEY 约束“UQ__Exchange__4D119D59097C43E3”。不能在对象“dbo.ExchangeStudent”中插入重复键。重复键值为 ()。
    问题原因:因为我数据库表中studentNo这个字段设置了唯一性约束,所以当excel插入的时候,数据库里面已经存在了该条记录,所以不让插入。
    解决方案:我在dao层里加入一个方法,就是上面的queryByStudentNo()方法,用来根据excel表里的学号数据去数据库查询有无这个学生,有的话就不插入了,继续下一次循环。
  • 3、导入excel表中的日期数据报错:java.text.ParseException: Unparseable date: "" 解决参考这里
//思路是借用下面这一段的,我自己写的在上面实例代码。
                DateTime accessTime = null;
                    //日期格式处理方式:
                    if(sheet.getCell(2, j).getType() == CellType.DATE){
                         DateCell dc = (DateCell)sheet.getCell(2, j);
                         Date date = dc.getDate();    //获取单元格的date类型
                         accessTime =  new DateTime(date); 
                    }
  • 4、JXL读取Excel文件,调用sheet.getRows()竟然返回多余的行数。这个有点玄学,明明excel数据有3行数据而已,在程序获取的时候竟然显示有33行数据,最后还是莫名其妙的就恢复正常了。
  • 5、报错:jxl读取excel文件异常,Unable to recognize OLE stream。这是jxl导入excel数据只支持后缀为.xls文件的问题。解决方案这里
  • 6、ssh上传文件到服务器隔一会文件会自动删除。解决方案这里
  • 7、导入时遇到多表关联的数据,比如我在导入国际班学生的excel数据时,excel表里有班级的字段,报错:org.hibernate.TransientObjectException: object references an unsaved tran。经排查,问题代码锁定在下图这块。
    图片.png
    参考了这篇文章,虽然不报错了,导入成功,但是并没有真正解决我的需求,因为这篇文章是级联保存主从表,即主表和从表都插入了数据,但我的需求是数据库里国际班学生表关联班级表,在批量插入国际班学生的信息时班级表不能保存,于是我这又真不知怎么办了,但我要让苍天知道我不认输,一点点思考排查。。。慢慢地,我逐渐了解到事情的本质,我这段代码问题出在我是新创建了一个班级对象,所以刚开始导入国际班学生信息时才会提示班级主表的信息没保存,于是我想到,那我不新创建班级对象,我直接去数据库里查这个班级对象,在设值进去国际班学生对象的属性里不就搞定了!最后代码就变成了这样,问题解决。
    图片.png

总结:这2功能我花了2天时间,第一次接触,期间遇到各种诡异问题,现在总算是实现了,代码路上,不放弃才能熬到最后。

四、参考文章

1、java通过jxl导入导出excel
2、poi和jxl操作office

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容