SpringBoot+Mybatis-多数据源动态切换+动态加载

问题引入:
公司云平台项目每个商户一个数据库,所以在写java领域层REST Server端的时候需要,根据应用层传递过来的商户id,进行动态切换数据;

一.普及知识

  • 一个数据源,也就代表一个数据库,=数据的源头
  • 数据源实例:一个数据库连接,就代表一个数据源实例对象;
  • 多数据源实例:多个数据库连接对象;

二.寻找解决办法

  • 我们的项目使用SpringBoot+Mybatis开发的领域层,默认只连接一个数据库;
  • 网上查询大部分的做法都是多数据源之间动态切换,也就是说在配置文件中提前配置好几个数据库连接信息,自己获取配置文件中的这些配置,然后在springBoot启动的使用想办法自动创建这 几个数据源实例
  • 在后续需要切换数据库的时候,只需要指定对应的数据源key,进行动态切换即可;
  • 可是我们的需求并不是这样的,我们需要根据外部的变量进行动态创建数据源实例,然后在切换到该数据源上
  • 对于多数据源的切换和加载,以下这篇文件讲的非常到位:
    基于Spring Boot实现Mybatis的多数据源切换和动态数据源加载
  • 所以我的项目主要需要解决的是多数据源动态加载,当然有了动态加载,动态切换就很简单了;

pom.xml需要添加

        <!-- 引入aop -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
        </dependency>
        <!-- druid数据源 -->
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid-spring-boot-starter</artifactId>
            <version>1.1.6</version>
        </dependency>

三.在 application.yml 中配置多个数据库连接信息如下:

db:
  default:
    #url: jdbc:mysql://localhost:3306/product_master?useSSL=false&useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&autoReconnect=true
    driver-class-name: com.mysql.jdbc.Driver
    url-base: jdbc:mysql://
    host: localhost
    port: 3306
    dbname: ljyun_share
    url-other: ?useSSL=false&useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&autoReconnect=true
    username: common
    password: common

  #蓝景商城数据库
  dbMall:
    driver-class-name: com.mysql.jdbc.Driver
    url-base: jdbc:mysql://
    host: localhost
    port: 3306
    dbname: db_mall
    url-other: ?useSSL=false&useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&autoReconnect=true
    username: common
    password: common
  #云平台私有库
  privateDB:
    driver-class-name: com.mysql.jdbc.Driver
    url-base: jdbc:mysql://
    host: localhost
    port: 3306
    dbname: ljyun_{id}_merchant
    url-other: ?useSSL=false&useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&autoReconnect=true
    username: common
    password: common

四.项目目录

多数据源-1.jpg

五.动态数据设置以及获取,本类属于单例;

  • DynamicDataSource 需要继承 AbstractRoutingDataSource
package domain.dbs;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;
import org.springframework.stereotype.Component;

import java.util.HashMap;
import java.util.Map;

/**
 * 动态数据设置以及获取,本类属于单例
 * @author lxf 2018-09-29
 */
@Component
public class DynamicDataSource extends AbstractRoutingDataSource {

    private final Logger logger = LoggerFactory.getLogger(getClass());
    //单例句柄
    private static DynamicDataSource instance;
    private static byte[] lock=new byte[0];
    //用于存储已实例的数据源map
    private static Map<Object,Object> dataSourceMap=new HashMap<Object, Object>();

    /**
     * 获取当前数据源
     * @return
     */
    @Override
    protected Object determineCurrentLookupKey() {
        logger.info("Current DataSource is [{}]", DynamicDataSourceContextHolder.getDataSourceKey());
        return DynamicDataSourceContextHolder.getDataSourceKey();
    }

    /**
     * 设置数据源
     * @param targetDataSources
     */
    @Override
    public void setTargetDataSources(Map<Object, Object> targetDataSources) {
        super.setTargetDataSources(targetDataSources);
        dataSourceMap.putAll(targetDataSources);
        super.afterPropertiesSet();// 必须添加该句,否则新添加数据源无法识别到
    }

    /**
     * 获取存储已实例的数据源map
     * @return
     */
    public Map<Object, Object> getDataSourceMap() {
        return dataSourceMap;
    }

    /**
     * 单例方法
     * @return
     */
    public static synchronized DynamicDataSource getInstance(){
        if(instance==null){
            synchronized (lock){
                if(instance==null){
                    instance=new DynamicDataSource();
                }
            }
        }
        return instance;
    }

    /**
     * 是否存在当前key的 DataSource
     * @param key
     * @return 存在返回 true, 不存在返回 false
     */
    public static boolean isExistDataSource(String key) {
        return dataSourceMap.containsKey(key);
    }
}

六.数据源配置类

  • DataSourceConfigurer 在tomcat启动时触发,在该类中生成多个数据源实例并将其注入到 ApplicationContext 中;
  • 该类通过使用 @Configuration@Bean 注解,将创建好的多数据源实例自动注入到 ApplicationContext上下中,供后期切换数据库用;
package domain.dbs;

import com.alibaba.druid.pool.DruidDataSource;
import org.apache.ibatis.session.SqlSessionFactory;
import org.mybatis.spring.SqlSessionFactoryBean;
import org.mybatis.spring.SqlSessionTemplate;
import org.mybatis.spring.boot.autoconfigure.MybatisProperties;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.env.Environment;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;
import org.springframework.transaction.PlatformTransactionManager;

import javax.sql.DataSource;
import java.util.HashMap;
import java.util.Map;

/**
 * 数据源配置类,在tomcat启动时触发,在该类中生成多个数据源实例并将其注入到 ApplicationContext 中
 * @author lxf 2018-09-27
 */

@Configuration
@EnableConfigurationProperties(MybatisProperties.class)
public class DataSourceConfigurer {
    //日志logger句柄
    private final Logger logger = LoggerFactory.getLogger(getClass());
    //自动注入环境类,用于获取配置文件的属性值
    @Autowired
    private Environment evn;

    private MybatisProperties mybatisProperties;
    public DataSourceConfigurer(MybatisProperties properties) {
        this.mybatisProperties = properties;
    }


    /**
     * 创建数据源对象
     * @param dbType 数据库类型
     * @return data source
     */
    private DruidDataSource createDataSource(String dbType) {
        //如果不指定数据库类型,则使用默认数据库连接
        String dbName = dbType.trim().isEmpty() ? "default" : dbType.trim();
        DruidDataSource dataSource = new DruidDataSource();
        String prefix = "db." + dbName +".";
        String dbUrl = evn.getProperty( prefix + "url-base")
                        + evn.getProperty( prefix + "host") + ":"
                        + evn.getProperty( prefix + "port") + "/"
                        + evn.getProperty( prefix + "dbname") + evn.getProperty( prefix + "url-other");
        logger.info("+++default默认数据库连接url = " + dbUrl);
        dataSource.setUrl(dbUrl);
        dataSource.setUsername(evn.getProperty( prefix + "username"));
        dataSource.setPassword(evn.getProperty( prefix + "password"));
        dataSource.setDriverClassName(evn.getProperty( prefix + "driver-class-name"));
        return dataSource;
    }

    /**
     * spring boot 启动后将自定义创建好的数据源对象放到TargetDataSources中用于后续的切换数据源用
     *             (比如:DynamicDataSourceContextHolder.setDataSourceKey("dbMall"),手动切换到dbMall数据源
     * 同时指定默认数据源连接
     * @return 动态数据源对象
     */
    @Bean
    public DynamicDataSource dynamicDataSource() {
        //获取动态数据库的实例(单例方式)
        DynamicDataSource dynamicDataSource = DynamicDataSource.getInstance();
        //创建默认数据库连接对象
        DruidDataSource defaultDataSource = createDataSource("default");
        //创建db_mall数据库连接对象
        DruidDataSource mallDataSource = createDataSource("dbMall");

        Map<Object,Object> map = new HashMap<>();
        //自定义数据源key值,将创建好的数据源对象,赋值到targetDataSources中,用于切换数据源时指定对应key即可切换
        map.put("default", defaultDataSource);
        map.put("dbMall", mallDataSource);
        dynamicDataSource.setTargetDataSources(map);
        //设置默认数据源
        dynamicDataSource.setDefaultTargetDataSource(defaultDataSource);

        return dynamicDataSource;
    }

    /**
     * 配置mybatis的sqlSession连接动态数据源
     * @param dynamicDataSource
     * @return
     * @throws Exception
     */
    @Bean
    public SqlSessionFactory sqlSessionFactory(
            @Qualifier("dynamicDataSource") DataSource dynamicDataSource)
            throws Exception {
        SqlSessionFactoryBean bean = new SqlSessionFactoryBean();
        bean.setDataSource(dynamicDataSource);
        bean.setMapperLocations(mybatisProperties.resolveMapperLocations());
        bean.setTypeAliasesPackage(mybatisProperties.getTypeAliasesPackage());
        bean.setConfiguration(mybatisProperties.getConfiguration());
        return bean.getObject();
    }
    @Bean(name = "sqlSessionTemplate")
    public SqlSessionTemplate sqlSessionTemplate(
            @Qualifier("sqlSessionFactory") SqlSessionFactory sqlSessionFactory)
            throws Exception {
        return new SqlSessionTemplate(sqlSessionFactory);
    }

    /**
     * 将动态数据源添加到事务管理器中,并生成新的bean
     * @return the platform transaction manager
     */
    @Bean
    public PlatformTransactionManager transactionManager() {
        return new DataSourceTransactionManager(dynamicDataSource());
    }
}

七.通过 ThreadLocal 获取和设置线程安全的数据源 key

  • DynamicDataSourceContextHolder类的实现
package domain.dbs;

/**
 * 通过 ThreadLocal 获取和设置线程安全的数据源 key
 */
public class DynamicDataSourceContextHolder {

    /**
     * Maintain variable for every thread, to avoid effect other thread
     */
    private static final ThreadLocal<String> contextHolder = new ThreadLocal<String>() {
        /**
         * 将 default 数据源的 key 作为默认数据源的 key
         */
//        @Override
//        protected String initialValue() {
//            return "default";
//        }
    };


    /**
     * To switch DataSource
     *
     * @param key the key
     */
    public static synchronized void setDataSourceKey(String key) {
        contextHolder.set(key);
    }

    /**
     * Get current DataSource
     *
     * @return data source key
     */
    public static String getDataSourceKey() {
        return contextHolder.get();
    }

    /**
     * To set DataSource as default
     */
    public static void clearDataSourceKey() {
        contextHolder.remove();
    }
}

八.AOP实现在DAO层做动态数据源切换(本项目没有用到

package domain.dbs;

/**
 * 动态数据源切换的切面,切 DAO 层,通过 DAO 层方法名判断使用哪个数据源,实现数据源切换
 * 关于切面的 Order 可以可以不设,因为 @Transactional 是最低的,取决于其他切面的设置,
 * 并且在 org.springframework.core.annotation.AnnotationAwareOrderComparator 会重新排序
 *
 * 注意:本项目因为是外部传递进来的云编号,根据动态创建数据源实例,并且进行切换,而这种只用dao层切面的方式,
 *    适用于进行多个master/slave读写分类用的场景,所以我们的项目用不到这种方式(我们如果使用这种方式,
 *      就需要修改daoAai入参方式,在前置处理器获取dao的方法参数,根据参数切换数据库,这样就需要修改dao接口,
 *      以及对应mapper.xml,需要了解动态代理的知识,所以目前我们没有使用该方式,目前我们使用的是
 *      在service或controller层手动切库)
 */

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

//@Aspect
//@Component
public class DynamicDataSourceAspect {
    private static final Logger logger = LoggerFactory.getLogger(DynamicDataSourceAspect.class);

    private final String[] QUERY_PREFIX = {"select"};

    @Pointcut("execution( * domain.dao.impl.*.*(..))")
    public void daoAspect() {
    }

    @Before("daoAspect()")
    public void switchDataSource(JoinPoint point) {
        Object[] params = point.getArgs();
        System.out.println(params.toString());
        String param = (String) params[0];
        for (Object string:params
             ) {
            System.out.println(string.toString());
        }
        System.out.println("###################################################");
        System.out.println(point.getSignature().getName());
        Boolean isQueryMethod = isQueryMethod(point.getSignature().getName());
        //DynamicDataSourceContextHolder.setDataSourceKey("slave");
        if (isQueryMethod) {
            DynamicDataSourceContextHolder.setDataSourceKey("slave");
            logger.info("Switch DataSource to [{}] in Method [{}]",
                    DynamicDataSourceContextHolder.getDataSourceKey(), point.getSignature());
        }
    }

    @After("daoAspect())")
    public void restoreDataSource(JoinPoint point) {
        DynamicDataSourceContextHolder.clearDataSourceKey();
        logger.info("Restore DataSource to [{}] in Method [{}]",
                DynamicDataSourceContextHolder.getDataSourceKey(), point.getSignature());
    }

    private Boolean isQueryMethod(String methodName) {
        for (String prefix : QUERY_PREFIX) {
            if (methodName.startsWith(prefix)) {
                return true;
            }
        }
        return false;
    }
}

九.SwitchDB手动切换数据库类

  • ControllerService 需要切换数据库的使用,需要使用 SwitchDB.change() 方法.
package domain.dbs;

import com.alibaba.druid.pool.DruidDataSource;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.env.Environment;
import org.springframework.transaction.PlatformTransactionManager;

import java.util.HashMap;
import java.util.Map;

/**
 * 切换数据库类
 * @author lxf 2018-09-28
 */
@Configuration
@Slf4j
public class SwitchDB {
    @Autowired
    private Environment evn;
    //私有库数据源key
    private static String  ljyunDataSourceKey = "ljyun_" ;

    @Autowired
    DynamicDataSource dynamicDataSource;

    @Autowired
    private PlatformTransactionManager transactionManager;

    /**
     * 切换数据库对外方法,如果私有库id参数非0,则首先连接私有库,否则连接其他已存在的数据源
     * @param dbName 已存在的数据库源对象
     * @param ljyunId 私有库主键
     * @return 返回当前数据库连接对象对应的key
     */
    public String change(String dbName,int ljyunId)
    {
        if( ljyunId == 0){
            toDB(dbName);
        }else {
            toYunDB(ljyunId);
        }
        //获取当前连接的数据源对象的key
        String currentKey = DynamicDataSourceContextHolder.getDataSourceKey();
        log.info("=====当前连接的数据库是:" + currentKey);
        return currentKey;
    }

    /**
     * 切换已存在的数据源
     * @param dbName
     */
    private void toDB(String dbName)
    {
        //如果不指定数据库,则直接连接默认数据库
        String dbSourceKey = dbName.trim().isEmpty() ? "default" : dbName.trim();
        //获取当前连接的数据源对象的key
        String currentKey = DynamicDataSourceContextHolder.getDataSourceKey();
        //如果当前数据库连接已经是想要的连接,则直接返回
        if(currentKey == dbSourceKey) return;
        //判断储存动态数据源实例的map中key值是否存在
        if( DynamicDataSource.isExistDataSource(dbSourceKey) ){
            DynamicDataSourceContextHolder.setDataSourceKey(dbSourceKey);
            log.info("=====普通库: "+dbName+",切换完毕");
        }else {
            log.info("切换普通数据库时,数据源key=" + dbName + "不存在");
        }
    }

    /**
     * 创建新的私有库数据源
     * @param ljyunId
     */
    private void  toYunDB(int ljyunId){
        //组合私有库数据源对象key
        String dbSourceKey = ljyunDataSourceKey+String.valueOf(ljyunId);
        //获取当前连接的数据源对象的key
        String currentKey = DynamicDataSourceContextHolder.getDataSourceKey();
        if(dbSourceKey == currentKey) return;

        //创建私有库数据源
        createLjyunDataSource(ljyunId);

        //切换到当前数据源
        DynamicDataSourceContextHolder.setDataSourceKey(dbSourceKey);
        log.info("=====私有库: "+ljyunId+",切换完毕");
    }

    /**
     * 创建私有库数据源,并将数据源赋值到targetDataSources中,供后切库用
     * @param ljyunId
     * @return
     */
    private DruidDataSource createLjyunDataSource(int ljyunId){
        //创建新的数据源
        if(ljyunId == 0)
        {
            log.info("动态创建私有库数据时,私有库主键丢失");
        }
        String yunId = String.valueOf(ljyunId);
        DruidDataSource dataSource = new DruidDataSource();
        String prefix = "db.privateDB.";
        String dbUrl = evn.getProperty( prefix + "url-base")
                + evn.getProperty( prefix + "host") + ":"
                + evn.getProperty( prefix + "port") + "/"
                + evn.getProperty( prefix + "dbname").replace("{id}",yunId) + evn.getProperty( prefix + "url-other");
        log.info("+++创建云平台私有库连接url = " + dbUrl);
        dataSource.setUrl(dbUrl);
        dataSource.setUsername(evn.getProperty( prefix + "username"));
        dataSource.setPassword(evn.getProperty( prefix + "password"));
        dataSource.setDriverClassName(evn.getProperty( prefix + "driver-class-name"));

        //将创建的数据源,新增到targetDataSources中
        Map<Object,Object> map = new HashMap<>();
        map.put(ljyunDataSourceKey+yunId, dataSource);
        DynamicDataSource.getInstance().setTargetDataSources(map);
        return dataSource;
    }
}

十.Service中根据外部变量手动切换数据库,使用SwitchDB.change()

  • TestTransaction实现
package domain.service.impl.exhibition;

import domain.dao.impl.ExhibitionDao;
import domain.dbs.DynamicDataSource;
import domain.dbs.DynamicDataSourceContextHolder;
import domain.dbs.SwitchDB;
import domain.domain.DomainResponse;
import domain.domain.Exhibition;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Isolation;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.transaction.interceptor.TransactionAspectSupport;

import java.util.HashMap;
import java.util.Map;

/**
 * 测试切库后的事务类
 * @author lxf 2018-09-28
 */
@Service
@Slf4j
public class TestTransaction {
    @Autowired
    private ExhibitionDao dao;
    @Autowired
    private SwitchDB switchDB;

    @Autowired
    DynamicDataSource dynamicDataSource;

    public DomainResponse testProcess(int kaiguan, int ljyunId, String dbName){
        switchDB.change(dbName,ljyunId);
        //获取当前已有的数据源实例
        System.out.println("%%%%%%%%"+dynamicDataSource.getDataSourceMap());
        return process(kaiguan,ljyunId,dbName);
    }

    /**
     * 事务测试
     * 注意:(1)有@Transactional注解的方法,方法内部不可以做切换数据库操作
     *      (2)在同一个service其他方法调用带@Transactional的方法,事务不起作用,(比如:在本类中使用testProcess调用process())
     *         可以用其他service中调用带@Transactional注解的方法,或在controller中调用.
     * @param kaiguan
     * @param ljyunId
     * @param dbName
     * @return
     */
    //propagation 传播行为 isolation 隔离级别  rollbackFor 回滚规则
    @Transactional(propagation = Propagation.REQUIRED,isolation = Isolation.DEFAULT,timeout=36000,rollbackFor=Exception.class)
    public DomainResponse process(int kaiguan, int ljyunId, String dbName ) {
        String currentKey = DynamicDataSourceContextHolder.getDataSourceKey();
        log.info("=====service当前连接的数据库是:" + currentKey);
            Exhibition exhibition = new Exhibition();
            exhibition.setExhibitionName("A-001-003");
            //return new DomainResponse<String>(1, "新增成功", "");
            int addRes = dao.insert(exhibition);
            if(addRes>0 && kaiguan==1){
                exhibition.setExhibitionName("B-001-002");
                int addRes2 = dao.insert(exhibition);
                return new DomainResponse<String>(1, "新增成功", "");
            }else
            {
                Map<String,String> map = new HashMap<>();
                String a = map.get("hello");
                //log.info("-----a="+a.replace("a","b"));
                //手动回滚事务
                TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
                return new DomainResponse<String>(0, "新增错误,事务已回滚", "");
            }
        }
}

十一.切库与事务

  • 需要在 DataSourceConfigurer 类中添加如下配置,让事务管理器与动态数据源对应起来;

      /**
       * 将动态数据源添加到事务管理器中,并生成新的bean
       * @return the platform transaction manager
       */
      @Bean
      public PlatformTransactionManager transactionManager() {
          return new DataSourceTransactionManager(dynamicDataSource());
      }
    ``` 
    
    
  • @Transactional注解的方法,方法内部不可以做切换数据库 操作

  • 同一个service其他方法调用带@Transactional的方法,事务不起作用,(比如:在本类中使用testProcess调用process()),参考这篇文章:https://blog.csdn.net/qq_33696896/article/details/82013095,知道的;

  • 可以用其他service中调用带@Transactional注解的方法,或在controller中调用.

关于多数据源的参考文章:
spring 动态切换、添加数据源实现以及源码浅析
Spring Boot 中使用 MyBatis 下实现多数据源动态切换,读写分离

关于事务的参考文章:
透彻的掌握 Spring 中@transactional 的使用

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,600评论 18 139
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,748评论 6 342
  • 这部分的参考文档涉及数据访问和数据访问层和业务或服务层之间的交互。 Spring的综合事务管理支持覆盖很多细节,然...
    竹天亮阅读 1,033评论 0 0
  • 你有没有发现记忆力在逐渐减退,可是越久远的事情反而记得更清楚,有的人说这叫做“选择性记忆”,有的人却说这是...
    半夏伊人阅读 431评论 0 0
  • A今天我学到了什么 html和CSS的作用 常用的html标签 常用 的css样式 B今天我掌握了什么 html和...
    余昌帅阅读 373评论 1 0