1.3 Spring入门

Spring全家桶

  • Spring Framework
    核心
  • Spring Boot
    构建项目,在其基础上开发更容易更方便
  • Spring Cloud
    微服务,当一个规模较大的项目维护起来比较困难时,可以利用spring cloud将项目拆分成若干个子项目,然后把他们再集成在一起,每一个子项目规模小,就便于维护,但是开发难度会有所提高
  • Spring Cloud Data Flow
    是spring做数据集成的功能,比如一个应用有很多客户端,包括pc端、移动端、可穿戴设备、汽车、传感器等,这些客户端所采集的数据形态各异,怎样把这些数据集成整合在一起形成更有价值的数据就是Spring Cloud Data Flow所作的事情

此项目不会把这四大功能都用上,项目本身就是某网的一个模块,没有必要往下拆分,微服务就不用去做了,同时这个项目就只做浏览器客户端,用不到data flow。只用到spring frameworkspring boot。spring全家桶的使用手册在https://spring.io上可以查到

Spring具体介绍:

1.Spring Framework

本项目中用到的四大部分

  • Spring Core
    • IoC(用来管理Bean的思想,面向对象的)、AOP (面向界面)
    • 可以说spring的其他所有功能都是以Ioc为基础的,因此要先学好Ioc
    • 有了这两种思想能够管理一切Bean,哪怕把一个第三方的框架拿过来,它也能对其进行整合。spring厉害就厉害在它并非功能多,而是它这两种管理Bean的思想能够管理一切。
  • Spring Data Access (spring访问数据库的功能)
    • Transactions(管理事务)、Spring MyBatis(整合mybatis)
  • Web Servlet (web开发)
    • Spring MVC
  • Integration (集成)
    • Email(发邮件)、Scheduling(定时任务)、AMQP(消息队列发送消息)、Security(安全控制、权限管理)
Snipaste_2020-04-08_18-14-47.png

2.Spring IoC

  • Inversion of Control
    • 控制反转,是一种面向对象编程的设计思想。
      比如,当我们自己管理对象时,通常是new A,new B,A.set传入B,这样A和B就有了联系,能够通过A去调用B,这种做法比较直观,但是缺点是A与B产生了耦合,直接耦合在一起,当项目规模比较大时,可能就会不容易去维护,若想对A做出更改,则需要把B也更改;那么IoC思想是通过别的办法简化Bean之间的关系,从而减少Bean之间的耦合度,让项目能够便于维护。
      实现这种思想的方式:依赖注入。
      依赖注入的实现基于IoC容器。
  • Dependency Injection
    • 依赖注入,是IoC思想的实现方式。
  • IoC Container
    • IoC容器,是实现依赖注入的关键,本质上是一个工厂。


      演示IoC容器的原理

代码示例:

类CommunityApplication的代码如下

package com.nowcoder.community;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
public class CommunityApplication {

    public static void main(String[] args) {
        SpringApplication.run(CommunityApplication.class, args);
    }

}

下面解释一下这个类运行时做了哪些事情:

  • main方法只有一句话:运行Spring应用
  • 底层启动了tomcat,但是底层不只是启动了tomcat,也自动创建了spring容器。web项目当中spring容器不需要我们主动去创建而是被自动创建的
  • spring容器被创建以后,它会自动去扫描某些包下的某些bean,将这些bean装配到容器里,至于哪些bean会被扫描到,需要看配置类
  • 由代码SpringApplication.run(CommunityApplication.class, args);知,这个类CommunityApplication是run方法的参数,spring应用在启动的时候是需要配置的,因此这个类其实就是一个配置文件
  • 注解@SpringBootApplication所标识的类就表明这个类就是一个配置文件
    按Ctrl并点击该注解可以进入注解底层查看,这个注解是由其他注解所组成的
    如下图:
    @SpringBootApplication注解底层.png

    src.png

对注解@ComponentScan注解的进一步说明:
CommunityApplication这个配置类所在的包com.nowcoder.community和这个包的子包controller,这些包中的bean都会被扫描到,但是bean上需要有@controller注解才能够被扫描。

@Controller出现的地方

@Contoller功能相同的注解还有@Service@Component
说明:
navigate->classes(或者Ctrl+N) 搜索Service,打开后可以看到@Service是由@Component来实现的,因此注解用@Component也同样可以实现。@Controller也是由@Component实现的,@Repository也有相同的功能。

Service.class

总结:
@Controller
@Service
@Component
@Repository
四个注解都可用于将其加到某一类上,然后使这个类被容器所扫描。区别在于语义上的区别:
开发业务组件--@Service
处理请求--@Controller
数据库访问组件--@Repository
通用--@Component

代码演示IoC的使用方式:

在测试类中进行演示

test.png

然而IDEA测试类使用@RunWith注解时发现并没有这个注解,此时查看导入的test依赖(pom.xml)

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
            <exclusions>
                <exclusion>
                    <groupId>org.junit.vintage</groupId>
                    <artifactId>junit-vintage-engine</artifactId>
                </exclusion>
            </exclusions>
        </dependency>

解决办法:把代码删掉即可正常使用,既导入的test依赖如下

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

保存后重新在CommunityApplicationTests.java中引用注解
@RunWith(SpringRunner.class)

在测试类中若想引用正式类中的配置类,则需要引入注释
@ContextConfiguration(classes = CommunityApplication.class)
那么测试代码则能以CommunityApplication.class为配置类进行配置。

IoC是个自动创建的核心Spring容器,那么如何去得到这个容器呢:哪个类想得到Spring容器,就去实现这样一个接口:在类名后添加implements ApplicationContextAware

implements.png

set方法,传入一个参数叫做applicationContext,这个参数实际就是Spring容器对象(对applicationContext的理解详见文章Spring容器和应用上下文)应用上下文,可以简单的理解成就是将你需要Spring帮你管理的对象放入容器的一种容器对象,本质上说就是一个维护Bean定义以及对象之间协作关系的高级接口。Ctrl+click查看applicationContext底层

application继承.png

同样再查看ListableBeanFactory和HierarchicalBeanFactory底层,发现他们都继承于BeanFactory
HierarchicalBeanFactory.png

ListableBeanFactory.png

由此可见,applicationContext是继承了BeanFactory后派生而来的,而BeanFactory就是Spring容器的顶层接口,ApplicationContext接口,它由BeanFactory接口派生而来,ApplicationContext包含BeanFactory的所有功能,功能更强大,通常建议比BeanFactory优先使用。

如果一个类实现了ApplicationContextAware接口,Spring容器会自动调用ApplicationContextAware接口中的setApplicationContext方法,在当spring容器初始化的时候,会自动的将ApplicationContext注入进来,通过这个上下文环境对象(ApplicationContext)得到Spring容器中的Bean

为了暂存spring容器,在测试类当中加入一个成员变量用于记录容器

private ApplicationContext applicationContext;

因而当程序运行的时候,application就可以被自动传进来并被记录,那么就可以使用在别的地方了。
test容器.png

下面写一个测试方法,在测试方法中去使用spring容器,代码如下:

    @Test
    public void testApplicationContext() {
        System.out.println(applicationContext);
    }

然后运行testApplicationContext(),报错,原因是test包导错了,在@Test有两个包,一个是org.junit.jupiter.api.Test,另一个是org.junit.Test,而测试需要的Junit是org.junit.Test。因此将 import org.junit.jupiter.api.Test; 替换为 import org.junit.Test,重新运行。

org.junit.test.png

运行结束后查看对象的输出形式,如图
输出形式.png

GenericWebApplicationContext@5e3d57c7
对象的输出形式是:类名@hashcode
总之就是证明了这个容器是存在、可见的;

下面演示如何去用这个容器来管理bean,那么首先得有bean让其去管理,写一个bean。新建一个包,用于存放访问数据库的bean,给这个包命名为dao(data access object 数据库访问对象)
然后在dao包下创建访问数据库的接口new->java class->选择Interface()接口类


Interface.png

接口中需要定义一些方法

public interface AlphaDao {

    String select();
    //定义select查询方法,没有参数,返回一个字符串
}

接口的实现需要有实现类,那么再创建一个该接口的实现类AlphaDaoHibernateImpl,命名解释:这个实现类采用Hibernate技术来实现(implement)查询。接下来在这个类中实现接口:implements AlphaDao,然后Ctrl+I增加接口中的方法。实现类创建情况如下图

实现类.png

但是这个bean还不能被容器管理,或者说容易还不能扫描并装配这个bean,因此需要引入前面提到的访问数据库的注解@Repository,如此spring容器就可以自动扫描到这个bean然后把它装配到容器里。

回到测试类中,

    @Test
    public void testApplicationContext() {
        System.out.println(applicationContext);

        AlphaDao alphaDao = applicationContext.getBean(AlphaDao.class);  //getBean()可以通过名字或类型获取,这里是通过类型获取
        //从容器里获取自动装配的bean
        System.out.println(alphaDao.select());  //调用查询方法并将结果输出出来
    }

运行结果如图:


运行结果.png

成功返回了Hibernate,说明成功地获取到了bean(AlphaDaoHibernateImpl.java)并得到了查询的结果。

下面再创建一个bean,通过这个例子来体现spring容器的优势。比如当项目发展到一定阶段,引入了新技术Mybatis,假设mybatis比Hibernate更有优势,想要把项目中的Hibernate替换成mybatis,做法是在dao项目中创建一个新的实现类AlphaDaoMyBatisImpl,同样去实现接口即implements AlphaDao,并增加接口待实现方法,返回"MyBatis"字符串,同时加上注解@Repository

MyBatis.png

但是若这时候去运行测试中的testApplicationContext()方法还存在问题,问题在于getBean()是要按照类型获取bean,如:getBean(AlphaDao.class),这个括号中的类型是一个接口 ,那么这个接口满足条件的bean有两个--AlphaDaoMyBatisImpl和AlphaDaoHibernateImpl,那么spring容器就不知道该用哪个bean了。解决办法:在要用到的bean上加上注解@Primary,即注明这个bean有更高的优先级。(这里将AlphaDaoMyBatisImpl定为最高优先级)

运行testApplicationContext(),运行成功后返回了MyBatis


结果返回MyBatis.png

也就是说,原先测试程序当中dao的实现类都是由HibernateImpl来实现的,当需要替换掉bean的时候,只需要再创建/添加一个bean,并在此新添加bean上加一个@Primary注解就可以了,测试程序中调用的地方并不需要发生变化,因为调用的不是bean本身而是依赖bean的接口,因此即使是实现类变了,调用代码本身是不用改的。因此,spring容器就是用这种方法降低了bean之间的耦合度,调用方和实现类不会发生任何直接的关系。

但是新的问题是若程序某一个程序块仍想用Hibernate而不想用mybatis,如何去实现呢?bean默认的名字是类名(首字母小写),当然也可以自定义bean的名字,在bean必须有的注解后边用("字符串")的形式另外命名,例如@Repository("alphaHibernate"),那么这个bean的名字就命名为了alphaHibernate。当出现一个程序中想调用两个bean的时候就通过名字强制容器返回非primary的bean。

    @Test
    public void testApplicationContext() {
        System.out.println(applicationContext);

        AlphaDao alphaDao = applicationContext.getBean(AlphaDao.class);  //getBean()可以通过名字或类型获取,这里是通过类型获取
        //从容器里获取自动装配的bean
        System.out.println(alphaDao.select());  //调用查询方法并将结果输出出来


        //下面面重新获取一次bean
        alphaDao = applicationContext.getBean("alphaHibernate", AlphaDao.class);
        //通过名字而非类型获取bean的默认返回类型为object,因此需要强制转型,或者再加一个类型参数(如上)
        //意在说明将object类型转换为AlphaDao.class
        System.out.println(alphaDao.select());
    }

运行结果:


两个bean.png

以上就是我们从容器中获取bean的基本方式。

那么容器管理bean不仅是能帮我们创建bean,还有更多管理bean的手段,下面演示spring容器管理bean更多的方法。

spring容器除了创建bean还能帮助管理bean的初始化以及销毁bean。先创建一个新的包service用于存放开发的业务组件,然后在此包中创建类AlphaService。
在具体Bean的实例化过程中,@PostConstruct注释的方法,会在构造方法之后,init方法之前进行调用。为了便于观察@PostConstruct注释的方法调用的时刻是否是在构造方法之后,再给类AlphaService加一个构造方法AlphaService进行对比。除了管理初始化方法还能管理销毁方法,那么再给这个类加一个销毁方法。

@Service
public class AlphaService {

    public AlphaService() {
        System.out.println("实例化AlphaService");
    }

    @PostConstruct
    //要想让容器帮助管理这个方法,就是让容器在合适的时候自动调用这个方法,那么就加上这个注解
    //@PostConstruct注释的方法,会在构造方法之后,init方法之前进行调用
    public void init() {
        System.out.println("初始化AlphaService");
    }

    @PreDestroy
    //当Bean在容器销毁之前,调用被@PreDestroy注解的方法
    public void destroy() {
        System.out.println("销毁AlphaService");
    }
}

进行测试,看一看能不能通过容器自动调用初始化以及销毁的方法。回到测试类,再写一个新的测试方法。

    @Test
    public void testBeanManagement() {//测试bean管理的方式
        //通过容器去获取Service,看会出现什么样的情况
        AlphaService alphaService = applicationContext.getBean(AlphaService.class);
        System.out.println(alphaService);
    }

运行testBeanManagement() 方法,结果如下图:


testBeanManagement.png

看控制台,最终确实打印出了该对象,说明这个bean可以被实例化。再看程序启动过程输出的内容,包括spring容器创建的过程,在这个过程当中会看到这样的输出:


实例化的过程.png
销毁方法.png

可以说明bean的初始化方法确实是在构造器之后被调用的,销毁方法是在对象销毁之前(程序结束之前)调用的。在初始化和销毁之间打印出了bean的实例。

在程序启动时bean被实例化,在程序停止的时候bean被销毁,说明bean只被实例化一次、只被销毁一次,在容器中只有一个实例,即bean是单例。因为main方法只启动一次,随意bean只被实例化一次,那就只能有一个实例。下面再通过测试来验证这个说法:再来获取一下bean并打印,看最终实例化了几次。

    @Test
    public void testBeanManagement() {//测试bean管理的方式
        //通过容器去获取Service,看会出现什么样的情况
        AlphaService alphaService = applicationContext.getBean(AlphaService.class);
        System.out.println(alphaService);


        //再来获取以下bean并打印,看最终实例化了两次还是一次
        alphaService = applicationContext.getBean(AlphaService.class);
        System.out.println(alphaService);
    }

控制台打印结果为:


控制台打印结果.png

可见两次输出hashcode是相同的,因此是一个对象。

实例化过程日志.png

日志显示也只实例化了一次,说明被spring容器管理的bean默认是单例的。

若想要每次getBean()的时候都能新建一个实例(多例),就需要在bean上加一个注解@Scope("XXX") Scope的默认参数是singleton(表示单例),prototype(多例),当加上这个表示多例的注解后,每次访问bean就会给我们创建一个新的实例。

因此在AlphaService中加入@Scope("prototype")

Scope.png

回到测试类中,运行testBeanManagement(),控制台输出结果如下:


多例 控制台.png

这次并不是启动时初始化,而是每次getBean()都实例化一个对象,实例化两次就是两个bean,hashcode也不同。但是通常很少使用多例的形式。

以上用spring容器管理的都是我们自己写的bean,若在容器中装配第三方bean,就不能在这个第三方包中随意加注解(总是就是不能更改别人写的包),因此这时候就需要自己写一个配置类,在配置类当中通过bean注解进行声明,从而来解决这个问题。下面演示装配第三方bean、自己写配置类的操作方法:
随着项目的展开,以后会有更多的配置类将会放到同一个包下。新建一个包命名为config(配置),此包下创建一个配置类实例AlphaConfig.java。标识注解@Configration表示这个类是一个配置类不是普通类。要定义第三方bean则需要加入注解@Bean,同时定义方法。

@Configuration
public class AlphaConfig {

    @Bean
    //SimpleDateFormat方法返回的类型就是SimpleDateFormat,实例化一次就可以调用多次
    //注意:方法名simpleDateFormat就是Bean的名字
    public SimpleDateFormat simpleDateFormat() {
        return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); 
        //实例化simpledateformat,并给它指定一个格式
    }
}
//这段话的意思就是这个方法返回的对象将被装配到容器中,这个bean的名字就是simpleDateFormat。

回到测试类中,重新写一段测试代码:

    @Test
    public void testBeanConfig() {
        SimpleDateFormat simpleDateFormat = applicationContext.getBean(SimpleDateFormat.class);
        System.out.println(simpleDateFormat.format(new Date()));    //格式化当前日期
    }

运行结果:


输出当前的年月日时分秒.png

spring容器基本就是这样去使用的,但以上这些方法都是通过我们主动操作去获取容器中的bean,但是这种方法较为麻烦,spring还有更简便的使用容器的方式。即主动获取。通过以上比较笨拙的方式去理解了spring容器是什么、它的底层有什么方法、怎样直接去使用它。
事实上在日常开发使用spring容器过程中是通过依赖注入的方式以一种更便捷的方式去使用spring容器获取bean。

下面演示的过程用于说明什么是依赖注入。若当前的bean要使用AlphaDao,则没有必要去通过容器的get方法获取,而只需要声明给当前的bean注入AlphaDao就可以,注入要使用注解@Autowired,后边紧接成员变量

@Autowired的解释:

掘金--彻底搞明白Spring中的自动装配和Autowired
从Spring2.5开始,开始支持使用注解来自动装配Bean的属性。它允许更细粒度的自动装配,我们可以选择性的标注某一个属性来对其应用自动装配。
Spring支持的几种不同的应用于自动装配的注解中,Spring自带的@Autowired注解是其中的一种。
使用@Autowired很简单,在需要注入的属性加入注解即可。

@Autowired
UserService userService;

那么Autowired是按照什么策略来自动装配的呢?
关于这个问题,不能一概而论,你不能简单的说按照类型或者按照名称。但可以确定的一点的是,它默认是按照类型来自动装配的,即byType。
@Autowired默认使用byType来装配属性,如果匹配到类型的多个实例,再通过byName来确定Bean。

对自动装配的解释:

Spring提供了4种自动装配策略
1、byName
它的意思是,把与Bean的属性具有相同名字的其他Bean自动装配到Bean的对应属性中。
2、byType
如果不使用属性名称来对应,你也可以选择使用类型来自动装配。它的意思是,把与Bean的属性具有相同类型的其他Bean自动装配到Bean的对应属性中。
3、constructor
它是说,把与Bean的构造器入参具有相同类型的其他Bean自动装配到Bean构造器的对应入参中。值的注意的是,具有相同类型的其他Bean这句话说明它在查找入参的时候,还是通过Bean的类型来确定。
4、autodetect
它首先会尝试使用constructor进行自动装配,如果失败再尝试使用byType。不过,它在Spring3.0之后已经被标记为@Deprecated。
5、默认自动装配
默认情况下,default-autowire属性被设置为none,标示所有的Bean都不使用自动装配,除非Bean上配置了autowire属性。
如果你需要为所有的Bean配置相同的autowire属性,有个办法可以简化这一操作。
在根元素Beans上增加属性default-autowire="byType"。

测试类中代码如下:

    @Autowired
    private AlphaDao alphaDao;
    //让spring容器将AlphaDao注入给属性alphaDao,于是该属性便可直接使用

    @Autowired
    private AlphaService alphaService;

    @Autowired
    private SimpleDateFormat simpleDateFormat;

    @Test
    public void testDI() {  //DI--Dependency Injection 依赖注入
        //在测试方法中直接使用成员变量,看能不能直接取到这个bean
        System.out.println(alphaDao);
        System.out.println(alphaService);
        System.out.println(simpleDateFormat);
    }

运行testDI(),运行结果如下:


依赖注入.png

若想要alphaDao注入的不是默认优先级Mybatis而是Hibernate,就需要再加一个注解@Qualifier("alphaHibernate"),那么spring容器就会把名为alphaHibernate的bean注入给他。

spring以依赖注入的方式管理bean,只需要声明一个属性,写一个注解,bean就可以拿来用了。若bean依赖的是接口,也会降低耦合度。依赖注入的实现方式简单方便。注解也可以加在类构造器前,通过构造器注入;或者加到set方法之前,通过set方法注入;但通常都是加到属性之前。

那么在实际开发过程中怎样去运用spring容器的依赖注入思想?下面进行综合的演示。
开发过程中由controller来处理浏览器的请求,它在处理浏览器请求的过程中会调用业务组件去处理当前的业务,业务组件会调用dao(data access object)去访问数据库。即controller调用service,service调用dao,他们是互相依赖的,依赖关系就以依赖注入的方式来实现。
Service调用dao:

@Service
//@Scope("prototype")     //每次访问bean就会给创建一个新的实例(多例)
public class AlphaService {

    @Autowired
    private AlphaDao alphaDao;

    public String find() {
        return alphaDao.select(); //调用alphaDao来实现查询,直接返回查询结果
    }                             //Service依赖于dao的方式

}

controller调用service:

@Controller
@RequestMapping("/alpha")   //(“/alpha”)是给类取一个访问的名字,浏览器通过这个名字来访问类
                            //以上两个注解是spring MVC的注解
public class AlphaController {

    @Autowired
    private AlphaService alphaService;

    @RequestMapping("/data")
    @ResponseBody
    public String getData() {
        return alphaService.find(); //把find的结果返回给浏览器
        //浏览器访问这个方法的前提是得有注解声明
    }
}

之后运行正式的代码CommunityApplication,然后打开浏览器访问localhost:8080/alpha/data


浏览器.png

controller调用了service的find()方法,而find()调用的是dao的select()方法,dao默认注入的是优先级高的mybatis。

End

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

推荐阅读更多精彩内容