04_Spring

今天内容

  • Spring框架的事务管理

技术分析之Spring框架的事务管理相关的类和API
1,PlatformTransactionManager接口    -- 平台事务管理器.(真正管理事务的类)。该接口有具体的实现类,根据不同的持久层框架,需要选择不同的实现类!
2,TransactionDefinition接口       -- 事务定义信息.(事务的隔离级别,传播行为,超时,只读)
3,TransactionStatus接口        -- 事务的状态
4,总结:上述对象之间的关系:平台事务管理器真正管理事务对象.根据事务定义的信息TransactionDefinition 进行事务管理,在管理事务中产生一些状态.将状态记录到TransactionStatus中
5,PlatformTransactionManager接口中实现类和常用的方法,
① 接口的实现类

  • 如果使用的Spring的JDBC模板或者MyBatis框架,需要选择DataSourceTransactionManager实现类
  • 如果使用的是Hibernate的框架,需要选择HibernateTransactionManager实现类

② 该接口的常用方法

  • void commit(TransactionStatus status)
  • TransactionStatus getTransaction(TransactionDefinition definition)
  • void rollback(TransactionStatus status)

6,TransactionDefinition,
① 事务隔离级别的常量

  • static int ISOLATION_DEFAULT          -- 采用数据库的默认隔离级别
  • static int ISOLATION_READ_UNCOMMITTED    --
  • static int ISOLATION_READ_COMMITTED
  • static int ISOLATION_REPEATABLE_READ
  • static int ISOLATION_SERIALIZABLE

② 事务的传播行为常量(不用设置,使用默认值)

  • 先解释什么是事务的传播行为:解决的是业务层之间的方法调用!!

  • PROPAGATION_REQUIRED(默认值)   -- A中有事务,使用A中的事务.如果没有,B就会开启一个新的事务,将A包含进来.(保证A,B在同一个事务中),默认值!!

  • PROPAGATION_SUPPORTS       -- A中有事务,使用A中的事务.如果A中没有事务.那么B也不使用事务.

  • PROPAGATION_MANDATORY      -- A中有事务,使用A中的事务.如果A没有事务.抛出异常.

  • PROPAGATION_REQUIRES_NEW(记)  -- A中有事务,将A中的事务挂起.B创建一个新的事务.(保证A,B没有在一个事务中)

  • PROPAGATION_NOT_SUPPORTED    -- A中有事务,将A中的事务挂起.

  • PROPAGATION_NEVER         -- A中有事务,抛出异常.

  • PROPAGATION_NESTED(记)     -- 嵌套事务.当A执行之后,就会在这个位置设置一个保存点.如果B没有问题.执行通过.如果B出现异常,运行客户根据需求回滚(选择回滚到保存点或者是最初始状态)


技术分析之搭建事务管理转账案例的环境(强调:简化开发,以后DAO可以继承JdbcDaoSupport类)
1, 步骤一:创建WEB工程,引入需要的jar包

  • IOC的6个包
    com.springsource.org.apache.commons.logging-1.1.1.jar com.springsource.org.apache.log4j-1.2.15.jar spring-beans-4.2.4.RELEASE.jar spring-context-4.2.4.RELEASE.jar spring-core-4.2.4.RELEASE.jar spring-expression-4.2.4.RELEASE.jar

  • AOP的4个包
    com.springsource.org.aopalliance-1.0.0.jar com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar spring-aop-4.2.4.RELEASE.jar spring-aspects-4.2.4.RELEASE.jar

  • C3P0的1个包
    com.springsource.com.mchange.v2.c3p0-0.9.1.2.jar

  • MySQL的1个驱动包
    mysql-connector-java-5.1.7-bin.jar

  • JDBC目标2个包
    spring-jdbc-4.2.4.RELEASE.jar spring-tx-4.2.4.RELEASE.jar

  • 整合JUnit测试包
    spring-test-4.2.4.RELEASE.jar

2, 步骤二:引入配置文件

  • 引入配置文件

  • 引入log4j.properties

  • 在src目录下引入applicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:aop="http://www.springframework.org/schema/aop"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="http://www.springframework.org/schema/beans 
    http://www.springframework.org/schema/beans/spring-beans.xsd
    http://www.springframework.org/schema/context
    http://www.springframework.org/schema/context/spring-context.xsd
    http://www.springframework.org/schema/aop
    http://www.springframework.org/schema/aop/spring-aop.xsd
    http://www.springframework.org/schema/tx 
    http://www.springframework.org/schema/tx/spring-tx.xsd">
</beans>
  • applicationContext.xml中添加数据库连接
<!-- 配置C3P0连接池 -->
    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
        <property name="driverClass" value="com.mysql.jdbc.Driver"/>
        <property name="jdbcUrl" value="jdbc:mysql:///spring_day03"/>
        <property name="user" value="root"/>
        <property name="password" value="root"/>
    </bean>

3, 步骤三:创建对应的包结构和类

  • com.huachao.demo1
  • AccountService
  • AccountServlceImpl
  • AccountDao
  • AccountDaoImpl
package com.huachao.demo1;
public interface AccountService {
    public void pay(String out , String in , int money);
}
package com.huachao.demo1;
public class AccountServiceImp implements AccountService {
    private AccountDao accountDao;
    public void setAccountDao(AccountDao accountDao) {
        this.accountDao = accountDao;
    }
    @Override
    public void pay(String out, String in, int money) {
        accountDao.outMoney(out, money);
        accountDao.inMoney(in, money);
    }
}
package com.huachao.demo1;
public interface AccountDao {
    /**
     * 扣钱
     * @param out
     * @param money
     */
    public void outMoney(String out , int money);
    /**
     * 加钱
     * @param in
     * @param money
     */
    public void inMoney(String in , int money);
}
package com.huachao.demo1;
import org.springframework.jdbc.core.support.JdbcDaoSupport;
public class AccountDaoImp extends JdbcDaoSupport implements AccountDao {
    /**
     * 设置JdbcTemplate的代码每次都要写很麻烦
     * 如果能写在父类中,那就可以只写一次
     * Spring提供了这样的父类JdbcDaoSupport
     */
//    private JdbcTemplate jdbcTemplate;
//    public void setTemplate(JdbcTemplate jdbcTemplate) {
//        this.jdbcTemplate = jdbcTemplate;
//    }
    @Override
    public void outMoney(String out, int money) {
        System.out.println(out+"-"+money+"元");
        getJdbcTemplate().update("update t_account set money=money-? where name=?", money,out);
    }
    @Override
    public void inMoney(String in, int money) {
        System.out.println(in+"+"+money+"元");
        getJdbcTemplate().update("update t_account set money=money+? where name=?", money,in);
    }
}

4, 步骤四:引入Spring的配置文件,将类配置到Spring中
5, 步骤五:在业务层注入DAO ,在DAO中注入JDBC模板(强调:简化开发,以后DAO可以继承JdbcDaoSupport类)
6, 步骤六:编写DAO和Service中的方法

7,对于JdbcTemplate有两种方式,一种是在applicationContext.xml中配置,一种是在JdbcDaoSupport中注入dataSource,JdbcDaoSupport会自己创建JdbcTemplate
对应如下两种配置
方式一

<!-- 方式一 -->
    <!-- 配置JDBC模板 -->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="dataSource"/>
    </bean>
    <bean id="accountService" class="com.huachao.demo1.AccountServiceImp">
        <property name="accountDao" ref="accountDao"/>
    </bean>
    <bean id="accountDao" class="com.huachao.demo1.AccountDaoImp">
        <property name="jdbcTemplate" ref="jdbcTemplate"/>
    </bean>

方式二


    <!-- 方式二 -->
    <!-- 
        不配置JdbcTemplate的bean
        不注入jdbcTemplate,而是注入dataSource
        因为在JdbcDaoSupport类中有setDataSource方法,会自己创建一个JdbcTemplate
     -->
    <!-- 配置业务层和持久层 -->
    <bean id="accountService" class="com.huachao.demo1.AccountServiceImp">
        <property name="accountDao" ref="accountDao"/>
    </bean>
    <bean id="accountDao" class="com.huachao.demo1.AccountDaoImp">
        <property name="dataSource" ref="dataSource"/>
    </bean>

7, 步骤七:编写测试程序.

package com.huachao.demo1;

import javax.annotation.Resource;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class Demo1 {
    @Resource(name="accountService")
    private AccountService as;
    @Test
    public void Run1()
    {
        as.pay("冠希", "美美", 100);
    }
}

注:applicationContext.xml完整代码如下:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:aop="http://www.springframework.org/schema/aop"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="http://www.springframework.org/schema/beans 
    http://www.springframework.org/schema/beans/spring-beans.xsd
    http://www.springframework.org/schema/context
    http://www.springframework.org/schema/context/spring-context.xsd
    http://www.springframework.org/schema/aop
    http://www.springframework.org/schema/aop/spring-aop.xsd
    http://www.springframework.org/schema/tx 
    http://www.springframework.org/schema/tx/spring-tx.xsd">
    
    <!-- 配置C3P0连接池 -->
    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
        <property name="driverClass" value="com.mysql.jdbc.Driver"/>
        <property name="jdbcUrl" value="jdbc:mysql:///spring_day03"/>
        <property name="user" value="root"/>
        <property name="password" value="root"/>
    </bean>
    
    
    <!-- 方式一 -->
    <!-- 配置JDBC模板 -->
    <!-- <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="dataSource"/>
    </bean>
    <bean id="accountService" class="com.huachao.demo1.AccountServiceImp">
        <property name="accountDao" ref="accountDao"/>
    </bean>
    <bean id="accountDao" class="com.huachao.demo1.AccountDaoImp">
        <property name="jdbcTemplate" ref="jdbcTemplate"/>
    </bean> -->
    
    <!-- 方式二 -->
    <!-- 
        不配置JdbcTemplate的bean
        不注入jdbcTemplate,而是注入dataSource
        因为在JdbcDaoSupport类中有setDataSource方法,会自己创建一个JdbcTemplate
     -->
    <!-- 配置业务层和持久层 -->
    <bean id="accountService" class="com.huachao.demo1.AccountServiceImp">
        <property name="accountDao" ref="accountDao"/>
    </bean>
    <bean id="accountDao" class="com.huachao.demo1.AccountDaoImp">
        <property name="dataSource" ref="dataSource"/>
    </bean>
    
</beans>


技术分析之Spring框架的事务管理的分类
1, Spring的事务管理的分类

  • Spring的编程式事务管理(不推荐使用)
  • 通过手动编写代码的方式完成事务的管理(不推荐)

2, Spring的声明式事务管理(底层采用AOP的技术)

  • 通过一段配置的方式完成事务的管理(重点掌握注解的方式)

技术分析之Spring框架的事务管理之编程式的事务管理(了解)
1,说明:Spring为了简化事务管理的代码:提供了模板类 TransactionTemplate,所以手动编程的方式来管理事务,只需要使用该模板类即可!!

2,手动编程方式的具体步骤如下:
① 步骤一:配置一个事务管理器,Spring使用PlatformTransactionManager接口来管理事务,所以咱们需要使用到他的实现类!!
② 步骤二:配置事务管理的模板

<!-- 配置事务管理器 -->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"/>
    </bean>
    <!-- 声明手动编码,提供了类模板,使用该类管理事务比较简单,配置事务管理的模板 -->
    <bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate">
        <property name="transactionManager" ref="transactionManager"/>
    </bean>

③ 步骤三:在需要进行事务管理的类中,注入事务管理的模板.


    <bean id="accountService" class="com.huachao.demo1.AccountServiceImp">
        <property name="accountDao" ref="accountDao"/>
        <property name="transactionTemplate" ref="transactionTemplate"/>
    </bean>

④ 步骤四:在业务层使用模板管理事务:

package com.huachao.demo1;

import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.support.TransactionCallbackWithoutResult;
import org.springframework.transaction.support.TransactionTemplate;

public class AccountServiceImp implements AccountService {
    private AccountDao accountDao;
    public void setAccountDao(AccountDao accountDao) {
        this.accountDao = accountDao;
    }
    
    private TransactionTemplate transactionTemplate;
    public void setTransactionTemplate(TransactionTemplate transactionTemplate) {
        this.transactionTemplate = transactionTemplate;
    }

    @Override
    public void pay(final String out, final String in, final int money) {
        transactionTemplate.execute(new TransactionCallbackWithoutResult() {
            @Override
            protected void doInTransactionWithoutResult(TransactionStatus arg0) {
                accountDao.outMoney(out, money);
                int a = 100/0;
                accountDao.inMoney(in, money);
            }
        });
    }
}


Spring框架的事务管理之声明式事务管理,即通过配置文件来完成事务管理(AOP思想)

  • 声明式事务管理又分成两种方式
  • 基于AspectJ的XML方式(重点掌握)
  • 基于AspectJ的注解方式(重点掌握)

Spring框架的事务管理之基于AspectJ的XML方式(重点掌握)
1,将上面所有的类文件拷贝到包com.huachao.demo2
2, AccountServiceImp.java

package com.huachao.demo2;

public class AccountServiceImp implements AccountService {
    private AccountDao accountDao;
    public void setAccountDao(AccountDao accountDao) {
        this.accountDao = accountDao;
    }
    @Override
    public void pay(final String out, final String in, final int money) {
        accountDao.outMoney(out, money);
        int a = 100/0;
        accountDao.inMoney(in, money);
    }
}

3,applicationContext2.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:aop="http://www.springframework.org/schema/aop"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="http://www.springframework.org/schema/beans 
    http://www.springframework.org/schema/beans/spring-beans.xsd
    http://www.springframework.org/schema/context
    http://www.springframework.org/schema/context/spring-context.xsd
    http://www.springframework.org/schema/aop
    http://www.springframework.org/schema/aop/spring-aop.xsd
    http://www.springframework.org/schema/tx 
    http://www.springframework.org/schema/tx/spring-tx.xsd">
    
    <!-- 配置C3P0连接池 -->
    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
        <property name="driverClass" value="com.mysql.jdbc.Driver"/>
        <property name="jdbcUrl" value="jdbc:mysql:///spring_day03"/>
        <property name="user" value="root"/>
        <property name="password" value="root"/>
    </bean>
    
    <!-- 配置事务管理器 -->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"/>
    </bean>
    
    <!-- 配置业务层和持久层 -->
    <bean id="accountService" class="com.huachao.demo2.AccountServiceImp">
        <property name="accountDao" ref="accountDao"/>
    </bean>
    <bean id="accountDao" class="com.huachao.demo2.AccountDaoImp">
        <property name="dataSource" ref="dataSource"/>
    </bean>
    
</beans>

注:以上步骤将demo2包下的转账操作恢复为不带事务的

4,步骤一:恢复转账开发环境

5,步骤二:引入AOP的开发包,
6,步骤三:配置事务管理器

 <!-- 配置事务管理器 -->
 <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
 <property name="dataSource" ref="dataSource"/>
 </bean>

7, 步骤四:配置事务增强

 <!-- 配置事务增强 -->
 <tx:advice id="txAdvice" transaction-manager="transactionManager">
 <tx:attributes>
 <!--
 name :绑定事务的方法名,可以使用通配符,可以配置多个。
 propagation :传播行为
 isolation :隔离级别
 read-only :是否只读
 timeout :超时信息
 rollback-for:发生哪些异常回滚.
 no-rollback-for:发生哪些异常不回滚.
  -->
 <!-- 哪些方法加事务 -->
 <tx:method name="pay" propagation="REQUIRED"/>
 </tx:attributes>
 </tx:advice>

7,步骤五:配置AOP的切面

 <!-- 配置AOP切面产生代理 -->
 <aop:config>
      <aop:advisor advice-ref="myAdvice" pointcut="execution(* com.itheima.demo2.AccountServiceImpl.pay(..))"/>
     </aop:config>
  • 注意:如果是自己编写的切面,使用<aop:aspect>标签,如果是系统制作的,使用<aop:advisor>标签。

8,步骤六:编写测试类

 @RunWith(SpringJUnit4ClassRunner.class)
 @ContextConfiguration("classpath:applicationContext2.xml")
 public class Demo2 {
 
 @Resource(name="accountService")
 private AccountService accountService;
 
 @Test
 public void run1(){
 accountService.pay("冠希", "美美", 1000);
 }
 }

注:applicationContext2.xml完整代码如下

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:aop="http://www.springframework.org/schema/aop"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="http://www.springframework.org/schema/beans 
    http://www.springframework.org/schema/beans/spring-beans.xsd
    http://www.springframework.org/schema/context
    http://www.springframework.org/schema/context/spring-context.xsd
    http://www.springframework.org/schema/aop
    http://www.springframework.org/schema/aop/spring-aop.xsd
    http://www.springframework.org/schema/tx 
    http://www.springframework.org/schema/tx/spring-tx.xsd">
    
    <!-- 配置C3P0连接池 -->
    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
        <property name="driverClass" value="com.mysql.jdbc.Driver"/>
        <property name="jdbcUrl" value="jdbc:mysql:///spring_day03"/>
        <property name="user" value="root"/>
        <property name="password" value="root"/>
    </bean>
    
    <!-- 配置事务管理器 -->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"/>
    </bean>
    <!-- 声明事务(采用xml配置文件的方式) -->
    <!-- 先配置通知 -->
    <tx:advice id="myAdvisor" transaction-manager="transactionManager">
        <tx:attributes>
            <tx:method name="pay" propagation="REQUIRED"/>
        </tx:attributes>
    </tx:advice>
    
    <!-- 配置AOP切面:如果是自检编写的AOP使用aop:aspect配置
        使用的是spring框架提供的通知aop:advisor
     -->
     <aop:config>
         <aop:advisor advice-ref="myAdvisor" pointcut="execution(public * com.huachao.demo2.AccountServiceImp.pay(..))"/>
     </aop:config>
    
    <!-- 配置业务层和持久层 -->
    <bean id="accountService" class="com.huachao.demo2.AccountServiceImp">
        <property name="accountDao" ref="accountDao"/>
    </bean>
    <bean id="accountDao" class="com.huachao.demo2.AccountDaoImp">
        <property name="dataSource" ref="dataSource"/>
    </bean>
    
</beans>


Spring框架的事务管理之基于AspectJ的注解方式(重点掌握,最简单的方式)
0,按上面的方法,拷贝一份代码到com.huachao.demo3包下,恢复位不带事务的转账操作
1, 步骤一:恢复转账的开发环境
2, 步骤二:配置事务管理器

 <!-- 配置事务管理器  -->
 <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
 <property name="dataSource" ref="dataSource"/>
 </bean>

3, 步骤三:开启注解事务

 <!-- 开启注解事务 -->
 <tx:annotation-driven transaction-manager="transactionManager"/>

4, 步骤四:在业务层上添加一个注解:@Transactional

5, 编写测试类

 @RunWith(SpringJUnit4ClassRunner.class)
 @ContextConfiguration("classpath:applicationContext3.xml")
 public class Demo3 {
 
 @Resource(name="accountService")
 private AccountService accountService;
 
 @Test
 public void run1(){
 accountService.pay("冠希", "美美", 1000);
 }
 }

Eclipse需要做设置

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

推荐阅读更多精彩内容