SpringBoot框架搭建(redis多数据源+数据库多数据源+全局异常捕获)

java骨架

目录结构

├── mvnw
├── mvnw.cmd
├── pom.xml
├── README.md
├── src/              源文件目录
│   ├── main
│   │   ├── java
│   │   │   └── cn
│   │   │       └── sigo
│   │   │           └── archtype
│   │   │               ├── ArchtypeApplication.java 启动类
│   │   │               ├── common/          公用参数配置目录
│   │   │               │   ├── ArchtypeConstantsConfig.java    公共参数配置
│   │   │               │   ├── CompareStatus.java
│   │   │               │   ├── Error.java
│   │   │               │   ├── ErrorCode.java
│   │   │               │   └──......
│   │   │               └── config/        公共配置目录
│   │   │                   ├── JwtAuthenticationFilter.java    解决前端到后台跨域问题
│   │   │                   ├── redis/    redis配置
│   │   │                   │   ├── DefaultRedisConfig.java
│   │   │                   │   ├── RedisConfig.java
│   │   │                   │   └── ......
│   │   │                   ├── controller/   控制器目录
│   │   │                   │   ├── TestController.java
│   │   │                   │   └── .....
│   │   │                   ├── mapper/          mapper 目录 
│   │   │                   │   └── .....
│   │   │                   ├── model/        Model 目录 (绑定数据表)
│   │   │                   │   ├── po/      此对象与数据库表结构一一对应,通过 mapper 层向上传输数据源对象(数据库)
│   │   │                   │   │   └── .....
│   │   │                   │   ├── dto/       数据传输对象, Service 或 Manager 向外传输的对象。(输出)
│   │   │                   │        └── .....
│   │   │                   │   ├── bo/      业务对象, 由 Service 层输出的封装业务逻辑的对象(业务处理,输入)
│   │   │                   │        └── .....
│   │   │                   │   ├── query/      数据查询对象,各层接收上层的查询请求。 注意超过 2 个参数的查询封装,禁止
                                                    使用 Map 类来传输(数据库逻辑处理)
│   │   │                   │        └── .....
│   │   │                   │   └── .....
│   │   │                   ├── service/      服务接口目录     
│   │   │                   │   ├── impl/     服务接口实现目录
│   │   │                   │       ├── TestServiceImpl.java
│   │   │                   │        └── .....
│   │   │                   │   ├── TestService.java
│   │   │                   │   └── .....
│   │   │                   └── utils/      公共工具类目录
│   │   │                        ├── DesUtil     DEC(ASC码加密)-带向量及密钥
│   │   │                        ├── EncrypDES     DEC(ASC码加密)-不带向量及密钥
│   │   │                        ├── JsonUtils     Json数据处理工具类
│   │   │                        ├── LogHelper     日志处理工具类
│   │   │                        ├── MD5Utils     MD5通用类
│   │   │                        ├── ObjectUtils     驼峰及下划线互转处理工具
│   │   │                        ├── RedisHelper     redis工具类
│   │   │                        ├── SendSmsUtil     发送短信工具类
│   │   │                        └── .....
│   │   └── resources/          资源总目录
│   │       ├── generator/ 
│   │       │   ├── generatorConfig.xml           代码自动生成工具
│   │       │   └── .....
│   │       ├── mapping/                           数据库sql配置文件
│   │       ├── static/                            静态资源文件目录
│   │       ├── templates/                         web 模板目录
│   │       ├── application-dev.yml                开发配置
│   │       ├── application-prod.yml               产品配置
│   │       ├── application.yml                    当前配置
│   │       ├── application-test.yml               测试配置
│   │       └── archtype.properties                公共参数properties

Redis多数据源

将redis数据库信息配置到application-test.yml,application-prod.yml,application-dev.yml配置文件中,
如业务需求需要连接多个redis数据源,配置多个DefaultRedisConfig.java文件。
import org.springframework.beans.factory.annotation.Value;
import org.springframework.cache.annotation.EnableCaching;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.jedis.JedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.StringRedisSerializer;
import redis.clients.jedis.JedisPoolConfig;

/**
 * @Author: panwei
 * @Date: 2019/1/22 11:58
 * @apiNote: Redis数据连接池及公共参数
 * @Version 1.0
 */
@EnableCaching
@Configuration
public class RedisConfig {
    @Value("${spring.redis.jedis.pool.max-active}")
    private int redisPoolMaxActive;

    @Value("${spring.redis.jedis.pool.max-wait}")
    private int redisPoolMaxWait;

    @Value("${spring.redis.jedis.pool.max-idle}")
    private int redisPoolMaxIdle;

    @Value("${spring.redis.jedis.pool.min-idle}")
    private int redisPoolMinIdle;


    /**
     * 创建redis连接工厂
     *
     * @param dbIndex
     * @param host
     * @param port
     * @param password
     * @param timeout
     * @return
     */
    public JedisConnectionFactory createJedisConnectionFactory(int dbIndex, String host, int port, String password, int timeout) {
        JedisConnectionFactory jedisConnectionFactory = new JedisConnectionFactory();
        jedisConnectionFactory.setDatabase(dbIndex);
        jedisConnectionFactory.setHostName(host);
        jedisConnectionFactory.setPort(port);
        jedisConnectionFactory.setPassword(password);
        jedisConnectionFactory.setTimeout(timeout);
        jedisConnectionFactory.setPoolConfig(setPoolConfig(redisPoolMaxIdle, redisPoolMinIdle, redisPoolMaxActive, redisPoolMaxWait, true));
        return jedisConnectionFactory;
    }

    /**
     * 设置连接池属性
     *
     * @param maxIdle
     * @param minIdle
     * @param maxActive
     * @param maxWait
     * @param testOnBorrow
     * @return
     */
    public JedisPoolConfig setPoolConfig(int maxIdle, int minIdle, int maxActive, int maxWait, boolean testOnBorrow) {
        JedisPoolConfig poolConfig = new JedisPoolConfig();
        poolConfig.setMaxIdle(maxIdle);
        poolConfig.setMinIdle(minIdle);
        poolConfig.setMaxTotal(maxActive);
        poolConfig.setMaxWaitMillis(maxWait);
        poolConfig.setTestOnBorrow(testOnBorrow);
        return poolConfig;
    }

    /**
     * 设置RedisTemplate的序列化方式
     *
     * @param redisTemplate
     */
    public void setSerializer(RedisTemplate redisTemplate) {
        StringRedisSerializer stringRedisSerializer =new StringRedisSerializer();
        //设置键(key)的序列化方式
        redisTemplate.setKeySerializer(stringRedisSerializer);
        //设置值(value)的序列化方式
        redisTemplate.setValueSerializer(stringRedisSerializer);
        redisTemplate.setHashKeySerializer(stringRedisSerializer);
        redisTemplate.setHashValueSerializer(stringRedisSerializer);
        redisTemplate.afterPropertiesSet();
    }
}
import org.springframework.beans.factory.annotation.Value;
import org.springframework.cache.annotation.EnableCaching;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;

/**
 * @Author: panwei
 * @Date: 2019/1/22 11:58
 * @apiNote: 默认的redis数据源
 * @Version 1.0
 */
@Configuration
@EnableCaching
public class DefaultRedisConfig extends RedisConfig {
    @Value("${spring.redis.database}")
    private int dbIndex;

    @Value("${spring.redis.redis1.host}")
    private String host;

    @Value("${spring.redis.redis1.port}")
    private int port;

    @Value("${spring.redis.redis1.password}")
    private String password;

    @Value("${spring.redis.timeout}")
    private int timeout;

    /**
     * 配置redis连接工厂
     *
     * @return
     */
    @Bean
    public RedisConnectionFactory defaultRedisConnectionFactory() {
        return createJedisConnectionFactory(dbIndex, host, port, password, timeout);
    }

    /**
     * 配置redisTemplate 注入方式使用@Resource(name="") 方式注入
     *
     * @return
     */
    @Bean(name = "defaultRedisTemplate")
    public RedisTemplate defaultRedisTemplate() {
        RedisTemplate template = new RedisTemplate();
        template.setConnectionFactory(defaultRedisConnectionFactory());
        setSerializer(template);
        template.afterPropertiesSet();
        return template;
    }
}
import org.springframework.beans.factory.annotation.Value;
import org.springframework.cache.annotation.EnableCaching;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;

/**
 * @Author: panwei
 * @Date: 2019/1/22 11:58
 * @apiNote: redis数据源-其他
 * @Version 1.0
 */
@Configuration
@EnableCaching
public class CacheRedisConfig extends RedisConfig {
    @Value("${spring.redis.database}")
    private int dbIndex;

    @Value("${spring.redis.redis2.host}")
    private String host;

    @Value("${spring.redis.redis2.port}")
    private int port;

    @Value("${spring.redis.redis2.password}")
    private String password;

    @Value("${spring.redis.timeout}")
    private int timeout;

    /**
     * 配置redis连接工厂
     *
     * @return
     */
    @Primary
    @Bean
    public RedisConnectionFactory cacheRedisConnectionFactory() {
        return createJedisConnectionFactory(dbIndex, host, port, password, timeout);
    }

    /**
     * 配置redisTemplate 注入方式使用@Resource(name="") 方式注入
     *
     * @return
     */
    @Bean(name = "cacheRedisTemplate")
    public RedisTemplate cacheRedisTemplate() {
        RedisTemplate template = new RedisTemplate();
        template.setConnectionFactory(cacheRedisConnectionFactory());
        setSerializer(template);
        template.afterPropertiesSet();
        return template;
    }
}
  • 具体例子请查看TestController.java,其中也包含了切换的db方法。
  • 注意DefaultRedisConfig.java中@Bean对应的name必须唯一
 redis:
      #数据库索引
      database: 0
      redis1:
        host: 192.168.5.58
        port: 6379
        password:
      redis2:
        host: 192.168.5.87
        port: 6379
        password:
      jedis:
        pool:
          #最大连接数
          max-active: 8
          #最大阻塞等待时间(负数表示没限制)
          max-wait: -1
          #最大空闲
          max-idle: 8
          #最小空闲
          min-idle: 0
      #连接超时时间
      timeout: 10000
      keytimeout: 10000

数据库多数据源

项目中配置了两个mysql数据库和一个sqlserver数据库的数据源连接,均采用druid的连接方式
sqlserver:
  datasource:
    url: jdbc:sqlserver://10.0.180.152\ERPDB152;database=SigoBackEnd
    username: sa
    password: sigo.cn@123
    driverClassName: com.microsoft.sqlserver.jdbc.SQLServerDriver
mysql:
  datasource:
    url: jdbc:mysql://localhost:3306/test?useUnicode=true&characterEncoding=UTF-8&allowMultiQueries=true&serverTimezone=GMT%2B8
    username: root
    password: 2wsx@WSX
    driverClassName: com.mysql.jdbc.Driver
  • 具体的数据库Config配置方法请查看MysqlDataSourceConfig.javaMysqlDataSourceConfig1.javaSqlServerDataSourceConfig.java
    import com.alibaba.druid.pool.DruidDataSource;
    import org.apache.ibatis.session.SqlSessionFactory;
    import org.mybatis.spring.SqlSessionFactoryBean;
    import org.mybatis.spring.annotation.MapperScan;
    import org.springframework.beans.factory.annotation.Qualifier;
    import org.springframework.beans.factory.annotation.Value;
    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.context.annotation.Primary;
    import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
    import org.springframework.jdbc.datasource.DataSourceTransactionManager;
    
    import javax.sql.DataSource;
    
    /**
     * @Author: panwei
     * @Date: 2019/1/23 11:58
     * @apiNote: Mysql扫描 Mapper 接口并容器管理
     * @Version 1.0
     */
    @Configuration
    @MapperScan(basePackages = MysqlDataSourceConfig.PACKAGE, sqlSessionFactoryRef = "mysqlSqlSessionFactory")
    public class MysqlDataSourceConfig {
        // 精确到 mysql 目录,以便跟其他数据源隔离
    
        /**
         * 对应的mysql dao层包
        */
        static final String PACKAGE = "cn.sigo.api.mapper.mysql";
    
        /**
         * 对应mysql的mapper.xml文件
         */
        static final String MAPPER_LOCATION = "classpath:mapping/mysql/*.xml";
    
        @Value("${spring.datasource.url}")
        private String url;
    
        @Value("${spring.datasource.username}")
        private String user;
    
        @Value("${spring.datasource.password}")
        private String password;
    
        @Value("${spring.datasource.driver-class-name}")
        private String driverClass;
    
        @Bean(name = "mysqlDataSource")
        @Primary
        public DataSource mysqlDataSource() {
            DruidDataSource dataSource = new DruidDataSource();
            dataSource.setDriverClassName(driverClass);
            dataSource.setUrl(url);
            dataSource.setUsername(user);
            dataSource.setPassword(password);
            return dataSource;
        }
    
        @Bean(name = "mysqlTransactionManager")
        @Primary
        public DataSourceTransactionManager mysqlTransactionManager() {
            return new DataSourceTransactionManager(mysqlDataSource());
        }
    
        @Bean(name = "mysqlSqlSessionFactory")
        @Primary
        public SqlSessionFactory mysqlSqlSessionFactory(@Qualifier("mysqlDataSource") DataSource mysqlDataSource) throws Exception {
            final SqlSessionFactoryBean sessionFactory = new SqlSessionFactoryBean();
            sessionFactory.setDataSource(mysqlDataSource);
            sessionFactory.setMapperLocations(
                    new PathMatchingResourcePatternResolver().getResources(MysqlDataSourceConfig.MAPPER_LOCATION));
            return sessionFactory.getObject();
        }
    }
import com.alibaba.druid.pool.DruidDataSource;
import org.apache.ibatis.session.SqlSessionFactory;
import org.mybatis.spring.SqlSessionFactoryBean;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;

import javax.sql.DataSource;

/**
 * @Author: panwei
 * @Date: 2019/1/23 13:56
 * @apiNote: sqlserver扫描 Mapper 接口并容器管理
 * @Version 1.0
 */
@Configuration
@MapperScan(basePackages=SqlServerDataSourceConfig.PACKAGE,sqlSessionFactoryRef="sqlserverSqlSessionFactory")
public class SqlServerDataSourceConfig {
    // 精确到 sqlserver 目录,以便跟其他数据源隔离

    static final String PACKAGE = "cn.sigo.api.mapper.sqlserver";
    static final String MAPPER_LOCATION = "classpath:mapping/sqlserver/*.xml";

    @Value("${sqlserver.datasource.url}")
    private String url;

    @Value("${sqlserver.datasource.username}")
    private String user;

    @Value("${sqlserver.datasource.password}")
    private String password;

    @Value("${sqlserver.datasource.driverClassName}")
    private String driverClass;

    @Bean(name = "sqlserverDataSource")
    public DataSource sqlserverDataSource() {
        DruidDataSource dataSource = new DruidDataSource();
        dataSource.setDriverClassName(driverClass);
        dataSource.setUrl(url);
        dataSource.setUsername(user);
        dataSource.setPassword(password);
        return dataSource;
    }

    @Bean(name = "sqlserverTransactionManager")
    public DataSourceTransactionManager sqlserverTransactionManager() {
        return new DataSourceTransactionManager(sqlserverDataSource());
    }

    @Bean(name = "sqlserverSqlSessionFactory")
    public SqlSessionFactory sqlserverSqlSessionFactory(@Qualifier("sqlserverDataSource") DataSource sqlserverDataSource) throws Exception {
        final SqlSessionFactoryBean sessionFactory = new SqlSessionFactoryBean();
        sessionFactory.setDataSource(sqlserverDataSource);
        sessionFactory.setMapperLocations(
                new PathMatchingResourcePatternResolver().getResources(SqlServerDataSourceConfig.MAPPER_LOCATION));
        return sessionFactory.getObject();
    }
}

全局异常捕获

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