mybatis的原理总结部份

MyBatis 是支持定制化 SQL、存储过程以及高级映射的优秀的持久层框架,其主要就完成2件事情:

封装JDBC操作

利用反射打通Java类与SQL语句之间的相互转换

MyBatis的主要设计目的就是让我们对执行SQL语句时对输入输出的数据管理更加方便,所以方便地写出SQL

和方便地获取SQL的执行结果才是MyBatis的核心竞争力

MyBatis的主要成员

Configuration        MyBatis所有的配置信息都保存在Configuration对象之中,配置文件中的大部分

    配置都会存储到该类中

SqlSession          作为MyBatis工作的主要顶层API,表示和数据库交互时的会话,

    完成必要数据库增删改查功能

Configuration        MyBatis所有的配置信息都保存在Configuration对象之中,配置文件中的大部分配置都会存储到该类中

SqlSession            作为MyBatis工作的主要顶层API,表示和数据库交互时的会话,完成必要数据库增删改查功能

Executor               MyBatis执行器,是MyBatis 调度的核心,负责SQL语句的生成和查询缓存的维护

StatementHandler 封装了JDBC Statement操作,负责对JDBC statement 的操作,如设置参数等

ParameterHandler  负责对用户传递的参数转换成JDBC Statement 所对应的数据类型

ResultSetHandler   负责将JDBC返回的ResultSet结果集对象转换成List类型的集合

TypeHandler          负责java数据类型和jdbc数据类型(也可以说是数据表列类型)之间的映射和转换

MappedStatement  MappedStatement维护一条节点的封装

SqlSource              负责根据用户传递的parameterObject,动态地生成SQL语句,将信息封装到BoundSql对象中,并返回

BoundSql              表示动态生成的SQL语句以及相应的参数信息

以上主要成员在一次数据库操作中基本都会涉及,在SQL操作中重点需要关注的是SQL参数什么时候被设置和结果集怎么转换为JavaBean对象的,这两个过程正好对应StatementHandler和ResultSetHandler类中的处理逻辑。

MyBatis的初始化

MyBatis的初始化的过程其实就是解析配置文件和初始化Configuration的过程,MyBatis的初始化过程可用以下几行代码来表述:

String resource = "mybatis.xml";// 加载mybatis的配置文件(它也加载关联的映射文件)InputStream inputStream =null;try {

    inputStream = Resources.getResourceAsStream(resource);

} catch (IOException e) {

    e.printStackTrace();

}// 构建sqlSession的工厂sessionFactory =newSqlSessionFactoryBuilder().build(inputStream);

首先会创建SqlSessionFactory建造者对象,然后由它进行创建SqlSessionFactory。这里用到的是建造者模式,建造者模式最简单的理解就是不手动new对象,而是由其他类来进行对象的创建。

// SqlSessionFactoryBuilder类public SqlSessionFactory build(InputStream inputStream, String environment, Properties properties) {

    try {

        XMLConfigBuilder parser =new XMLConfigBuilder(inputStream, environment, properties);

        returnbuild(parser.parse());// 开始进行解析了 :)}catch (Exception e) {

        throwExceptionFactory.wrapException("Error building SqlSession.", e);

    } finally {

        ErrorContext.instance().reset();

        try {

            inputStream.close();

        } catch (IOException e) {

            // Intentionally ignore. Prefer previous error.        }

    }

}

XMLConfigBuilder对象会进行XML配置文件的解析,实际为configuration节点的解析操作。

// XMLConfigBuilder类public Configuration parse() {

    if (parsed) {

        thrownewBuilderException("Each XMLConfigBuilder can only be used once.");

    }

    parsed =true;

    parseConfiguration(parser.evalNode("/configuration"));

    return configuration;

}privatevoid parseConfiguration(XNode root) {

    try {

        //issue #117 read properties firstpropertiesElement(root.evalNode("properties"));

        Properties settings = settingsAsProperties(root.evalNode("settings"));

        loadCustomVfs(settings);

        typeAliasesElement(root.evalNode("typeAliases"));

        pluginElement(root.evalNode("plugins"));

        objectFactoryElement(root.evalNode("objectFactory"));

        objectWrapperFactoryElement(root.evalNode("objectWrapperFactory"));

        reflectorFactoryElement(root.evalNode("reflectorFactory"));

        settingsElement(settings);

        // read it after objectFactory and objectWrapperFactory issue #631/* 处理environments节点数据 */        environmentsElement(root.evalNode("environments"));

        databaseIdProviderElement(root.evalNode("databaseIdProvider"));

        typeHandlerElement(root.evalNode("typeHandlers"));

        mapperElement(root.evalNode("mappers"));

    } catch (Exception e) {

        thrownewBuilderException("Error parsing SQL Mapper Configuration. Cause: " + e, e);

    }

}

在configuration节点下会依次解析properties/settings/.../mappers等节点配置。在解析environments节点时,会根据transactionManager的配置来创建事务管理器,根据dataSource的配置来创建DataSource对象,这里面包含了数据库登录的相关信息。在解析mappers节点时,会读取该节点下所有的mapper文件,然后进行解析,并将解析后的结果存到configuration对象中。

// XMLConfigBuilder类privatevoidenvironmentsElement(XNode context)throws Exception {

    if(context !=null) {

        if(environment ==null) {

            environment = context.getStringAttribute("default");

        }

        for (XNode child : context.getChildren()) {

            String id = child.getStringAttribute("id");

            if (isSpecifiedEnvironment(id)) {

                /* 创建事务管理器 */                TransactionFactory txFactory = transactionManagerElement(child.evalNode("transactionManager"));

                DataSourceFactory dsFactory = dataSourceElement(child.evalNode("dataSource"));

                DataSource dataSource = dsFactory.getDataSource();

                /* 建造者模式 设计模式 */                Environment.Builder environmentBuilder =new Environment.Builder(id)

                        .transactionFactory(txFactory)

                        .dataSource(dataSource);

                configuration.setEnvironment(environmentBuilder.build());

            }

        }

    }

}// 解析单独的mapper文件privatevoidmapperElement(XNode parent)throws Exception {

    if(parent !=null) {

      for (XNode child : parent.getChildren()) {

        if("package".equals(child.getName())) {

          String mapperPackage = child.getStringAttribute("name");

          configuration.addMappers(mapperPackage);

        } else {

          String resource = child.getStringAttribute("resource");

          String url = child.getStringAttribute("url");

          String mapperClass = child.getStringAttribute("class");

          if(resource !=null&& url ==null&& mapperClass ==null) {

            ErrorContext.instance().resource(resource);

            InputStream inputStream = Resources.getResourceAsStream(resource);

            XMLMapperBuilder mapperParser =new XMLMapperBuilder(inputStream, configuration, resource, configuration.getSqlFragments());

            mapperParser.parse(); // 开始解析mapper文件了 :)}elseif(resource ==null&& url !=null&& mapperClass ==null) {

            ErrorContext.instance().resource(url);

            InputStream inputStream = Resources.getUrlAsStream(url);

            XMLMapperBuilder mapperParser =new XMLMapperBuilder(inputStream, configuration, url, configuration.getSqlFragments());

            mapperParser.parse();

          } elseif(resource ==null&& url ==null&& mapperClass !=null) {

            Class mapperInterface = Resources.classForName(mapperClass);

            configuration.addMapper(mapperInterface);

          } else {

            thrownewBuilderException("A mapper element may only specify a url, resource or class, but not more than one.");

          }

        }

      }

    }

  }

解析完MyBatis配置文件后,configuration就初始化完成了,然后根据configuration对象来创建SqlSession,到这里时,MyBatis的初始化的征程已经走完了。

// SqlSessionFactoryBuilder类public SqlSessionFactory build(Configuration config) {

    returnnew DefaultSqlSessionFactory(config);

}


MyBatis的SQL查询流程

SQL语句的执行才是MyBatis的重要职责,该过程就是通过封装JDBC进行操作,然后使用Java反射技术完成JavaBean对象到数据库参数之间的相互转换,这种映射关系就是有TypeHandler对象来完成的,在获取数据表对应的元数据时,会保存该表所有列的数据库类型,大致逻辑如下所示:

/* Get resultSet metadata */ResultSetMetaData metaData = resultSet.getMetaData();intcolumn = metaData.getColumnCount();for(inti = 1; i <= column; i++) {

    JdbcType jdbcType = JdbcType.forCode(metaData.getColumnType(i));

    typeHandlers.add(TypeHandlerRegistry.getTypeHandler(jdbcType));

    columnNames.add(metaData.getColumnName(i));

}


使用如下代码进行SQL查询操作:

sqlSession = sessionFactory.openSession();

User user = sqlSession.selectOne("com.luo.dao.UserDao.getUserById", 1);

System.out.println(user);

创建sqlSession的过程其实就是根据configuration中的配置来创建对应的类,然后返回创建的sqlSession对象。调用selectOne方法进行SQL查询,selectOne方法最后调用的是selectList,在selectList中,会查询configuration中存储的MappedStatement对象,mapper文件中一个sql语句的配置对应一个MappedStatement对象,然后调用执行器进行查询操作。

// DefaultSqlSession类public T selectOne(String statement, Object parameter) {

    // Popular vote was to return null on 0 results and throw exception on too many.List list =this.selectList(statement, parameter);

    if(list.size() == 1) {

        returnlist.get(0);

    } elseif(list.size() > 1) {

        thrownewTooManyResultsException("Expected one result (or null) to be returned by selectOne(), but found: " + list.size());

    } else {

        returnnull;

    }

}public List selectList(String statement, Object parameter, RowBounds rowBounds) {

    try {

        MappedStatement ms = configuration.getMappedStatement(statement);

        return executor.query(ms, wrapCollection(parameter), rowBounds, Executor.NO_RESULT_HANDLER);

    } catch (Exception e) {

        throwExceptionFactory.wrapException("Error querying database.  Cause: " + e, e);

    } finally {

        ErrorContext.instance().reset();

    }

}

执行器在query操作中,优先会查询缓存是否命中,命中则直接返回,否则从数据库中查询。

// CachingExecutor类public List query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler)throws SQLException {

    /* 获取关联参数的sql,boundSql */    BoundSql boundSql = ms.getBoundSql(parameterObject);

    /* 创建cache key值 */    CacheKey key = createCacheKey(ms, parameterObject, rowBounds, boundSql);

    return query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);

}public List query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql)

      throws SQLException {

    /* 获取二级缓存实例 */    Cache cache = ms.getCache();

    if(cache !=null) {

        flushCacheIfRequired(ms);

        if(ms.isUseCache() && resultHandler ==null) {

            ensureNoOutParams(ms, parameterObject, boundSql);

            @SuppressWarnings("unchecked")

            List list = (List) tcm.getObject(cache, key);

            if(list ==null) {

                list = delegate. query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);

                tcm.putObject(cache, key, list); // issue #578 and #116            }

            return list;

        }

    }

    returndelegate. query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);

}private List queryFromDatabase(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql)throws SQLException {

    List list;

    /**    * 先往localCache中插入一个占位对象,这个地方

    */    localCache.putObject(key, EXECUTION_PLACEHOLDER);

    try {

        list = doQuery(ms, parameter, rowBounds, resultHandler, boundSql);

    } finally {

        localCache.removeObject(key);

    }

    /* 往缓存中写入数据,也就是缓存查询结果 */    localCache.putObject(key, list);

    if(ms.getStatementType() == StatementType.CALLABLE) {

        localOutputParameterCache.putObject(key, parameter);

    }

    return list;

}

真正的doQuery操作是由SimplyExecutor代理来完成的,该方法中有2个子流程,一个是SQL参数的设置,另一个是SQL查询操作和结果集的封装。

// SimpleExecutor类public List doQuery(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql)throws SQLException {

    Statement stmt =null;

    try {

        Configuration configuration = ms.getConfiguration();

        StatementHandler handler = configuration.newStatementHandler(wrapper, ms, parameter, rowBounds, resultHandler, boundSql);

        /* 子流程1: SQL查询参数的设置 */        stmt = prepareStatement(handler, ms.getStatementLog());

        /* 子流程2: SQL查询操作和结果集封装 */returnhandler.query(stmt, resultHandler);

    } finally {

        closeStatement(stmt);

    }

}

子流程1 SQL查询参数的设置:

首先获取数据库connection连接,然后准备statement,然后就设置SQL查询中的参数值。打开一个connection连接,在使用完后不会close,而是存储下来,当下次需要打开连接时就直接返回。

// SimpleExecutor类privateStatement prepareStatement(StatementHandler handler, Log statementLog)throws SQLException {

    Statement stmt;

    /* 获取Connection连接 */    Connection connection = getConnection(statementLog);

    /* 准备Statement */    stmt = handler.prepare(connection, transaction.getTimeout());

    /* 设置SQL查询中的参数值 */    handler.parameterize(stmt);

    return stmt;

}// DefaultParameterHandler类publicvoid setParameters(PreparedStatement ps) {

    /**    * 设置SQL参数值,从ParameterMapping中读取参数值和类型,然后设置到SQL语句中

    */    ErrorContext.instance().activity("setting parameters").object(mappedStatement.getParameterMap().getId());

    List parameterMappings = boundSql.getParameterMappings();

    if(parameterMappings !=null) {

        for(inti = 0; i < parameterMappings.size(); i++) {

            ParameterMapping parameterMapping = parameterMappings.get(i);

            if(parameterMapping.getMode() != ParameterMode.OUT) {

                Object value;

                String propertyName = parameterMapping.getProperty();

                if(boundSql.hasAdditionalParameter(propertyName)) {// issue #448 ask first for additional paramsvalue = boundSql.getAdditionalParameter(propertyName);

                } elseif(parameterObject ==null) {

                    value =null;

                } elseif (typeHandlerRegistry.hasTypeHandler(parameterObject.getClass())) {

                    value = parameterObject;

                } else {

                    MetaObject metaObject = configuration.newMetaObject(parameterObject);

                    value = metaObject.getValue(propertyName);

                }

                TypeHandler typeHandler = parameterMapping.getTypeHandler();

                JdbcType jdbcType = parameterMapping.getJdbcType();

                if(value ==null&& jdbcType ==null) {

                    jdbcType = configuration.getJdbcTypeForNull();

                }

                try {

                    typeHandler.setParameter(ps, i + 1, value, jdbcType);

                } catch (TypeException e) {

                    thrownewTypeException("Could not set parameters for mapping: " + parameterMapping + ". Cause: " + e, e);

                } catch (SQLException e) {

                    thrownewTypeException("Could not set parameters for mapping: " + parameterMapping + ". Cause: " + e, e);

                }

            }

        }

    }

}

子流程2 SQL查询结果集的封装:

// SimpleExecutor类public List query(Statement statement, ResultHandler resultHandler)throws SQLException {

    PreparedStatement ps = (PreparedStatement) statement;

    // 执行查询操作    ps.execute();

    // 执行结果集封装returnresultSetHandler. handleResultSets(ps);

}// DefaultReseltSetHandler类publicList handleResultSets(Statement stmt)throws SQLException {

    ErrorContext.instance().activity("handling results").object(mappedStatement.getId());

    finalList multipleResults =newArrayList();

    intresultSetCount = 0;

    /**    * 获取第一个ResultSet,同时获取数据库的MetaData数据,包括数据表列名、列的类型、类序号等。

    * 这些信息都存储在了ResultSetWrapper中了

    */    ResultSetWrapper rsw = getFirstResultSet(stmt);

    List resultMaps = mappedStatement.getResultMaps();

    intresultMapCount = resultMaps.size();

    validateResultMapsCount(rsw, resultMapCount);

    while(rsw !=null&& resultMapCount > resultSetCount) {

      ResultMap resultMap = resultMaps.get(resultSetCount);

      handleResultSet(rsw, resultMap, multipleResults, null);

      rsw = getNextResultSet(stmt);

      cleanUpAfterHandlingResultSet();

      resultSetCount++;

    }

    String[] resultSets = mappedStatement.getResultSets();

    if(resultSets !=null) {

      while(rsw !=null&& resultSetCount < resultSets.length) {

        ResultMapping parentMapping = nextResultMaps.get(resultSets[resultSetCount]);

        if(parentMapping !=null) {

          String nestedResultMapId = parentMapping.getNestedResultMapId();

          ResultMap resultMap = configuration.getResultMap(nestedResultMapId);

          handleResultSet(rsw, resultMap, null, parentMapping);

        }

        rsw = getNextResultSet(stmt);

        cleanUpAfterHandlingResultSet();

        resultSetCount++;

      }

    }

    return collapseSingleResultList(multipleResults);

  }

ResultSetWrapper是ResultSet的包装类,调用getFirstResultSet方法获取第一个ResultSet,同时获取数据库的MetaData数据,包括数据表列名、列的类型、类序号等,这些信息都存储在ResultSetWrapper类中了。然后调用handleResultSet方法来来进行结果集的封装。

// DefaultResultSetHandler类privatevoidhandleResultSet(ResultSetWrapper rsw, ResultMap resultMap, List multipleResults, ResultMapping parentMapping)throws SQLException {

    try {

        if(parentMapping !=null) {

            handleRowValues(rsw, resultMap, null, RowBounds.DEFAULT, parentMapping);

        } else {

            if(resultHandler ==null) {

                DefaultResultHandler defaultResultHandler =new DefaultResultHandler(objectFactory);

                handleRowValues(rsw, resultMap, defaultResultHandler, rowBounds, null);

                multipleResults.add(defaultResultHandler.getResultList());

            } else {

                handleRowValues(rsw, resultMap, resultHandler, rowBounds, null);

            }

        }

    } finally {

        // issue #228 (close resultsets)        closeResultSet(rsw.getResultSet());

    }

}

这里调用handleRowValues方法进行结果值的设置。

// DefaultResultSetHandler类publicvoidhandleRowValues(ResultSetWrapper rsw, ResultMap resultMap, ResultHandler resultHandler, RowBounds rowBounds, ResultMapping parentMapping)throws SQLException {

    if (resultMap.hasNestedResultMaps()) {

        ensureNoRowBounds();

        checkResultHandler();

        handleRowValuesForNestedResultMap(rsw, resultMap, resultHandler, rowBounds, parentMapping);

    } else {

        // 封装数据        handleRowValuesForSimpleResultMap(rsw, resultMap, resultHandler, rowBounds, parentMapping);

    }

}privatevoidhandleRowValuesForSimpleResultMap(ResultSetWrapper rsw, ResultMap resultMap, ResultHandler resultHandler, RowBounds rowBounds, ResultMapping parentMapping)

        throws SQLException {

    DefaultResultContext resultContext =newDefaultResultContext();

    skipRows(rsw.getResultSet(), rowBounds);

    while(shouldProcessMoreRows(resultContext, rowBounds) && rsw.getResultSet().next()) {

        ResultMap discriminatedResultMap = resolveDiscriminatedResultMap(rsw.getResultSet(), resultMap,null);

        Object rowValue = getRowValue(rsw, discriminatedResultMap);

        storeObject(resultHandler, resultContext, rowValue, parentMapping, rsw.getResultSet());

    }

}privateObject getRowValue(ResultSetWrapper rsw, ResultMap resultMap)throws SQLException {

    finalResultLoaderMap lazyLoader =new ResultLoaderMap();

    // createResultObject为新创建的对象,数据表对应的类Object rowValue = createResultObject(rsw, resultMap, lazyLoader,null);

    if(rowValue !=null&& !hasTypeHandlerForResultObject(rsw, resultMap.getType())) {

        finalMetaObject metaObject = configuration.newMetaObject(rowValue);

        booleanfoundValues =this.useConstructorMappings;

        if(shouldApplyAutomaticMappings(resultMap,false)) {

            // 这里把数据填充进去,metaObject中包含了resultObject信息foundValues = applyAutomaticMappings(rsw, resultMap, metaObject,null) || foundValues;

        }

        foundValues = applyPropertyMappings(rsw, resultMap, metaObject, lazyLoader,null) || foundValues;

        foundValues = lazyLoader.size() > 0 || foundValues;

        rowValue = (foundValues || configuration.isReturnInstanceForEmptyRow()) ? rowValue :null;

    }

    return rowValue;

}privatebooleanapplyAutomaticMappings(ResultSetWrapper rsw, ResultMap resultMap, MetaObject metaObject, String columnPrefix)throws SQLException {

    List autoMapping = createAutomaticMappings(rsw, resultMap, metaObject, columnPrefix);

    booleanfoundValues =false;

    if(autoMapping.size() > 0) {

        // 这里进行for循环调用,因为user表中总共有7列,所以也就调用7次for (UnMappedColumnAutoMapping mapping : autoMapping) {

            // 这里将esultSet中查询结果转换为对应的实际类型finalObject value = mapping.typeHandler.getResult(rsw.getResultSet(), mapping.column);

            if(value !=null) {

                foundValues =true;

            }

            if(value !=null|| (configuration.isCallSettersOnNulls() && !mapping.primitive)) {

                // gcode issue #377, call setter on nulls (value is not 'found')                metaObject.setValue(mapping.property, value);

            }

        }

    }

    return foundValues;

}

mapping.typeHandler.getResult会获取查询结果值的实际类型,比如我们user表中id字段为int类型,那么它就对应Java中的Integer类型,然后通过调用statement.getInt("id")来获取其int值,其类型为Integer。metaObject.setValue方法会把获取到的Integer值设置到Java类中的对应字段。

// MetaObject类publicvoid setValue(String name, Object value) {

    PropertyTokenizer prop =new PropertyTokenizer(name);

    if (prop.hasNext()) {

        MetaObject metaValue = metaObjectForProperty(prop.getIndexedName());

        if(metaValue == SystemMetaObject.NULL_META_OBJECT) {

            if(value ==null&& prop.getChildren() !=null) {

                // don't instantiate child path if value is nullreturn;

            } else {

                metaValue = objectWrapper.instantiatePropertyValue(name, prop, objectFactory);

            }

        }

        metaValue.setValue(prop.getChildren(), value);

    } else {

        objectWrapper.set(prop, value);

    }

}

metaValue.setValue方法最后会调用到Java类中对应数据域的set方法,这样也就完成了SQL查询结果集的Java类封装过程。最后贴一张调用栈到达Java类的set方法中的快照:


MyBatis缓存

MyBatis提供查询缓存,用于减轻数据库压力,提高性能。MyBatis提供了一级缓存和二级缓存。

一级缓存是SqlSession级别的缓存,每个SqlSession对象都有一个哈希表用于缓存数据,不同SqlSession对象之间缓存不共享。同一个SqlSession对象对象执行2遍相同的SQL查询,在第一次查询执行完毕后将结果缓存起来,这样第二遍查询就不用向数据库查询了,直接返回缓存结果即可。MyBatis默认是开启一级缓存的。

二级缓存是mapper级别的缓存,二级缓存是跨SqlSession的,多个SqlSession对象可以共享同一个二级缓存。不同的SqlSession对象执行两次相同的SQL语句,第一次会将查询结果进行缓存,第二次查询直接返回二级缓存中的结果即可。MyBatis默认是不开启二级缓存的,可以在配置文件中使用如下配置来开启二级缓存:

当SQL语句进行更新操作(删除/添加/更新)时,会清空对应的缓存,保证缓存中存储的都是最新的数据。MyBatis的二级缓存对细粒度的数据级别的缓存实现不友好,比如如下需求:对商品信息进行缓存,由于商品信息查询访问量大,但是要求用户每次都能查询最新的商品信息,此时如果使用mybatis的二级缓存就无法实现当一个商品变化时只刷新该商品的缓存信息而不刷新其它商品的信息,因为mybaits的二级缓存区域以mapper为单位划分,当一个商品信息变化会将所有商品信息的缓存数据全部清空。解决此类问题需要在业务层根据需求对数据有针对性缓存,具体业务具体实现。

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

推荐阅读更多精彩内容