Druid 连接池连接 OceanBase 数据库示例程序

本文将介绍如何使用 Druid 连接池、MySQL Connector/J 和 OceanBase 数据库构建一个应用程序,实现基本的数据库操作,包括创建表、插入数据、更新数据、删除数据、查询数据和删除表等。

image.png点击下载 druid-mysql-client 示例工程

前提条件

  • 您已安装 OceanBase 数据库并且创建了 MySQL 模式租户。

  • 您已安装 JDK 1.8 和 Maven。

  • 您已安装 Eclipse。

    说明

    本文档运行代码使用的工具是 Eclipse IDE for Java Developers 2022-03 版本,您也可以根据个人喜好选择适合自己的工具运行的示例代码。

操作步骤

说明

本文中给出的操作步骤是在 Windows 环境下使用 Eclipse IDE for Java Developers 2022-03 编译和运行该项目的步骤。如果您使用的是其他操作系统环境或编译器,那么操作步骤可能会略有不同。

  1. druid-mysql-client项目导入到 Eclipse 中。

  2. 获取 OceanBase 数据库 URL。

  3. 修改druid-mysql-client项目中的数据库连接信息。

  4. 运行druid-mysql-client项目。

步骤一:将 druid-mysql-client 项目导入到 Eclipse 中

  1. 打开 Eclipse,在菜单栏上选择File->Open Projects from File System

  2. 在弹出的对话框中,点击Directory按钮选择项目所在的目录,然后点击Finish完成导入。

    说明

    当使用 Eclipse 导入 Maven 项目时,Eclipse 会自动检测项目中的pom.xml文件,并根据文件中描述的依赖关系自动下载所需的依赖库,并将它们添加到项目中。

    image.png

  3. 查看项目情况。

    image.png

步骤二:获取 OceanBase 数据库 URL

  1. 联系 OceanBase 数据库部署人员或者管理员获取相应的数据库连接串。

    示例如下:

    obclient -hxxx.xxx.xxx.xxx -P3306 -utest_user001 -p****** -Dtest

    更多连接串的信息,请参见 获取连接参数

  2. 根据 OceanBase 数据库连接串信息填写下面 URL 的对应信息。

    jdbc:mysql://$host:$port/$database_name?user=$user_name&password=$password&useSSL=false

    参数说明:

    • $hostOceanBase 数据库连接的域名。

    • $port:OceanBase 数据库连接端口,MySQL 模式租户默认是 3306。

    • $database_name:需要访问的数据库名称。

    • $user_name:租户的连接账号。

    • $password:提供账户密码。

    更多有关 MySQL Connector/J 连接属性信息,请参见 Configuration Properties

    示例如下:

    jdbc:mysql://xxx.xxx.xxx.xxx:3306/test?user=test_user001&password=******&useSSL=false
    

步骤三:修改 druid-mysql-client 项目中的数据库连接信息

根据步骤二:获取 OceanBase 数据库 URL中获取的信息修改druid-mysql-client/src/main/resources/db.properties文件中的数据库连接信息。

image.png

示例如下:

  • OBServer 节点的 IP 地址为xxx.xxx.xxx.xxx

  • 访问端口使用的是 3306。

  • 需要访问的数据库名称为test

  • 租户的连接账户是test_user001

  • 密码是******

代码如下:

...
url=jdbc:mysql://xxx.xxx.xxx.xxx:3306/test?useSSL=false
username=test_user001
password=******
...

步骤四:运行 druid-mysql-client 项目

  1. 在项目导航器视图中,找到并展开druid-mysql-client/src/main/java目录。

  2. 右键点击Main.java文件,然后选择Run As->Java Application

    image.png

  3. 在 Eclipse 的控制台窗口中来查看输出结果。

    image.png

项目代码介绍

点击 druid-mysql-client 下载项目代码,是一个名称为druid-mysql-client.zip的压缩包。

解压后,得到一个名为druid-mysql-client的文件夹。目录结构如下所示:

druid-mysql-client
├── src
│   └── main
│       ├── java
│       │   └── com
│       │       └── example
│       │           └── Main.java
│       └── resources
│           └── db.properties
└── pom.xml

文件说明:

  • src:源代码根目录。

  • main:主代码目录,包含应用程序的主要逻辑。

  • java:Java 源代码目录。

  • com:Java 包目录。

  • example:示例项目的包目录。

  • Main.java:主类程序示例文件,包含创建表、插入、删除、更新和查询数据等逻辑。

  • resources:资源文件目录,包含配置文件等。

  • db.properties:连接池的配置文件,包含了数据库连接的相关参数。

  • pom.xml:Maven 项目的配置文件,用于管理项目的依赖和构建设置。

pom.xml 代码介绍

pom.xml文件是 Maven 项目的配置文件,定义了项目的依赖项、插件和构建规则等信息。Maven 是一个 Java 项目管理工具,可以自动下载依赖项、编译和打包项目等操作。

本文pom.xml文件的代码主要包括以下几个部分:

  1. 文件声明语句。

    声明本文件是一个 XML 文件,使用的 XML 版本是1.0,字符编码方式是UTF-8

    代码如下:

    <?xml version="1.0" encoding="UTF-8"?>
  2. 配置 POM 的命名空间和 POM 模型版本。

    1. 通过xmlns指定 POM 的命名空间为http://maven.apache.org/POM/4.0.0

    2. 通过xmlns:xsi指定 XML 命名空间http://www.w3.org/2001/XMLSchema-instance

    3. 通过xsi:schemaLocation指定 POM 的命名空间为http://maven.apache.org/POM/4.0.0和 POM 的 XSD 文件的位置为http://maven.apache.org/xsd/maven-4.0.0.xsd

    4. 通过<modelVersion>元素指定了该 POM 文件使用的 POM 模型版本为4.0.0

    代码如下:

    <project xmlns="http://maven.apache.org/POM/4.0.0"
             xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
             xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
      <modelVersion>4.0.0</modelVersion>
    
     <!-- 其他配置 -->
    
    </project>
  3. 配置基本信息。

    1. 通过<groupId>指定项目所属组织为com.example

    2. 通过<artifactId>指定项目的名称为druid-mysql-client

    3. 通过<version>项目的版本号为1.0-SNAPSHOT

    代码如下:

        <groupId>com.example</groupId>
        <artifactId>druid-mysql-client</artifactId>
        <version>1.0-SNAPSHOT</version>
  4. 配置项目源文件的属性。

    指定 Maven 的编译器插件为maven-compiler-plugin,并设置了源代码和目标 Java 版本都为 8。这意味着项目的源代码使用 Java 8 特性编写,且编译后的字节码也将兼容 Java 8 运行时环境。这样设置可以确保项目在编译和运行时能够正确地处理 Java 8 的语法和特性。

    说明

    Java 1.8 和 Java 8 是同一个版本的不同命名方式。

    代码如下:

        <build>
            <plugins>
                <plugin>
                    <groupId>org.apache.maven.plugins</groupId>
                    <artifactId>maven-compiler-plugin</artifactId>
                    <configuration>
                        <source>8</source>
                        <target>8</target>
                    </configuration>
                </plugin>
            </plugins>
        </build>
  5. 配置项目所依赖组件。

    1. 添加mysql-connector-java依赖库,用于与数据库进行交互:

      1. 通过<groupId>指定依赖项所属的组织为mysql

      2. 通过<artifactId>指定依赖项的名称为mysql-connector-java

      3. 通过<version>指定依赖项的版本号为5.1.40

      代码如下:

              <dependency>
                  <groupId>mysql</groupId>
                  <artifactId>mysql-connector-java</artifactId>
                  <version>5.1.40</version>
              </dependency>
    2. 添加druid依赖库:

      1. 通过<groupId>指定依赖项所属的组织为com.alibaba

      2. 通过<artifactId>指定依赖项的名称为druid

      3. 通过<version>指定依赖项的版本号为1.2.8

      代码如下:

              <dependency>
                  <groupId>com.alibaba</groupId>
                  <artifactId>druid</artifactId>
                  <version>1.2.8</version>
              </dependency>

db.properties 代码介绍

db.properties是本文示例的连接池配置文件,其中包含了连接池的配置属性。这些属性包括数据库 URL、用户名、密码、连接池其他可选项。

本文db.properties文件的代码主要包括以下几个部分:

  1. 配置数据库连接参数。

    1. 指定数据库驱动程序的类名为com.oceanbase.jdbc.Driver

    2. 指定数据库连接的 URL,包括主机 IP、端口号、需要访问的 Schema。

    3. 指定数据库的用户名。

    4. 指定数据库的密码。

    代码如下:

    driverClassName=com.oceanbase.jdbc.Driver
    url=jdbc:oceanbase://$host:$port/$database_name?useSSL=false
    username=$user_name
    password=$password

    参数解释:

    • $hostOceanBase 数据库连接的域名。

    • $port:OceanBase 数据库连接端口,MySQL 模式租户默认是 3306。

    • $database_name:需要访问的数据库名称。

    • $user_name:租户的连接账号。

    • $password:提供账户密码。

  2. 配置其他连接池的参数。

    1. 指定验证连接的 SQL 语句select 1

    2. 指定连接池的初始连接数量为 3,表示在启动连接池时,会创建 3 个初始连接。

    3. 指定连接池的最大活动连接数为 30,表示连接池中最多可以同时存在 30 个连接。

    4. 指定是否打印废弃连接的日志为true,表示当回收废弃连接时,会在错误日志中输出信息。在测试环境中可以设置为true,而在线上环境中设置为false,以避免对性能产生影响。

    5. 指定连接池的最小空闲连接数为 5,表示当连接池中的空闲连接数少于 5 个时,连接池会自动创建新的连接。

    6. 指定获取连接时的最大等待时间 1000,单位为毫秒,表示如果连接池中的连接都被占用,且等待时间超过 1000 毫秒,则获取连接操作将会抛出超时异常。

    7. 指定连接在空闲时的最小保持时间为 300000,单位为毫秒,表示连接在空闲 300000 毫秒(5 分钟)后,如果没有被使用,会被回收。

    8. 指定是否回收废弃的连接为true,表示当连接超过removeAbandonedTimeout定义的时间时,会被回收。

    9. 指定废弃连接的超时时间为 300,单位为秒,表示超过 300 秒(5 分钟)未被使用的连接将被回收。

    10. 指定空闲连接回收线程的运行间隔时间为 10000,单位为毫秒,表示每隔 10000 毫秒(10 秒),空闲连接回收线程会执行一次空闲连接的回收操作。

    11. 指定获取连接时是否验证连接的可用性为false。设置为false可以提高性能,但也可能导致获得的连接不可用。

    12. 指定返回连接时是否验证连接的可用性为false。设置为false可以提高性能,但也可能导致返回的连接不可用。

    13. 指定连接在空闲时是否进行验证为true。设置为true时,连接池会定期执行validationQuery来验证连接的可用性。

    14. 指定是否启用长连接保活功能为false。设置为false表示不启用长连接保活。

    15. 指定了连接的空闲时间阈值为 60000,单位为毫秒,表示如果连接的空闲时间超过了设定的阈值 60000 毫秒(即 1 分钟),连接保活机制会对该连接进行检测,以确保连接的可用性。如果连接在阈值时间内有任何操作,连接的空闲时间会重新计算。

    代码如下:

    validationQuery=select 1
    initialSize=3
    maxActive=30
    logAbandoned=true
    minIdle=5
    maxWait=1000
    minEvictableIdleTimeMillis=300000
    removeAbandoned=true
    removeAbandonedTimeout=300
    timeBetweenEvictionRunsMillis=10000
    testOnBorrow=false
    testOnReturn=false
    testWhileIdle=true
    keepAlive=false
    keepAliveBetweenTimeMillis=60000
    
重要

具体的属性(参数)配置取决于项目需求和数据库的特点,建议您根据实际情况进行调整和配置。

Druid 连接池常用配置参数:

参数

描述

url

指定连接数据库的 URL,包括数据库类型、主机名、端口号、数据库名称等信息。

username

指定连接数据库所需的用户名。

password

指定连接数据库所需的密码。

driverClassName

指定数据库驱动类名。如果不显式地进行配置driverClassName,Druid 连接池会根据url自动识别数据库类型(dbType),然后选择相应的driverClassName。这种自动识别的机制可以减少配置的工作量,简化配置过程。但是需要注意的是,如果url无法被正确解析,或者需要使用非标准的数据库驱动类,那么就需要显式地配置driverClassName参数,以确保正确的驱动类被加载。

initialSize

指定初始化连接池时创建的连接数量。在应用启动时,连接池会创建指定数量的连接并放入连接池中。

maxActive

指定连接池中最大的活动连接数量。当连接池的活动连接数量达到最大值时,后续的连接请求将等待,直到有连接被释放。

maxIdle

指定连接池中最大的空闲连接数量(该属性已被弃用)。当连接池的空闲连接数量达到最大值时,多余的连接将被关闭。

minIdle

指定连接池中最小的空闲连接数量。当连接池的空闲连接数量低于最小值时,连接池会创建新的连接。

maxWait

指定获取连接的最大等待时间,超过此时间将抛出异常。如果设置为正数,则表示等待的毫秒数,超过此时间将抛出异常。

poolPreparedStatements

指定是否缓存PreparedStatement。如果设置为true,则会缓存PreparedStatement对象,提高性能。

validationQuery

指定连接验证的 SQL 查询语句。当连接被从连接池中取出时,会执行此查询语句来验证连接是否有效。

timeBetweenEvictionRunsMillis

指定连接池检测空闲连接的间隔时间,单位为毫秒。如果连接的空闲时间超过了timeBetweenEvictionRunsMillis的设定值,连接将被关闭。

minEvictableIdleTimeMillis

指定连接池中连接的最小空闲时间,超过此时间将被回收,单位为毫秒。如果设置为负数,则表示连接不会被回收。

testWhileIdle

指定是否在空闲时对连接进行测试。如果设置为true,则会在连接空闲时执行validationQuery来验证连接是否有效。

testOnBorrow

指定是否在获取连接时对连接进行测试。如果设置为true,则会在获取连接时执行validationQuery来验证连接是否有效。

testOnReturn

指定是否在归还连接时对连接进行测试。如果设置为true,则会在归还连接时执行validationQuery来验证连接是否有效。

filters

指定连接池中预定义的一系列过滤器。这些过滤器可以按照特定的顺序对连接进行预处理和后处理操作,以提供额外的功能和增强连接池的性能。常见的过滤器包括:

  1. stat:用于统计连接池的性能指标,例如活跃连接数、请求次数、错误次数等。

  2. wall:用于 SQL 防火墙,可以拦截和禁用不安全的 SQL 语句,提高数据库的安全性。

  3. log4j:用于将连接池的日志输出到log4j中,方便日志记录和调试。

  4. slf4j:用于将连接池的日志输出到slf4j中,方便日志记录和调试。

  5. config:用于从外部配置文件中加载连接池的配置信息。

  6. encoding:用于设置连接池和数据库之间的字符编码。

通过将这些过滤器配置在filters属性中,连接池将按照指定的顺序依次应用这些过滤器。可以通过逗号分隔多个过滤器的名称,例如:filters=stat,wall,log4j

Main.java 代码介绍

Main.java文件是本文示例程序的的主程序,通过这个示例,可以了解如何使用数据源、连接对象和各种数据库操作方法来与数据库进行交互。

本文Main.java文件的代码主要包括以下几个部分:

  1. 导入必要的类和接口。

    1. 声明当前代码所属的包名为com.example

    2. 导入 Java 的IOException类,用于处理输入输出异常。

    3. 导入 Java 的InputStream类,用于从文件或其他来源获取输入流。

    4. 导入 Java 的Connection接口,用于表示与数据库的连接。

    5. 导入 Java 的ResultSet接口,用于表示数据库查询结果的数据集。

    6. 导入 Java 的SQLException类,用于处理 SQL 异常。

    7. 导入 Java 的Statement接口,用于执行 SQL 语句。

    8. 导入 Java 的PreparedStatement接口,用于预编译的 SQL 语句。

    9. 导入 Java 的Properties类,用于处理属性文件。

    10. 导入 Java 的DataSource接口,用于管理数据库连接。

    11. 导入 Alibaba Druid 连接池的DruidDataSourceFactory类,用于创建 Druid 数据源。

    代码如下:

    package com.example;
    
    import java.io.IOException;
    import java.io.InputStream;
    import java.sql.Connection;
    import java.sql.ResultSet;
    import java.sql.SQLException;
    import java.sql.Statement;
    import java.sql.PreparedStatement;
    import java.util.Properties;
    import javax.sql.DataSource;
    import com.alibaba.druid.pool.DruidDataSourceFactory;
  2. 创建一个Main类,并定义main方法。

    定义一个Main类和main方法,main方法用于演示如何使用连接池从数据库中执行一系列的操作。具体步骤如下:

    1. 定义一个名为Main的公共类,作为程序的入口点。类名需要与文件名保持一致。

    2. 定义了一个公共静态方法main,作为程序的入口点,接收命令行参数。

    3. 使用异常处理机制来捕获和处理可能发生的异常。

    4. 调用loadPropertiesFile方法,加载属性文件并返回Properties对象。

    5. 调用createDataSource()方法,根据属性文件中的配置创建数据源对象。

    6. 使用try-with-resources语句,获取数据库连接,并在连接使用完后自动关闭连接。

      1. 调用createTable()方法,创建表。

      2. 调用insertData()方法,插入数据。

      3. 调用selectData()方法,查询数据。

      4. 调用updateData()方法,更新数据。

      5. 再次调用selectData()方法,查询已更新的数据。

      6. 调用deleteData()方法,删除数据。

      7. 再次调用selectData()方法,查询删除后的数据。

      8. 调用dropTable()方法,删除表。

    代码如下:

    public class Main {
    
        public static void main(String[] args) {
            try {
                Properties properties = loadPropertiesFile();
                DataSource dataSource = createDataSource(properties);
                try (Connection conn = dataSource.getConnection()) {
                    // Create table
                    createTable(conn);
                    // Insert data
                    insertData(conn);
                    // Query data
                    selectData(conn);
    
                    // Update data
                    updateData(conn);
                    // Query the updated data
                    selectData(conn);
    
                    // Delete data
                    deleteData(conn);
                    // Query the data after deletion
                    selectData(conn);
    
                    // Drop table
                    dropTable(conn);
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    
        // 定义用于获取和使用属性文件中的配置信息的方法
        // 定义用于获取数据源对象的方法
        // 定义用于创建表的方法
        // 定义用于插入数据的方法
        // 定义用于更新数据的方法
        // 定义用于删除数据的方法
        // 定义用于查询数据的方法
        // 定义用于删除表的方法
    }
  3. 定义用于获取和使用属性文件中的配置信息的方法。

    定义一个私有的静态方法loadPropertiesFile(),用于加载属性文件并返回一个Properties对象。具体步骤如下:

    1. 定义一个私有的静态方法loadPropertiesFile(),它返回一个Properties对象并声明可能会抛出IOException异常。

    2. 创建一个Properties对象,用于存储属性文件中的键值对。

    3. 使用try-with-resources语句,通过类加载器获取属性文件db.properties的输入流is

    4. 使用load方法,将输入流中的属性加载到properties对象中。

    5. 返回加载后的properties对象。

    代码如下:

        private static Properties loadPropertiesFile() throws IOException {
            Properties properties = new Properties();
            try (InputStream is = Main.class.getClassLoader().getResourceAsStream("db.properties")) {
                properties.load(is);
            }
            return properties;
        }
  4. 定义用于获取数据源对象的方法。

    定义一个私有的静态方法createDataSource(),用于根据属性文件中的配置创建一个DataSource对象,用于管理和获取数据库连接。具体步骤如下:

    1. 定义一个私有的静态方法createDataSource(),它接收一个Properties对象作为参数,并声明可能会抛出Exception异常。

    2. 调用DruidDataSourceFactory类的createDataSource()方法,传入属性properties,返回一个DataSource对象。

    代码如下:

        private static DataSource createDataSource(Properties properties) throws Exception {
            return DruidDataSourceFactory.createDataSource(properties);
        }
  5. 定义用于创建表的方法。

    定义一个私有的静态方法createTable(),用于在数据库中创建数据表。具体步骤如下:

    1. 定义一个私有的静态方法createTable(),它接收一个Connection对象作为参数,并声明可能会抛出SQLException异常。

    2. 使用try-with-resources语句,通过连接对象conncreateStatement()方法,创建一个Statement对象stmt

    3. 定义一个字符串变量sql,存储创建表的 SQL 语句。

    4. 使用executeUpdate()方法执行 SQL 语句,创建数据表。

    5. 打印出创建表成功的提示信息。

    代码如下:

        private static void createTable(Connection conn) throws SQLException {
            try (Statement stmt = conn.createStatement()) {
                String sql = "CREATE TABLE test_druid (id INT, name VARCHAR(20))";
                stmt.executeUpdate(sql);
                System.out.println("Table created successfully.");
            }
        }
  6. 定义用于插入数据的方法。

    定义一个私有的静态方法insertData(),用于向数据库中插入数据。具体步骤如下:

    1. 定义一个私有的静态方法insertData(),它接收一个Connection对象作为参数,并声明可能会抛出SQLException异常。

    2. 定义一个字符串变量insertDataSql,存储插入数据的 SQL 语句。

    3. 定义一个整型变量insertedRows,初始值为 0,用于记录插入数据的行数。

    4. 使用try-with-resources语句,通过连接对象connprepareStatement()方法和插入数据的 SQL 语句,创建一个PreparedStatement对象insertDataStmt

    5. 使用for循环语句,迭代 5 次,表示插入 5 条数据。

      1. 使用setInt()方法,设置第一个参数的值为循环变量i

      2. 使用setString()方法,设置第二个参数的值为字符串test_insert加上循环变量i的值。

      3. 使用executeUpdate()方法执行插入数据的 SQL 语句,并将返回的操作行数累加到insertedRows变量中。

    6. 打印出插入数据成功的提示信息,以及插入的总行数。

    7. 返回插入的总行数。

    代码如下:

        private static int insertData(Connection conn) throws SQLException {
            String insertDataSql = "INSERT INTO test_druid (id, name) VALUES (?, ?)";
            int insertedRows = 0; 
            try (PreparedStatement insertDataStmt = conn.prepareStatement(insertDataSql)) {
                for (int i = 1; i < 6; i++) {
                    insertDataStmt.setInt(1, i);
                    insertDataStmt.setString(2, "test_insert" + i);
                    insertedRows += insertDataStmt.executeUpdate();
                }
                System.out.println("Data inserted successfully. Inserted rows: " + insertedRows);
            }
            return insertedRows;
        }
  7. 定义用于更新数据的方法。

    定义一个私有的静态方法updateData(),用于更新数据库中的数据。具体步骤如下:

    1. 定义一个私有的静态方法updateData(),它接收一个Connection对象作为参数,并声明可能会抛出SQLException异常。

    2. 使用try-with-resources语句,通过连接对象connprepareStatement()方法和更新数据的 SQL 语句,创建一个PreparedStatement对象pstmt

    3. 使用setString()方法,设置第一个参数的值为字符串test_update

    4. 使用setInt()方法,设置第二个参数的值为整型值 3。

    5. 使用executeUpdate()方法执行更新数据的 SQL 语句,并将返回的操作行数赋值给updatedRows变量。

    6. 打印出更新数据成功的提示信息,以及更新的总行数。

    代码如下:

        private static void updateData(Connection conn) throws SQLException {
            try (PreparedStatement pstmt = conn.prepareStatement("UPDATE test_druid SET name = ? WHERE id = ?")) {
                pstmt.setString(1, "test_update");
                pstmt.setInt(2, 3);
                int updatedRows = pstmt.executeUpdate();
                System.out.println("Data updated successfully. Updated rows: " + updatedRows);
            }
        }
  8. 定义用于删除数据的方法。

    定义一个私有的静态方法deleteData(),用于删除数据库中的数据。具体步骤如下:

    1. 定义一个私有的静态方法deleteData(),它接收一个Connection对象作为参数,并声明可能会抛出SQLException异常。

    2. 使用try-with-resources语句,通过连接对象connprepareStatement()方法和删除数据的 SQL 语句,创建一个PreparedStatement对象pstmt

    3. 使用setInt()方法,设置第一个参数的值为整型值 3。

    4. 使用executeUpdate()方法执行删除数据的 SQL 语句,并将返回的操作行数赋值给deletedRows变量。

    5. 打印出删除数据成功的提示信息,以及删除的总行数。

    代码如下:

        private static void deleteData(Connection conn) throws SQLException {
            try (PreparedStatement pstmt = conn.prepareStatement("DELETE FROM test_druid WHERE id < ?")) {
                pstmt.setInt(1, 3);
                int deletedRows = pstmt.executeUpdate();
                System.out.println("Data deleted successfully. Deleted rows: " + deletedRows);
            }
        }
  9. 定义用于查询数据的方法。

    定义一个私有的静态方法selectData(),用于从数据库中查询数据。具体步骤如下:

    1. 定义一个私有的静态方法selectData(),它接收一个Connection对象作为参数,并声明可能会抛出SQLException异常。

    2. 使用try-with-resources语句,通过连接对象conncreateStatement()方法,创建一个Statement对象stmt

    3. 定义一个字符串变量sql,存储查询数据的 SQL 语句。

    4. 使用executeQuery()方法执行查询数据的 SQL 语句,并将返回的结果集赋值给resultSet变量。

    5. 使用while循环语句,遍历结果集中的每一行数据。

      1. 使用getInt()方法,获取当前行的id字段的整型值,并赋值给id变量。

      2. 使用getString()方法,获取当前行的name字段的字符串值,并赋值给name变量。

      3. 打印出当前行的idname字段的值。

    代码如下:

        private static void selectData(Connection conn) throws SQLException {
            try (Statement stmt = conn.createStatement()) {
                String sql = "SELECT * FROM test_druid";
                ResultSet resultSet = stmt.executeQuery(sql);
                while (resultSet.next()) {
                    int id = resultSet.getInt("id");
                    String name = resultSet.getString("name");
                    System.out.println("id: " + id + ", name: " + name);
                }
            }
        }
    
  10. 定义用于删除表的方法。

    定义一个私有的静态方法dropTable(),用于删除数据库中的表。具体步骤如下:

    1. 定义一个私有的静态方法dropTable(),它接收一个Connection对象作为参数,并声明可能会抛出SQLException异常。

    2. 使用try-with-resources语句,通过连接对象conncreateStatement()方法,创建一个Statement对象stmt

    3. 定义一个字符串变量sql,存储删除表的 SQL 语句。

    4. 使用executeUpdate()方法执行删除表的 SQL 语句。

    5. 打印出删除表成功的提示信息。

    代码如下:

        private static void dropTable(Connection conn) throws SQLException {
            try (Statement stmt = conn.createStatement()) {
                String sql = "DROP TABLE test_druid";
                stmt.executeUpdate(sql);
                System.out.println("Table dropped successfully.");
            }
        }

完整的代码展示

pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.example</groupId>
    <artifactId>druid-mysql-client</artifactId>
    <version>1.0-SNAPSHOT</version>
    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <configuration>
                    <source>8</source>
                    <target>8</target>
                </configuration>
            </plugin>
        </plugins>
    </build>

    <dependencies>
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>5.1.40</version>
        </dependency>
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid</artifactId>
            <version>1.2.8</version>
        </dependency>
    </dependencies>
</project>

db.properties

Database Configuration
driverClassName=com.mysql.jdbc.Driver
url=jdbc:mysql://$host:$port/$database_name?useSSL=false
username=$user_name
password=$password

Connection Pool Configuration
#To check whether the database link is valid, MySQL must be configured to select 1; Oracle is select 1 from dual
validationQuery=select 1
#Initial number of connections
initialSize=3
#Maximum number of activations, that is, the maximum number of Connection pool
maxActive=30
#When closing the Abandoned connection, the error log is output. When the link is recycled, the console prints information. The test environment can add true, while the online environment is false. Will affect performance.
logAbandoned=true
#Minimum number of activations during idle time
minIdle=5
#The maximum waiting time for a connection, in milliseconds
maxWait=1000
#The maximum time to start the eviction thread is the survival time of a link (previous value: 25200000, the converted result of this time is: 2520000/1000/60/60=7 hours)
minEvictableIdleTimeMillis=300000
#Whether to recycle after exceeding the time limit
removeAbandoned=true
#Exceeding the time limit (in seconds), currently 5 minutes. If any business processing time exceeds 5 minutes, it can be adjusted appropriately.
removeAbandonedTimeout=300
Run the idle connection collector Destroy thread every 10 seconds to detect the interval time between connections, based on the judgment of testWhileIdle
timeBetweenEvictionRunsMillis=10000
#When obtaining a link, not verifying its availability can affect performance.
testOnBorrow=false
#Check whether the link is available when returning the link to the Connection pool.
testOnReturn=false
#This configuration can be set to true, without affecting performance and ensuring security. The meaning is: Detect when applying for a connection. If the idle time is greater than timeBetweenEviceRunsMillis, execute validationQuery to check if the connection is valid.
testWhileIdle=true
#Default false, if configured as true, connection detection will be performed in the DestroyConnectionThread thread (timeBetweenEvaluation once)
keepAlive=false
#If keepAlive rule takes effect and the idle time of the connection exceeds it, the connection will only be detected
keepAliveBetweenTimeMillis=60000

Main.java

package com.example;

import java.io.IOException;
import java.io.InputStream;
import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.sql.PreparedStatement;
import java.util.Properties;
import javax.sql.DataSource;
import com.alibaba.druid.pool.DruidDataSourceFactory;

public class Main {

    public static void main(String[] args) {
        try {
            Properties properties = loadPropertiesFile();
            DataSource dataSource = createDataSource(properties);
            try (Connection conn = dataSource.getConnection()) {
                // Create table
                createTable(conn);
                // Insert data
                insertData(conn);
                // Query data
                selectData(conn);

                // Update data
                updateData(conn);
                // Query the updated data
                selectData(conn);

                // Delete data
                deleteData(conn);
                // Query the data after deletion
                selectData(conn);

                // Drop table
                dropTable(conn);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    private static Properties loadPropertiesFile() throws IOException {
        Properties properties = new Properties();
        try (InputStream is = Main.class.getClassLoader().getResourceAsStream("db.properties")) {
            properties.load(is);
        }
        return properties;
    }

    private static DataSource createDataSource(Properties properties) throws Exception {
        return DruidDataSourceFactory.createDataSource(properties);
    }

    private static void createTable(Connection conn) throws SQLException {
        try (Statement stmt = conn.createStatement()) {
            String sql = "CREATE TABLE test_druid (id INT, name VARCHAR(20))";
            stmt.executeUpdate(sql);
            System.out.println("Table created successfully.");
        }
    }

    private static int insertData(Connection conn) throws SQLException {
        String insertDataSql = "INSERT INTO test_druid (id, name) VALUES (?, ?)";
        int insertedRows = 0; 
        try (PreparedStatement insertDataStmt = conn.prepareStatement(insertDataSql)) {
            for (int i = 1; i < 6; i++) {
                insertDataStmt.setInt(1, i);
                insertDataStmt.setString(2, "test_insert" + i);
                insertedRows += insertDataStmt.executeUpdate();
            }
            System.out.println("Data inserted successfully. Inserted rows: " + insertedRows);
        }
        return insertedRows;
    }

    private static void updateData(Connection conn) throws SQLException {
        try (PreparedStatement pstmt = conn.prepareStatement("UPDATE test_druid SET name = ? WHERE id = ?")) {
            pstmt.setString(1, "test_update");
            pstmt.setInt(2, 3);
            int updatedRows = pstmt.executeUpdate();
            System.out.println("Data updated successfully. Updated rows: " + updatedRows);
        }
    }

    private static void deleteData(Connection conn) throws SQLException {
        try (PreparedStatement pstmt = conn.prepareStatement("DELETE FROM test_druid WHERE id < ?")) {
            pstmt.setInt(1, 3);
            int deletedRows = pstmt.executeUpdate();
            System.out.println("Data deleted successfully. Deleted rows: " + deletedRows);
        }
    }

    private static void selectData(Connection conn) throws SQLException {
        try (Statement stmt = conn.createStatement()) {
            String sql = "SELECT * FROM test_druid";
            ResultSet resultSet = stmt.executeQuery(sql);
            while (resultSet.next()) {
                int id = resultSet.getInt("id");
                String name = resultSet.getString("name");
                System.out.println("id: " + id + ", name: " + name);
            }
        }
    }

    private static void dropTable(Connection conn) throws SQLException {
        try (Statement stmt = conn.createStatement()) {
            String sql = "DROP TABLE test_druid";
            stmt.executeUpdate(sql);
            System.out.println("Table dropped successfully.");
        }
    }
}

相关文档

更多 MySQL Connector/J 的信息,请参见 Overview of MySQL Connector/J