01_Spring

今天内容概述

  1. Spring框架的概述
  2. SpringIOC的快速入门
  3. IoC容器XML的方式
  4. 在web项目中集成Spring

案例一:使用Spring的IOC技术完成客户的保存功能


需求分析

  1. 使用Spring的IOC技术完成客户的保存功能

技术分析之Spring框架的概述和入门


技术分析之什么是Spring框架
1, Spring框架的概述

  • Spring是一个开源框架

  • Spring是于2003 年兴起的一个轻量级的Java开发框架,由Rod Johnson在其著作Expert One-On-One J2EE Development and Design中阐述的部分理念和原型衍生而来。

  • 它是为了解决企业应用开发的复杂性而创建的。框架的主要优势之一就是其分层架构,分层架构允许使用者选择使用哪一个组件,同时为 J2EE 应用程序开发提供集成的框架。

  • Spring使用基本的JavaBean来完成以前只可能由EJB完成的事情。然而,Spring的用途不仅限于服务器端的开发。从简单性、可测试性和松耦合的角度而言,任何Java应用都可以 从Spring中受益。

  • Spring的核心是控制反转(IoC)和面向切面(AOP)。简单来说,Spring是一个分层的JavaSE/EEfull-stack(一站式) 轻量级开源框架。

  • EE开发分成三层结构

  • WEB层    -- Spring MVC

  • 业务层   -- Bean管理:(IOC)

  • 持久层   -- Spring的JDBC模板.ORM模板用于整合其他的持久层框架


技术分析之Spring框架的特点
1, 为什么要学习Spring的框架

  • 方便解耦,简化开发
  • Spring就是一个大工厂,可以将所有对象创建和依赖关系维护,交给Spring管理
  • AOP编程的支持
  • Spring提供面向切面编程,可以方便的实现对程序进行权限拦截、运行监控等功能
  • 声明式事务的支持
  • 只需要通过配置就可以完成对事务的管理,而无需手动编程
  • 方便程序的测试
  • Spring对Junit4支持,可以通过注解方便的测试Spring程序
  • 方便集成各种优秀框架
  • Spring不排斥各种优秀的开源框架,其内部提供了对各种优秀框架(如:Struts2、Hibernate、MyBatis、Quartz等)的直接支持
  • 降低JavaEE API的使用难度
  • Spring 对JavaEE开发中非常难用的一些API(JDBC、JavaMail、远程调用等),都提供了封装,使这些API应用难度大大降低

2, Spring框架的版本

  • Spring3.x和Spring4.x的版本

技术分析之Spring框架的IOC核心功能快速入门
1, 什么是IOC的功能?

  • IoC -- Inverse of Control,控制反转,将对象的创建权反转给Spring!!
  • 使用IOC可以解决的程序耦合性高的问题!!

2, 步骤一:下载Spring框架的开发包

3, 步骤二:创建JavaWEB项目,引入Spring的开发包

  • 引入Spring框架IOC核心功能需要的具体的jar包
  • Spring框架的IOC的功能,那么根据Spring框架的体系结构图能看到,只需要引入如下的jar包
      - Beans
      - Core
      - Context
      - Expression Language
    如下:
    spring-expression-4.2.4.RELEASE.jar
    spring-beans-4.2.4.RELEASE.jar
    spring-context-4.2.4.RELEASE.jar
    spring-core-4.2.4.RELEASE.jar
  • Spring框架也需要引入日志相关的jar包

  • spring-framework-3.0.2.RELEASE-dependencies/org.apache.commons/com.springsource.org.apache.commons.logging/1.1.1
    com.springsource.org.apache.commons.logging-1.1.1.jar

  • 还需要引入log4j的jar包 spring-framework-3.0.2.RELEASE-dependencies\org.apache.log4j\com.springsource.org.apache.log4j\1.2.15

  • com.springsource.org.apache.log4j-1.2.15.jar

  • src目录下引入log4j的配置文件log4j.properties

### direct log messages to stdout ###
log4j.appender.stdout=org.apache.log4j.ConsoleAppender
log4j.appender.stdout.Target=System.err
log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n
### direct messages to file mylog.log ###
log4j.appender.file=org.apache.log4j.FileAppender
log4j.appender.file.File=c\:mylog.log
log4j.appender.file.layout=org.apache.log4j.PatternLayout
log4j.appender.file.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n
### set log levels - for more verbose logging change 'info' to 'debug' ###
log4j.rootLogger=info, stdout
  • Log4j测试
public class Demo1 {
    private Logger log = Logger.getLogger(Demo1.class);
    @Test
    public void Test()
    {
        System.out.println("执行。。。");
        log.info("执行。。。");
    }
}

4, 步骤三:创建对应的包结构,编写Java的类,要注意:以后使用Spring框架做开发,都需要来编写接口与实现类!!

  • com.itcast.demo1
  • UserService     -- 接口
  • UserServiceImpl   -- 具体的实现类
package com.huachao.demo2;
public interface UserService {
    public void sayHello();
}
package com.huachao.demo2;
public class UserServiceImp implements UserService {
    @Override
    public void sayHello() {
        System.out.println("Hello Spring");
    }
}

5, 步骤四:想把UserServiceImpl实现类的创建交给Spring框架来管理,需要创建Spring框架的配置文件,完成配置

  • 在src目录下创建applicationContext.xml的配置文件,名称是可以任意的,但是一般都会使用默认名称!!

  • 引入spring的schema约束,需要先找到具体的约束头信息!!

  • spring-framework-3.2.0.RELEASE\docs\spring-framework-reference\html\xsd-config.html,找到:40.2.12 the beans schema

  • 具体的约束如下:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://www.springframework.org/schema/beans 
    http://www.springframework.org/schema/beans/spring-beans.xsd">
    <!-- id是自己起的 -->
    <bean id="userService" class="com.huachao.demo2.UserServiceImp"/>
</beans>
  • 完成UserService的配置
    ``
    <bean id="userService" class="com.itcast.demo1.UserServiceImpl"/>

6, 步骤五:编写测试程序,采用Spring框架的工厂方式来获取到UserService接口的具体实现类!!


    @Test
    public void Run1()
    {
//        UserServiceImp us = new UserServiceImp();
        UserService us = new UserServiceImp();
        us.sayHello();
    }
    @Test
    public void Run2()
    {
        ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
        //userService为bean标签的id
//        UserServiceImp us = (UserServiceImp) ac.getBean("userService");
        UserService us = (UserService) ac.getBean("userService");
        us.sayHello();
    }

入门总结之Spring框架中的工厂(了解)
1, ApplicationContext接口

  • 使用ApplicationContext工厂的接口,使用该接口可以获取到具体的Bean对象
  • 该接口下有两个具体的实现类
  • ClassPathXmlApplicationContext     -- 加载类路径下的Spring配置文件
  • FileSystemXmlApplicationContext     -- 加载本地磁盘下的Spring配置文件

2, BeanFactory工厂(是Spring框架早期的创建Bean对象的工厂接口)

  • 使用BeanFactory接口也可以获取到Bean对象
 public void run(){
 BeanFactory factory = new XmlBeanFactory(new ClassPathResource("applicationContext.xml"));
 UserService us = (UserService) factory.getBean("us");
 us.sayHello();
 }
  • BeanFactory和ApplicationContext的区别
  • BeanFactory     -- BeanFactory采取延迟加载,第一次getBean时才会初始化Bean
  • ApplicationContext   -- 在加载applicationContext.xml时候就会创建具体的Bean对象的实例,还提供了一些其他的功能
      - 事件传递
      - Bean自动装配
      - 各种不同应用层的Context实现

入门总结之配置Spring框架编写XML的提示

  1. 步骤一:先复制, http://www.springframework.org/schema/beans/spring-beans.xsd
  2. 步骤二:打开Windows->Preferences,搜索XML Catalog,点击Add按钮
  3. 步骤三:先选择Location的schema的约束地址
  • E:\class\2016\JavaEE28\day35_Spring框架第一天\资料\spring-framework-4.2.4.RELEASE-schema\beans\spring-beans-4.2.xsd
  1. 步骤四:注意:Key type要选择:Schema location(这一步和之前不一样,之前选择的是URI
  2. 步骤五:Key把http://www.springframework.org/schema/beans/spring-beans.xsd复制上

技术分析之Spring框架的Bean管理的配置文件方式


技术分析之Spring框架中<bean>标签的配置
1, id属性和name属性的区别

  • id    -- Bean起个名字,在约束中采用ID的约束,唯一

  • 取值要求:必须以字母开始,可以使用字母、数字、连字符、下划线、句话、冒号 id:不能出现特殊字符

  • name   -- Bean起个名字,没有采用ID的约束(了解)

  • 取值要求:name:出现特殊字符.如果<bean>没有id的话 , name可以当做id使用

  • Spring框架在整合Struts1的框架的时候,Struts1的框架的访问路径是以/开头的,例如:/bookAction

2, class属性     -- Bean对象的全路径
3, scope属性     -- scope属性代表Bean的作用范围

  • singleton     -- 单例(默认值)
  • prototype     -- 多例,在Spring框架整合Struts2框架的时候,Action类也需要交给Spring做管理,配置把Action类配置成多例!!
  • request     -- 应用在Web项目中,每次HTTP请求都会创建一个新的Bean
  • session     -- 应用在Web项目中,同一个HTTP Session 共享一个Bean
  • globalsession    -- 应用在Web项目中,多服务器间的session

4, Bean对象的创建和销毁的两个属性配置(了解)

  • 说明:Spring初始化bean或销毁bean时,有时需要作一些处理工作,因此spring可以在创建和拆卸bean的时候调用bean的两个生命周期方法
  • init-method    -- 当bean被载入到容器的时候调用init-method属性指定的方法
  • destroy-method   -- 当bean从容器中删除的时候调用destroy-method属性指定的方法
  • 想查看destroy-method的效果,有如下条件
  • scope= singleton有效
  • web容器中会自动调用,但是main函数或测试用例需要手动调用(需要使用ClassPathXmlApplicationContext的close()方法)

例:演示init-method和destroy-method
UserServiceImp.java

package com.huachao.demo2;
public class UserServiceImp implements UserService {
    @Override
    public void sayHello() {
        System.out.println("Hello Spring");
    }
    public void init() {
        System.out.println("init....");
    }
    public void destory() {
        System.out.println("destory....");
    }
}

applicationContext.xml的bean配置

<bean id="userService" class="com.huachao.demo2.UserServiceImp" init-method="init" destroy-method="destory"/>

测试代码


    /**
     * 演示调用销毁方法
     */
    @Test
    public void Run3()
    {
        ClassPathXmlApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
        //userService为bean标签的id
//        UserServiceImp us = (UserServiceImp) ac.getBean("userService");
        UserService us = (UserService) ac.getBean("userService");
        us.sayHello();
        ac.close();
    }

技术分析之依赖注入(DI)

  1. IOC和DI的概念
  • IOC   -- Inverse of Control,控制反转,将对象的创建权反转给Spring!!
  • DI    -- Dependency Injection,依赖注入,在Spring框架负责创建Bean对象时,动态的将依赖对象注入到Bean组件中!!
  1. DI(依赖注入)
  • 例如:如果UserServiceImpl的实现类中有一个属性,那么使用Spring框架的IOC功能时,可以通过依赖注入把该属性的值传入进来!!
  • 具体的实现如下
public class UserServiceImp implements UserService {
    private String name;
    public void setName(String name) {
        this.name = name;
    }
    @Override
    public void sayHello() {
        System.out.println("Hello Spring,"+name);
    }
    public void init() {
        System.out.println("init....");
    }
    public void destory() {
        System.out.println("destory....");
    }
}
    <bean id="userService" class="com.huachao.demo2.UserServiceImp">
        <property name="name" value="美美"/>
    </bean>

例:注入对象
CustomerDao.java

package com.huachao.demo3;
public class CustomerDao {
    public void save()
    {
        System.out.println("Dao 的保存。。。");
    }
}

CustomerService.java

package com.huachao.demo3;
public class CustomerService {
    private CustomerDao customerDao;
    public void setCustomerDao(CustomerDao customerDao) {
        this.customerDao = customerDao;
    }

    public void save()
    {
        System.out.println("Service 的保存。。。");
        customerDao.save();
    }
}

applicationContext.xml配置的bean


    <bean id="customerDao" class="com.huachao.demo3.CustomerDao"/>
    <bean id="customerService" class="com.huachao.demo3.CustomerService">
        <property name="customerDao" ref="customerDao"/>
    </bean>

测试代码

@Test
    public void Run1()
    {
        ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
        CustomerService cs = (CustomerService) ac.getBean("customerService");
        cs.save();
    }

技术分析之Spring框架的属性注入
1,对于类成员变量,常用的注入方式有两种

  • 构造函数注入
  • 属性setter方法注入

2,在Spring框架中提供了前两种的属性注入的方式
①构造方法的注入方式,两步

  • 编写Java的类,提供构造方法
 public class Car {
 private String name;
 private double money;
 public Car(String name, double money) {
 this.name = name;
 this.money = money;
 }
 @Override
 public String toString() {
 return "Car [name=" + name + ", money=" + money + "]";
 }
 }
  • 编写配置文件
 <bean id="car" class="com.itheima.demo4.Car">
 <constructor-arg name="name" value="大奔"/>
 <constructor-arg name="money" value="100"/>
 </bean>

② 属性的setter方法的注入方式

  • 编写Java的类,提供属性和对应的set方法即可
  • 编写配置文件

③如果Java类的属性是另一个Java的类,那么需要怎么来注入值呢?

  • <property name="name" rel="具体的Bean的ID或者name的值"/>
  • 例如:
 <bean id="person" class="com.itheima.demo4.Person">
 <property name="pname" value="美美"/>
 <property name="car2" ref="car2"/>
 </bean>

例:构造函数注入和setter方法注入
car实例类,有带参数的构造函数

package com.huachao.demo4;
public class Car {
    private String cname;
    private Integer price;
    public Car(String cname, Integer price) {
        this.cname = cname;
        this.price = price;
    }
    @Override
    public String toString() {
        return "Car [cname=" + cname + ", price=" + price + "]";
    }
}

person实例类,有属性的setter方法

package com.huachao.demo4;
public class Person {
    private String pname;
    private Car car;
    public String getPname() {
        return pname;
    }
    public void setPname(String pname) {
        this.pname = pname;
    }
    public Car getCar() {
        return car;
    }
    public void setCar(Car car) {
        this.car = car;
    }
    @Override
    public String toString() {
        return "Person [pname=" + pname + ", car=" + car + "]";
    }
}

applicationContext.xml中bean的注册

    <bean id="car" class="com.huachao.demo4.Car">
        <!-- <constructor-arg name="cname" value="马自达"/>
        <constructor-arg name="price" value="25000"/> -->
        <constructor-arg index="0" value="马自达"/>
        <constructor-arg index="1" value="25000"/>
    </bean>
    <bean id="person" class="com.huachao.demo4.Person">
        <property name="pname" value="小小"/>
        <property name="car" ref="car"/>
    </bean>

测试代码

@Test
    public void Run1()
    {
        ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
        Person person =  (Person) ac.getBean("person");
        System.out.println(person.toString());
    }

技术分析之Spring的2.5版本中提供了一种:p名称空间的注入(了解)

  1. 步骤一:需要先引入 p 名称空间
  • 在schema的名称空间中加入该行:xmlns:p="http://www.springframework.org/schema/p"
  1. 步骤二:使用p名称空间的语法
  • p:属性名 = ""
  • p:属性名-ref = ""
  1. 步骤三:测试
  • <bean id="person" class="com.itheima.demo4.Person" p:pname="老王" p:car2-ref="car2"/>

技术分析之Spring的3.0提供了一种:SpEL注入方式(了解)
1, SpEL:Spring Expression Language是Spring的表达式语言,有一些自己的语法
2, 语法

  • #{SpEL}

3, 例如如下的代码

 <!-- SpEL的方式 -->
 <bean id="person" class="com.itheima.demo4.Person">
 <property name="pname" value="#{'小风'}"/>
 <property name="car2" value="#{car2}"/>
 </bean>

<font color=red>注意:</font>详细用法参考Spring_表达式语言.pdf

4, 还支持调用类中的属性或者方法

  • 定义类和方法,例如
 public class CarInfo {
 public String getCarname(){
 return "奇瑞QQ";
 }
 }

例:P名称空间注入
Car2实例类,有setter方法

package com.huachao.demo4;
public class Car2 {
    private String cname;
    private Integer price;
    public void setCname(String cname) {
        this.cname = cname;
    }
    public void setPrice(Integer price) {
        this.price = price;
    }
    @Override
    public String toString() {
        return "Car2 [cname=" + cname + ", price=" + price + "]";
    }
}

applicationContext.xml中beans标签是添加xmlns:p="http://www.springframework.org/schema/p"

<?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:p="http://www.springframework.org/schema/p"
    xsi:schemaLocation="http://www.springframework.org/schema/beans 
    http://www.springframework.org/schema/beans/spring-beans.xsd">

applicationContext.xml中添加bean

<bean id="car2" class="com.huachao.demo4.Car2" p:cname="长安福特" p:price="28000"/>

测试代码


    @Test
    public void Run2()
    {
        ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
        Car2 c2 =  (Car2) ac.getBean("car2");
        System.out.println(c2.toString());
    }

例:SpEL表达式注入
使用SpEL表达式表达式注入上例的Car2,xmlns:p="http://www.springframework.org/schema/p"不需要可以删除
applicationContext.xml中bean如下


    <bean id="car2" class="com.huachao.demo4.Car2">
        <property name="cname" value="#{'宝马'}"/>
        <property name="price" value="#{320000}"/>
    </bean>

技术分析之数组,集合(List,Set,Map),Properties等的注入
1, 如果是数组或者List集合,注入配置文件的方式是一样的

 <bean id="collectionBean" class="com.itheima.demo5.CollectionBean">
 <property name="arrs">
 <list>
 <value>美美</value>
 <value>小风</value>
 </list>
 </property>
 </bean>

2, 如果是Set集合,注入的配置文件方式如下:

 <property name="sets">
 <set>
 <value>哈哈</value>
 <value>呵呵</value>
 </set>
 </property>

3, 如果是Map集合,注入的配置方式如下:

 <property name="map">
 <map>
 <entry key="老王2" value="38"/>
 <entry key="凤姐" value="38"/>
 <entry key="如花" value="29"/>
 </map>
 </property>

4, 如果是properties属性文件的方式,注入的配置如下:

 <property name="pro">
 <props>
 <prop key="uname">root</prop>
 <prop key="pass">123</prop>
 </props>
 </property>

例:注入集合
User实例类中有各种集合,已经提供setter方法

package com.huachao.demo5;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
public class User {
    private String[] arr;
    private List<String> list;
    private Set<String> set;
    private Map<String,String> map;
    private Properties pro;
    public void setArr(String[] arr) {
        this.arr = arr;
    }
    public void setList(List<String> list) {
        this.list = list;
    }
    public void setSet(Set<String> set) {
        this.set = set;
    }
    public void setMap(Map<String, String> map) {
        this.map = map;
    }
    public void setPro(Properties pro) {
        this.pro = pro;
    }
    @Override
    public String toString() {
        return "User [arr=" + Arrays.toString(arr) + ", list=" + list + ", set=" + set + ", map=" + map + ", pro=" + pro
                + "]";
    }
}

applicationContext.xml中注入User的bean如下

<bean id="user" class="com.huachao.demo5.User">
        <property name="arr">
            <list>
                <value>数组元素1</value>
                <value>数组元素2</value>
            </list>
        </property>
        <property name="list">
            <list>
                <value>list元素1</value>
                <value>list元素2</value>
            </list>
        </property>
        <property name="set">
            <set>
                <value>set元素1</value>
                <value>set元素2</value>
            </set>
        </property>
        <property name="map">
            <map>
                <entry key="key1" value="map的value1"/>
                <entry key="key2" value="map的value2"/>
            </map>
        </property>
        <property name="pro">
            <props>
                <prop key="Properties key1">Properties value1</prop>
                <prop key="Properties key2">Properties value2</prop>
            </props>
        </property>
    </bean>

技术分析之Spring框架的配置文件分开管理(了解)
1, 例如:在src的目录下又多创建了一个配置文件,现在是两个核心的配置文件,那么加载这两个配置文件的方式有两种!

  • 主配置文件中包含其他的配置文件:
    <import resource="applicationContext2.xml"/>

  • 工厂创建的时候直接加载多个配置文件:
    ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml","applicationContext2.xml");


Spring框架整合WEB


Spring框架整合WEB(不是最终的方案)
1, 创建JavaWEB项目,引入Spring的开发包。编写具体的类和方法。

  • 环境搭建好后,启动服务器来测试项目,发送每访问一次都会加载一次配置文件,这样效率会非常非常慢!!

2, 解决上面的问题

  • 将工厂创建好了以后放入到ServletContext域中.使用工厂的时候,从ServletContext中获得.
  • ServletContextListener:用来监听ServletContext对象的创建和销毁的监听器.
  • 当ServletContext对象创建的时候:创建工厂 , 将工厂存入到ServletContext

3, Spring整合Web项目

  • 引入spring-web-4.2.4.RELEASE.jar包
  • 配置监听器
  <!-- 配置Spring的核心监听器: -->
  <listener>
   <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
  </listener>
  <context-param>
   <param-name>contextConfigLocation</param-name>
   <param-value>classpath:applicationContext.xml</param-value>
  </context-param>

4, 修改servlet的代码

  • 从ServletContext中获得工厂
  • 具体代码如下
 ServletContext servletContext = ServletActionContext.getServletContext();
 // 需要使用WEB的工厂的方式
 WebApplicationContext context = WebApplicationContextUtils.getWebApplicationContext(servletContext);
 CustomerService cs = (CustomerService) context.getBean("customerService");
 cs.save(); 

注意:
1,主要用于解决struts2和spring整合的过程中,在Action中获取Spring的工厂对象,不用每次都加载资源。
2,具体代码参考day35_crm


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

推荐阅读更多精彩内容