springboot配置druid连接池

Springboot配置druid数据连接池有两种方式,比较类似,但是如果搞混了容易导致部分配置不生效。

1.starter方式

这种比较简单

1.引入依赖

        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid-spring-boot-starter</artifactId>
            <version>1.1.20</version>
        </dependency>

2.添加属性

spring.datasource.type=com.alibaba.druid.pool.DruidDataSource
spring.datasource.url=jdbc:mysql://localhost:3306/workflow?useUnicode=true&characterEncoding=utf-8&useSSL=false
spring.datasource.username=workflow
spring.datasource.password=workflow
spring.datasource.driver-class-name=com.mysql.jdbc.Driver

spring.jpa.database-platform=org.hibernate.dialect.MySQL5InnoDBDialect

spring.datasource.druid.initial-size=1
spring.datasource.druid.max-active=20
spring.datasource.druid.min-idle=3
spring.datasource.druid.max-wait=60000
spring.datasource.druid.pool-prepared-statements=true
spring.datasource.druid.max-pool-prepared-statement-per-connection-size=20
spring.datasource.druid.filters=stat,wall,slf4j
spring.datasource.druid.connection-properties=druid.stat.mergeSql=true;druid.stat.slowSqlMillis=5000

这样启动项目就ok了。

注意,不需要自己添加新的数据连接池配置类DataSourceConfig。添加了之后可能会导致项目启动失败或者部分属性例如maxActive,minIdle等这些不生效,具体原理下面会说。

3.原理

简单说下原理,这里用到了自动装配原理。

关键是这个类

@Configuration
@ConditionalOnClass({DruidDataSource.class})
@AutoConfigureBefore({DataSourceAutoConfiguration.class})
@EnableConfigurationProperties({DruidStatProperties.class, DataSourceProperties.class})
@Import({DruidSpringAopConfiguration.class, DruidStatViewServletConfiguration.class, DruidWebStatFilterConfiguration.class, DruidFilterConfiguration.class})
public class DruidDataSourceAutoConfigure {
    private static final Logger LOGGER = LoggerFactory.getLogger(DruidDataSourceAutoConfigure.class);

    public DruidDataSourceAutoConfigure() {
    }

    @Bean(
        initMethod = "init"
    )
    @ConditionalOnMissingBean
    public DataSource dataSource() {
        LOGGER.info("Init DruidDataSource");
        return new DruidDataSourceWrapper();
    }
}

@Configuration用于定义配置类,被注解的类内部包含有一个或多个被@Bean注解的方法,这些方法将会被AnnotationConfigApplicationContextAnnotationConfigWebApplicationContext类进行扫描,并用于构建bean定义,初始化Spring容器。

@ConditionalOnClass({DruidDataSource.class})这段可以简单理解为只有当DruidDataSource.class这个类存在时候才会实例化DruidDataSourceAutoConfigure中定义的Bean。

@AutoConfigureBefore({DataSourceAutoConfiguration.class})表示DruidDataSourceAutoConfigure要在DataSourceAutoConfiguration这个类之前加载。

@EnableConfigurationProperties({DruidStatProperties.class, DataSourceProperties.class})表示开启自动装配属性,不需要手动set。与注解ConfigurationProperties配合使用,ConfigurationProperties负责将配置文件中的属性放到对象对应的属性中。

@Import({DruidSpringAopConfiguration.class, DruidStatViewServletConfiguration.class, DruidWebStatFilterConfiguration.class, DruidFilterConfiguration.class})表示把用到的资源导入到IOC容器中。在这里就表示将这几个类实例化为Bean后加载到IOC容器中。不仅可以导入类,也可以通过@ImportResource导入xml或者properties配置文件。

然后我们看类的实现,比较简单,除了构造方法只有一个方法public DataSource dataSource(),这个方法就是返回一个数据库连接池Bean。包含了两个注解:

@Bean( initMethod = "init" )表示返回一个Bean,同时在Bean初始化的时候要执行init这个方法。

@ConditionalOnMissingBean简单理解就是当dataSource这个Bean不存在时候才会加载这个Bean。

返回的Bean很简单就是new了一个DruidDataSourceWrapper对象。

接下来看下DruidDataSourceWrapper这个类

@ConfigurationProperties("spring.datasource.druid")
class DruidDataSourceWrapper extends DruidDataSource implements InitializingBean {
    @Autowired
    private DataSourceProperties basicProperties;

    DruidDataSourceWrapper() {
    }

    public void afterPropertiesSet() throws Exception {
        if (super.getUsername() == null) {
            super.setUsername(this.basicProperties.determineUsername());
        }

        if (super.getPassword() == null) {
            super.setPassword(this.basicProperties.determinePassword());
        }

        if (super.getUrl() == null) {
            super.setUrl(this.basicProperties.determineUrl());
        }

        if (super.getDriverClassName() == null) {
            super.setDriverClassName(this.basicProperties.getDriverClassName());
        }

    }

    @Autowired(
        required = false
    )
    public void autoAddFilters(List<Filter> filters) {
        super.filters.addAll(filters);
    }

    public void setMaxEvictableIdleTimeMillis(long maxEvictableIdleTimeMillis) {
        try {
            super.setMaxEvictableIdleTimeMillis(maxEvictableIdleTimeMillis);
        } catch (IllegalArgumentException var4) {
            super.maxEvictableIdleTimeMillis = maxEvictableIdleTimeMillis;
        }

    }
}

@ConfigurationProperties("spring.datasource.druid")这个注解上面已经提到就是为了自动装配属性,装配哪些属性由括号里面的决定,这里是装配配置文件中spring.datasource.druid开头的属性。看下之前的properties文件,发现主要是数据库连接池中maxActive,minIdle等这些属性。那么就会有个问题,连接池的最基础的属性url, username,password,driverClassName这些在配置文件中不是以spring.datasource.druid开头的,那么是怎么装配呢?有两种方法:

  1. 根据上面我们说的@ConfigurationProperties("spring.datasource.druid")可以装配以spring.datasource.druid开头的属性,将properties文件中这几项的配置改为:

    spring.datasource.druid.url=jdbc:mysql://localhost:3306/workflow?useUnicode=true&characterEncoding=utf-8&useSSL=false
    spring.datasource.druid.username=workflow
    spring.datasource.druid.password=workflow
    spring.datasource.druid.driver-class-name=com.mysql.jdbc.Driver
    

    类似这样。这样@ConfigurationProperties("spring.datasource.druid")注解会将属性装配进来。

  2. 我们发现不修改也会装配进来,即下面这种形式也会装配进来

    spring.datasource.url=jdbc:mysql://localhost:3306/workflow?useUnicode=true&characterEncoding=utf-8&useSSL=false
    spring.datasource.username=workflow
    spring.datasource.password=workflow
    spring.datasource.driver-class-name=com.mysql.jdbc.Driver
    

    为什么会这样呢?

    这与这个类DruidDataSourceWrapper里面的一个方法有关:afterPropertiesSet()

afterPropertiesSet这个方法是InitializingBean接口的实现,根据名称我们就可以猜到这个类和Bean的初始化有关系。这个方法会在Bean初始化的时候执行,并且是先于init方法执行的。具体分析我们以后文章会分析,可以关注下。

afterPropertiesSet这个方法我们看实现就是为了把url, username,password,driverClassName这几个属性放到数据连接池对象中对应的属性。就是简单的set方法,我们看到这几个属性来自于DataSourceProperties这个类的对象,那么这个类的对象从哪里获得连接的基本属性呢,看下类的实现:

@ConfigurationProperties(
    prefix = "spring.datasource"
)
public class DataSourceProperties implements BeanClassLoaderAware, InitializingBean {
    private ClassLoader classLoader;
    private String name;
    private boolean generateUniqueName;
    private Class<? extends DataSource> type;
    private String driverClassName;
    private String url;
    private String username;
    private String password;

类太长,我们看关键部分。

这里还是用到了@ConfigurationProperties("spring.datasource")这个注解,它负责把properties中spring.datasource开头的属性装配到类DataSourceProperties对象中对应的属性中。然后再通过afterPropertiesSet方法放到DruidDataSourceWrapper中对应的属性。

这样就会将所有的配置属性初始化到DataSourceBean中。

这种配置Druid连接池方法中用到的最关键的技术就是Springboot的自动装配原理

2.非Starter方法

  1. 引入依赖

    <dependency>
           <groupId>com.alibaba</groupId>
           <artifactId>druid</artifactId>
           <version>1.1.9</version>
    </dependency>
    
  2. 添加属性

    spring.datasource.druid.url=jdbc:mysql://localhost:3306/workflow?useUnicode=true&characterEncoding=utf-8&useSSL=false
    spring.datasource.druid.username=workflow
    spring.datasource.druid.password=workflow
    spring.datasource.druid.driver-class-name=com.mysql.jdbc.Driver
    
    spring.jpa.database-platform=org.hibernate.dialect.MySQL5InnoDBDialect
    
    spring.datasource.druid.initial-size=1
    spring.datasource.druid.max-active=20
    spring.datasource.druid.min-idle=3
    spring.datasource.druid.max-wait=60000
    spring.datasource.druid.pool-prepared-statements=true
    spring.datasource.druid.max-pool-prepared-statement-per-connection-size=20
    spring.datasource.druid.filters=stat,wall,slf4j
    spring.datasource.druid.connection-properties=druid.stat.mergeSql=true;druid.stat.slowSqlMillis=5000
    
  3. 添加数据源配置类

    public class DataSourceConfig {
        @Bean
        @ConfigurationProperties(prefix = "spring.datasource.druid")
        public DataSource dataSource() {
            return new DruidDataSource();
        }
    }
    

    这里要注意配置properties文件中的url, username,password,driverClassName属性的写法,是包含“druid”的,因为这里也是用到了注解@ConfigurationProperties(prefix = "spring.datasource.druid")来自动装配属性的,我们指定了自动装配以spring.datasource.druid开头的属性,如果写成spring.datasource.url则是没有办法自动装配的,属性不会生效。

    当然还有其他方法就是不使用自动装配,通过注解@Value获取到配置文件属性,然后手动用set方法将属性设置到DataSource对象对应的属性中去,这样比较麻烦,不推荐。

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

推荐阅读更多精彩内容