Jdbc從入門到入土

来源:https://www.cnblogs.com/coder-baozi/archive/2022/05/19/16290759.html
-Advertisement-
Play Games

二刷jdbc 作者小結:從第一次大概幾天快速刷完jdbc,到如今的二刷,才發現自己對jdbc的理解有點太淺。到學習javaweb是創建資料庫層時的迷茫,到現在對這種設計模式的理解。我深有體會到了:實打實走好每一步的必要性!這篇筆記較為完整的展示了jdbc的發展脈絡,從原理到手動封裝,再到第三方庫,循 ...


二刷jdbc

作者小結:從第一次大概幾天快速刷完jdbc,到如今的二刷,才發現自己對jdbc的理解有點太淺。到學習javaweb是創建資料庫層時的迷茫,到現在對這種設計模式的理解。我深有體會到了:實打實走好每一步的必要性!這篇筆記較為完整的展示了jdbc的發展脈絡,從原理到手動封裝,再到第三方庫,循序漸進。

## jdbc概述
  1. jdbc為訪問不同的資料庫提供了統一的介面。
  2. java程式員使用jdbc,可以連接任何提供了jdbc驅動程式的資料庫系統,從而完成對資料庫的各種操作
  3. jdbc的基本原理圖

image-20220519234042262

java程式通過制定一些介面,讓資料庫廠商實現這些介面

*************************模擬************************
//java制定的資料庫介面
Interface jdbcInterface{
    //連接
    public Object getConnection();
    //crud
    public void crud();
    //關閉連接
    public void close();
}

//mysql廠商繼承介面從而實現這些方法
public class MysqlJdbcImpl implements jdbcInterface{
    @Override
    public Object getConnetion() {
        System.out.println("mysql的實現");
        return null;
    }

    @Override
    public void crud() {

    }

    @Override
    public void close() {

    }
}

java程式使用

public class testjdbc{
    public static void main(String[] args) {
        //通過介面來調用實現類[動態綁定]
        jdbcInterface mysqlImpl = new mysqlImpl();
        //通過介面來調用實現類
        mysqlImpl.getConnetion();
        mysqlImpl.crud();
        mysqlImpl.close();
    }
}

通過介面來調用實現類的意義:(思考介面編程的好處)

​ 當用戶用其他資料庫廠商的實現類時只需動態綁定其他資料庫實現類

​ 例如上方:jdbcInterface mysqlImpl = new DB2Impl();

image-20220519234110459

jdbc程式編寫步驟

  1. 註冊驅動-載入Driver類
  2. 獲取連接-得到Connection
  3. 執行增刪改查-發送sql給相應的資料庫執行
  4. 釋放資源-關閉相關的連接

簡單的步驟:

  1. 導入對應的jar包,即驅動文件

  2. 註冊驅動

    Derver driver=new new com.mysql.jdbc.Driver();

    -->簡寫為 Derver driver=new driver();

  3. String url 解讀

    String url="jdbc::mysql://ip:port/資料庫名"

    jdbc::mysql://規定好的,表示一個協議,通過jdbc的方式連接mysql。

    ip 連接到的主機名稱

    3306 表示mysql監聽的埠

    資料庫名 表示連接到mysql dbms的哪一個資料庫

    image-20220519234216519

​ mysql的連接本質就是socket連接

  1. 將用戶名和密碼放入到properties對象中

    Properties properties = new Properties();
            properties.setProperty("user","book");//用戶
            properties.setProperty("password","xxxx");//密碼
    
  2. 得到連接

    Connection conn=driver.connect(url,properties);
    
  3. 執行sql

    String sql="select * from xxx";
    

    Statement用於執行sql語句

    Statement statement=conn.createStatement();
    int i=statement.excuteUpdate(sql);
    //i表示受影響的行數
    
  4. 關閉資源

    statemen.close();
    conn.close();
    

    不關閉資源造成的影響:會造成連接不到mysql

    image-20220519234243330

獲取資料庫連接的五種方式

1.方式一 獲取Driver實現類對象

Driver driver=new com.mysql.jdbc.Driver();
String url="jdbc:mysql://ip:port/資料庫名";
Properties properties=new Properties();
properties.setProperty("user","name");
properties.setProperty("password","xxxx");
Connection conn=driver.connect(url,properties);

通過new了一個第三方的driver,第三方的dirver 是靜態載入,靈活性不高。

2.方式二 使用反射機制,動態載入。

//使用反射載入Driver類
        Class clazz=Class.forName("com.mysql.jdbc.Driver");
        Driver driver= (Driver) clazz.newInstance();
        String url="jdbc:mysql://ip:port/資料庫名";
        Properties properties=new Properties();
        properties.setProperty("user","name");
        properties.setProperty("password","xxxx");
        Connection conn=driver.connect(url,properties);

反射動態載入,更加靈活,減少依賴性。

3.方式三使用DriverManager進行統一管理

//使用反射載入Driver
Class clazz=Class.forName("com.mysql.jdbc.Driver");
Driver driver= (Driver) clazz.newInstance();
//創建url user password
String url="jdbc:mysql://ip:port/資料庫名";
String user="root";
String password="xxxx";
//註冊Driver驅動
DriverManager.registerDriver(driver);
Connection conn=DriverManager.getConnection(url,user,password);

DriverManagaer用於管理一組jdbc驅動程式的基本服務

4.方式四使用forName()自動完成註冊驅動,簡化代碼--推薦使用

//使用反射載入Driver
Class clazz=Class.forName("com.mysql.jdbc.Driver");
//創建url user password
String url="jdbc:mysql://ip:port/資料庫名";
String user="root";
String password="xxxx";
Connection conn=DriverManager.getConnection(url,user,password);

Class.forName 在載入Driver類時自動完成了註冊

image-20220519234303076

tip:沒用顯示調用Class.forName("com.mysql.jdbc.Driver")仍然可以拿到資料庫的連接。建議寫上,更加明確

image-20220519234332516

image-20220519234346973

5.方式五通過寫配置文件,讓連接更加靈活

//通過Peoperties對象獲取配置文件的信息
        Properties properties=new Properties();
        properties.load(new FileInputStream("com\\mysql.properties"));
        //通過key獲取相關的值
        String user = properties.getProperty("user");
        String password = properties.getProperty("password");
        String url = properties.getProperty("url");
        String driver = properties.getProperty("driver");

        Class.forName(driver);
        DriverManager.getConnection(url,user,password);

在方式四的基礎上,增加配置文件,讓連接更加靈活。

配置文件:

#key=value
user=root
password=xxx
url=jdbc:mysql://ip:port/資料庫名
driver=com.mysql.jdbc.Driver

ResultSet結果集--底層(?)

概述:表示資料庫結果集的數據表,通常通過查詢資料庫的語句生成。ResultSet對象保持一個游標指向其當前的數據行,最初游標位於第一行,next方法將游標移動到下一行,並且由於在ResultSet對象中沒有更多行時返回false。類似於迭代器。

		//得到Statemen
        Statement statement = connection.createStatement();
        //sql語句
        String sql="SELECT * FROM xxx";
        //執行給定的sql語句,該語句返回單個ResultSet對象即為一張表
        java.sql.ResultSet resultSet = statement.executeQuery(sql);
        //迴圈取出
        while(resultSet.next()){//讓游標向後移動,如果沒有更多行,則返回false
            resultSet.getInt(1);//獲取改行的第一列數據
            resultSet.getString(2);//獲取該行第二列
        }
        //關閉資源
        resultSet.close();
        statement.close();
        connection.close();

statement--存在sql註入問題

概述:用於執行靜態的sql語句並返回其生成的結果的對象

statement是一個介面需要不同的資料庫廠商實現

解決方案:使用preperdStatement

代碼實現:

		//得到Statemen
        Statement statement = connection.createStatement();
        //sql語句xx
        String sql="SELECT * FROM xxx";
        //執行給定的sql語句,該語句返回單個ResultSet對象即為一張表
        java.sql.ResultSet resultSet = statement.executeQuery(sql);

如果將用戶輸入改成next()也可以防止sql註入,next遇到空格會停止。

PreperdStatement

概述:預處理Statement,是一個介面。

用法:

1.PreperdStatement執行的sql語句中的參數用問號(?)來表示,調用PreperdStatement對象的setXXX()方法來設置這些參數。setXXX()方法有兩個參數,第一個參數是要設置的sql語句中的參數的索引(即第幾個問號),第二個設置的是參數的值。image-20220518193746580

2.調用executeQuery(),返回ResultSet對象

3.調用excuteUpdate(),執行crud。返回影響行數

預處理好處:

​ 不在使用+拼接sql語句,減少語法錯誤,有效解決了sql註入問題,減少了編譯次數,效率較高。

預處理就是再執行sql之前就已經完成對sql的賦值。

代碼實現:

//sql語句xx,設置問號
String sql="SELECT * FROM xxx WHERE name=?and password=?";
//得到PreparedStatemen
PreparedStatement preparedStatement = connection.prepareStatement(sql);
//給問號賦值
preparedStatement.setString(1,user_name);
preparedStatement.setString(2,user_pass);
//執行註意執行的時候不需要再填sql
preparedStatement.executeQuery();

添加記錄dml

String sql="INSERT INTO xxx VALUES(?,?)";

API小結

image-20220518201735923

image-20220518202015875

封裝Utils類

簡介:在jdbc操作中,獲取資料庫連接和釋放資源是經常使用到的可以將其封裝為JDBC連接的工具類Utils。

image-20220518203730303

使用步驟:

  1. 定義相關的屬性(4個),因為只需要一份,所以用static修飾
  2. 在static代碼塊初始化
  3. 通過配置文件讀取相關的屬性值
  4. 寫連接函數,推薦使用DriverManager
  5. 寫釋放資源函數

代碼實現:

//定義相關的屬性(4個),因為只需要一份,所以用static修飾
    private static String user;//用戶名
    private static String password;//密碼
    private static String url;//資料庫url
    private static String driver;//驅動名

    //在static代碼塊初始化
    static{

        try {
            Properties properties=new Properties();
            properties.load(new FileInputStream("com\\mysql.properties"));
            //讀取相關的屬性值
            user = properties.getProperty("user");
            password=properties.getProperty("password");
            url=properties.getProperty("url");
            driver=properties.getProperty("driver");
        } catch (IOException e) {
            //在實際開發中,常常轉為運行異常拋出
            //將編譯異常轉為運行異常,調用者可以選擇捕獲該異常,也可以選擇預設處理該異常,比較方便。
            throw new RuntimeException(e);
        }
    }
    //連接資料庫,返回Connection
    public static Connection getConnection() throws SQLException, ClassNotFoundException {
        Class.forName(driver);
        return DriverManager.getConnection(url,user,password);
    }
    //關閉相應資源
    /*
        可能關閉的資源
        1.ResultSet結果集
        2.Statement和preparedStatement
        3.connection
        4.如果需要關閉資源,則傳入對象,否則傳入null
     */
    //用statement來接受因為statement是preparedStatement的父介面,都可以接收
    /*
    當一個對象被當作參數傳遞到一個方法後,此方法可改變這個對象的屬性,並可返回變化後的結果,那麼這裡到底是值傳遞還是引用傳遞?
    Java 編程語言只有值傳遞參數。當一個對象實例作為一個參數被傳遞到方法中時,參數的值就是對該對象的引用。對象的內容可以在被調用的方法中改變,但對象的引用是永遠不會改變的。
     */
    public static void close(ResultSet resultSet, Statement state,Connection conn) throws SQLException {
            if(resultSet!=null){
                resultSet.close();
            }
            if(state!=null){
                state.close();
            }
            if(conn!=null){
                state.close();
            }
    }

實際開發過程中異常處理:

在實際開發中,常常轉為運行異常拋出,將編譯異常轉為運行異常,調用者可以選擇捕獲該異常,也可以選擇預設處理該異常,比較方便。throw new RuntimeException(e);

Java是值傳遞:

Java 編程語言只有值傳遞參數。當一個對象實例作為一個參數被傳遞到方法中時,參數的值就是對該對象的引用。對象的內容可以在被調用的方法中改變,但對象的引用是永遠不會改變的。

Utils使用

使用步驟:

  1. 得到連接。
  2. 組織一個sql語句。
  3. 創建一個PreparedStatement對象。
  4. 執行sql語句。
  5. 釋放資源調用close()。

代碼實現:

public class use_utils {
    public void use_ut() throws SQLException {
        Connection conn=null;
        String sql="SELECT * FROM xxx";
        PreparedStatement preparedStatement=null;
        try {
            //得到連接
            conn=jdbcutils.getConnection();
            //創建PreparedStaement
            preparedStatement= conn.prepareStatement(sql);
            preparedStatement.executeQuery();
        } catch (SQLException e) {
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }finally {
            jdbcutils.close(null,preparedStatement,conn);
        }
    }
}

事務

概述:Jdbc程式中當一個Connection對象創建時,預設情況下是自動提交事務,不能回滾。並且jdbc程式中為了讓多個SQL語句作為一個整體執行,需要使用事務,調用Connection的setAutoCommit(false)可以取消自動提交事務,當所有的sql語句都執行後,調用Commit()方法即可提交事務,在其中某個操作失敗或出現異常時,調用rollback()方法即可回滾事務。

預設情況下,Connection對象是自動提交的。

應用實例:經典的轉賬業務。

代碼實現: "未使用事務"

Connection conn=null;
        String sql="update account set balance=balance-100 where id=1";
        String sql2="update account set balance=balance+100 where id=2";
        PreparedStatement preparedStatement=null;
        try {
            //得到連接
            conn=jdbcutils.getConnection();
            //創建PreparedStaement
            preparedStatement= conn.prepareStatement(sql);
            preparedStatement.executeQuery();//執行第一條sql
            int i=1/0;//拋出異常
            preparedStatement=conn.prepareStatement(sql2);
            preparedStatement.executeQuery();//執行第二條SQL
    }

上述代碼,在執行sql時如果沒用開啟事務,會造成第一條sql執行成功,而第二條sql未執行便被捕獲異常,在轉賬問題方面就會出現問題。

代碼實現: 開啟事務

//得到連接
      conn=jdbcutils.getConnection()            //創建PreparedStaement
preparedStatement= conn.prepareStatement(sql);
/*得到連接後將conn設置為不自動提交*/
conn.setAutoCommit(false);
preparedStatement.executeQuery();//執行第一條sql
            int i=1/0;//拋出異常
            preparedStatement=conn.prepareStatement(sql2);
            preparedStatement.executeQuery();//執行第二條SQL
/*在catch中即可處理異常,撤消先前已經執行的sql*/
				/*即回滾*/
catch(Exception e){
    conn.rollback();
}

rollback預設回滾到事務開啟的地方。

批處理

概述:當需要成批插入或者更新數據時,可以採用Java批量更新機制,這一機制允許將多條語句一次性提交給資料庫批量處理。

批處理步驟:

  1. 如果使用批處理時,需要在url中添加參數:

    ?rewriteBatchedStatements=true

  2. addBatch():添加需要批量處理的SQL語句或參數

  3. excuteBatch():執行批量處理的語句

  4. clearBatch():清空批處理包的語句

批處理優勢:批處理往往和PreparedStatement一起搭配使用,既可以減少編譯次數,又減少運行次數,讓效率提高。

代碼實現:

傳統代碼

public void nobatch() throws SQLException, ClassNotFoundException {
        Connection connection = jdbcutils.getConnection();
        String sql="insert into xxx values(null,?,?)";
        PreparedStatement preparedStatement = connection.prepareStatement(sql);
        for (int i = 0; i < 5000; i++) {
            preparedStatement.setString(1,"tom"+i);
            preparedStatement.setString(2,"xxx");
            preparedStatement.executeUpdate();
        }
        //關閉連接
        jdbcutils.close(null,preparedStatement,connection);
 }

批處理代碼

public void batch_() throws SQLException, ClassNotFoundException {
        Connection connection = jdbcutils.getConnection();
        String sql="insert into xxx values(null,?,?)";
        PreparedStatement preparedStatement = connection.prepareStatement(sql);
        for (int i = 0; i < 5000; i++) {
            preparedStatement.setString(1,"tom"+i);
            preparedStatement.setString(2,"xxx");
            /*preparedStatement.executeUpdate();*/
            //將sql語句加入到批處理包中 ->
            preparedStatement.addBatch();
            //當有1000條數據時,在批量執行
            if((i+1)%1000==0){
                //滿1000條
                preparedStatement.executeBatch();
                //清空
                preparedStatement.clearBatch();
            }
        }
        //關閉連接
        jdbcutils.close(null,preparedStatement,connection);
    }

關鍵代碼:

preparedStatement.addBatch();
//當有1000條數據時,在批量執行
if((i+1)%1000==0){
//滿1000條
preparedStatement.executeBatch();
//清空
preparedStatement.clearBatch();
}

註:源代碼未瞭解!

資料庫連接池

概述:傳統方式連接資料庫過多,由於沒用的連接資源未被及時斷開會造成,連接不上數s據庫,資料庫連接池就誕生了,資料庫連接池可以合理分配連接資源。

實現方式:

1.預先在緩衝池中放入一定數量的連接,當需要建立資料庫連接時,只需要從 "緩衝池"中取出一個,使用完畢之後再放回去。

2.資料庫連接池負責分配,管理和釋放資料庫連接,它允許應用程式重覆使用一個現有的資料庫連接,而不是重寫建立一個

3.當應用程式向連接池請求的連接數超過最大連接數量時,這些請求將被加入到等待隊列中。

image-20220518233952426

image-20220518234252208

資料庫連接池種類:

1.Jdbc的資料庫連接池使用java.sql.dateSource來表示,DateSource只是一個介面,該介面由第三方提供實現。

2.C3P0, DBCP, Proxool, BoneCP, Druid

C3P0

實現步驟:

一、傳統方式

1.創建一個數據源對象。

2.通過配置文件獲取相關的信息user,url...

3.給數據源ComboPooledDataSource(c3p0)設置相關的參數url,user...setInitialPoolSize()方法設置初始化連接數,setMaxPoolSize()方法設置最大連接數。

4.得到連接。

5.關閉連接--即放回到連接池中。

二、使用配置文件模板

概述:c3p0設計者提供了一個xml文件,方便配置

1.將C3P0提供的配置文件 c3p0.config.xml拷貝到src目錄下

2.創建一個數據源對象,參數即為c3p0.config.xml文件中的

3.得到連接

4.關閉連接--即放回到連接池中。

C3P0:

image-20220518235502696

配置文件:c3p0.config.xml

<c3p0-config>
    <!-- 數據源名稱代表連接池-->
    <name-config name="xxx">
        <!--  連接參數 -->
        <property name="driverClass">com.mysql.jdbc.Driver</property>
        <property name="jdbcUrl">jdbc:mysql://localhost:3306/web</property>
        <property name="user">root</property>
        <property name="password">ROOT</property>

        <!-- 連接池參數 -->
        <!-- 每次增長連接池可供連接數 -->
        <property name="acquireIncrement">10</property>
        <!-- 初始連接數 -->
        <property name="initialPoolSize">5</property>
        <!-- 最大連接數 -->
        <property name="maxPoolSize">10</property>
        <!-- 最大等待時間 -->
        <property name="checkoutTimeout">2000</property>
        <!-- 最大空閑回收時間 -->
        <property name="maxIdleTime">1000</property>
</c3p0-config>

代碼實現:

傳統方式

    public void testc3p0() throws IOException, PropertyVetoException, SQLException {
        //1.創建一個數據源對象
        ComboPooledDataSource comboPooledDataSource = new ComboPooledDataSource();
        //2.通過配置文件獲取相關的信息
        Properties properties=new Properties();
        properties.load(new FileInputStream("com\\mysql.properties"));
        String user = properties.getProperty("user");
        String password = properties.getProperty("password");
        String url = properties.getProperty("url");
        String driver = properties.getProperty("driver");
        //給數據源 comboPooledDataSource設置相關的參數。
        //我們連接的管理是由comboPooledDataSource來管理的。
        comboPooledDataSource.setDriverClass(driver);
        comboPooledDataSource.setUser(user);
        comboPooledDataSource.setPassword(password);
        comboPooledDataSource.setJdbcUrl(url);
        //設置連接數--初始化連接數
        comboPooledDataSource.setInitialPoolSize(10);
        //最大連接數
        comboPooledDataSource.setMaxPoolSize(50);

        Connection connection = comboPooledDataSource.getConnection();//這個方法就是從DateSource介面實現的
        connection.close();
    }

xml配置文件的方式

public void test04() throws SQLException {
        //1.將配置文件導入src目錄下

        //2.創建一個數據源對象,參數即為c3p0.config.xml文件中的 <name-config name="nihao">
        ////數據源會根據數據源名稱讀取xml文件中的內容
        ComboPooledDataSource comboPooledDataSource = new ComboPooledDataSource("nihao");

        Connection connection = comboPooledDataSource.getConnection();

        connection.close();
    }

:數據源名稱不能寫錯,並且xml文件的名稱是固定的,數據源會根據數據源名稱讀取xml文件中的內容,自動完成配置。

Druid

概述:

Druid連接池是阿裡實現的,獲取連接的速度比較快。

實現步驟:

1.添加jar包,和properties配置文件,將配置文件拷貝到項目的src目錄下。

​ driverClassName 底層用這個欄位來讀取資料庫驅動。

​ minIdle 空閑時候的連接數量

2.創建Properties對象來讀取配置文件

3.創建一個指定參數的資料庫連接池

4.得到連接

5.釋放連接

代碼實現:

public void druidx() throws Exception {
        /*
        1.添加jar包,和properties配置文件,將配置文件拷貝到項目的src目錄下。
		driverClassName 底層用這個欄位來讀取資料庫驅動。
		minIdle 空閑時候的連接數量

        2.創建Properties對象來讀取配置文件
         */
        Properties properties = new Properties();
        properties.load(new FileInputStream("src\\druid.properties"));
        //創建一個指定參數的資料庫連接池
        DataSource dataSource = DruidDataSourceFactory.createDataSource(properties);
    	//得到連接
        Connection connection = dataSource.getConnection();
        connection.close();
    }

配置文件: druid.properties

driverClassName=com.mysql.jdbc.Driver //驅動載入
url=jdbc:mysql://127.0.0.1:3306/student?characterEncoding=utf-8 //註冊驅動
username=root //連接資料庫的用戶名
password=sjw58586 //連接資料庫的密碼。
filters=stat //屬性類型的字元串,通過別名的方式配置擴展插件, 監控統計用的stat 日誌用log4j 防禦sql註入:wall
initialSize=2 //初始化時池中建立的物理連接個數。
maxActive=300 //最大的可活躍的連接池數量
maxWait=60000 //獲取連接時最大等待時間,單位毫秒,超過連接就會失效。配置了maxWait之後,預設啟用公平鎖,併發效率會有所下降, 如果需要可以通過配置useUnfairLock屬性為true使用非公平鎖。
timeBetweenEvictionRunsMillis=60000 // 連接回收器的運行周期時間,時間到了清理池中空閑的連接,testWhileIdle根據這個判斷
minEvictableIdleTimeMillis=300000
validationQuery=SELECT 1 //用來檢測連接是否有效的sql,要求是一個查詢語句。
testWhileIdle=true //建議配置為true,不影響性能,並且保證安全性。 申請連接的時候檢測,如果空閑時間大於timeBetweenEvictionRunsMillis, 執行validationQuery檢測連接是否有效。
testOnBorrow=false //申請連接時執行validationQuery檢測連接是否有效,做了這個配置會降低性能。設置為false
testOnReturn=false //歸還連接時執行validationQuery檢測連接是否有效,做了這個配置會降低性能,設置為flase
poolPreparedStatements=false //是否緩存preparedStatement,也就是PSCache。
maxPoolPreparedStatementPerConnectionSize=200 // 池中能夠緩衝的preparedStatements語句數量

將Jdbc工具類改成druid實現

代碼實現:

 private static DataSource ds;
    //在靜態代碼塊完成ds初始化
    static{
        Properties properties = new Properties();
        try {
            properties.load(new FileInputStream("src\\druid.properties"));
            ds= DruidDataSourceFactory.createDataSource(properties);
        } catch (Exception e) {
            new RuntimeException(e);
        }
    }
    //得到連接方法
    public Connection getConnection() throws SQLException {
        return ds.getConnection();
    }
    //釋放資源,Connection放回連接池,此是的close是資料庫連接池實現的close方法。
    public void close(ResultSet rs, Connection conn, Statement st) throws SQLException {
            if(rs!=null){
                rs.close();
            }
            if (conn!=null){
                conn.close();
            }
            if(st!=null){
                st.close();
            }
    }

註:此時調用的close方法,是連接池實現的close()方法並不會真正的關閉連接,而是將連接放回到資料庫連接池。

Bean,Domain,POJO

問題引出:

1.java程式使用Connection連接,Connection和ResultSet關聯,當關閉Connection無法再使用ResultSet。

2.如果一個程式返回ResultSet對象,這是已經關閉Connection,仍然無法拿到結果集對象,導致結果集只能使用一次。

解決方案:

寫一個Java類---->常被叫做Bean,Domain,POJO,該類中有查詢到的表中的欄位屬性,讓一個Actor對象對應查詢到的一條記錄,將結果集封裝到ArrayList中。

image-20220519180317108

ApachDBUtils

概述:面對ResultSet問題,ApachDBUtils工具類完美解決了這個問題。

傳統方式解決(土方法封裝)

實現步驟:

1.新建一個POJO類,用於封裝查詢到的表的記錄。

2.類中定義表中相對應的欄位,構造函數,setter,getter方法。

​ 註:一定要給一個預設構造函數[反射需要]。

3.在java程式中用ArrayList 來存貯數據。

4.在ResultSet遍歷的時候,封裝數據,存儲到集合中。

代碼實現:

Actor---POJO

import java.util.Date;

public class Actor {//POJO
    //和表的欄位相對應
    //細節建議用Integer裝箱
    private Integer id;
    private String name;
    private String sex;
    //Date用util包下的。
    private Date bornDate;
    private String phone;
    //一定要給一個無參構造器[反射會需要]
    public Actor(){

    }

    public Actor(Integer id, String name, String sex, Date bornDate, String phone) {
        this.id = id;
        this.name = name;
        this.sex = sex;
        this.bornDate = bornDate;
        this.phone = phone;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public void setName(String name) {
        this.name = name;
    }

    public void setSex(String sex) {
        this.sex = sex;
    }

    public void setBornDate(Date bornDate) {
        this.bornDate = bornDate;
    }

    public void setPhone(String phone) {
        this.phone = phone;
    }

    public Integer getId() {
        return id;
    }

    public String getName() {
        return name;
    }

    public String getSex() {
        return sex;
    }

    public Date getBornDate() {
        return bornDate;
    }

    public String getPhone() {
        return phone;
    }
}

java程式:

public void testSelecttoArraylist() throws Exception {
        Properties properties=new Properties();
        properties.load(new FileInputStream("com\\mysql.properties"));
        //通過key獲取相關的值
        String user_name;
        String user_pass;
        Scanner scanner = new Scanner(System.in);
        user_name=scanner.nextLine();
        user_pass=scanner.nextLine();
        String user = properties.getProperty("user");
        String password = properties.getProperty("password");
        String url = properties.getProperty("url");
        String driver = properties.getProperty("driver");
        //創建ArrayList對象,存放Actor對象
        ArrayList<Actor> list=new ArrayList<>();
        //註冊驅動
        Class.forName(driver);
        //得到連接
        Connection connection = DriverManager.getConnection(url, user, password);
        //sql語句xx,設置問號
        String sql="SELECT * FROM xxx WHERE name=?and password=?";
        //得到PreparedStatemen
        PreparedStatement preparedStatement = connection.prepareStatement(sql);
        //給問號賦值
        preparedStatement.setString(1,user_name);
        preparedStatement.setString(2,user_pass);
        //執行給定的sql語句,該語句返回單個ResultSet對象即為一張表
        java.sql.ResultSet resultSet = preparedStatement.executeQuery(sql);
        //迴圈取出
        while(resultSet.next()){//讓游標向後移動,如果沒有更多行,則返回false
            int id=resultSet.getInt("id");
            String name = resultSet.getString("name");
            String sex = resultSet.getString("sex");
            Date borndate = resultSet.getDate("borndate");
            String phone = resultSet.getString("phone");
            //把得到的resultSet記錄封裝到Actor對象,放入到list集合。
            list.add(new Actor(id,name,sex,borndate,phone));
        }
        //關閉資源
        resultSet.close();
        preparedStatement.close();
        connection.close();


    }
}

ApachDBUtils解決

概述:commons-dbutils是Apache組織提供的一個開源的JDBC工具類庫,它是對JDBC的封裝,使用dbutils能極大簡化jdbc編碼的工作量。

dbutils常用類和介面:

  1. QueryRunner類:該類封裝了SQl的執行,是線程安全的,可以實現增刪改查,批處理。
  2. ResultSetHandler介面:該介面用於處理java.sql.ResultSet,將數據按要求轉換為另一種形式。

ArrayHandler: 把結果集中的第一行數據轉成對象數組
ArrayListHandler: 把結果集中的每一行數據都轉成一個數組,再存放到List中。
BeanHandler: 將結果集中的第一行數據封裝到一個對應的JavaBean實例中。
BeanListHandler: 將結果集中的每一行數據都封裝到一個對應的JavaBean: 實例中,存放到List里。
ColumnListHandler: 將結果集中某一列的數據存放到List中。
KeyedHandler(name): 將結果集中的每行數據都封裝到Map里,再把這些map再存到一個map里,其key為指定的key。
MapHandler: 將結果集中的第一行數據封裝到一個Map里,key是列名,vaue就是對應的值。
MapListHandler: 將結果集中的每一行數據都封裝到一個Map里,然後再存放到List

使用步驟:druid+dbutils

  1. 引入commons jar包

  2. 使用自己封裝的DruidUtils得到連接

  3. 創建QueryRunner

  4. 調用QueryRunner的query方法執行sql返回ArrayList集合

    註:sql語句也可以查詢部分列

    BeanListHandler<>(Actor.class):在將ResultSet->Actor 對象->封裝到ArrayList

    1: 傳給sql中的問號的,可以有多個後邊是可變參數Object...params

    底層得到的ResultSet會在query執行後關閉,PreparedStatement也會自動關閉

    參數列表
    (connection, sql, new BeanListHandler<>(Actor.class), 1)
    
  5. 關閉連接。

代碼實現:

public void querytest() throws SQLException {
        //得到連接(druid)
        Connection connection = DruidUtils.getConnection();
        //使用DbUtils類和介面,引入相應的jar文件。
        //創建QueryRunner
        QueryRunner queryRunner = new QueryRunner();
        //QueryRunner就可以執行相關的方法,返回ArrayList結果集
        String sql="SELECT * FROM xxx WHERE id>=?";
        //sql語句也可以查詢部分列
        /*
         queryRunner.query方法就是執行一個sql語句得到ResultSet,--封裝到ArrayList集合中
         然後返回集合。
         參數Connection  連接
         sql: 執行的sql語句
         BeanListHandler<>(Actor.class):在將ResultSet->Actor 對象->封裝到ArrayList
         底層使用反射機制獲取Actor屬性進行封裝
         1: 傳給sql中的問號的,可以有多個後邊是可變參數Object...params
         底層得到的ResultSet會在query執行後關閉,PreparedStatement也會自動關閉
        * */
        List<Actor> list = queryRunner.query(connection, sql, new BeanListHandler<>(Actor.class), 1);
        DruidUtils.close(null,connection,null);
    }

dbutils查詢單行記錄(單個對象)

解決方案:返回單行記錄,單個對象使用的Hander是BeanHandler

new BeanHandler<>(Actor.class)

單行單列:返回的是一個Object,對象使用Handers是ScalarHandler()

new ScalarHandler()

dbutils+druid實現crud

執行dml操作使用queryRunner.update()

返回值是受影響的行數。

代碼實現:

public void testcrud() throws Exception{
        Connection connection = DruidUtils.getConnection();
        QueryRunner queryRunner = new QueryRunner();
        String sql="update actor set name=? where id=?";
        int affectedrows = queryRunner.update(connection, sql, "niuma", 4);

    }

對應關係:

註: java中全部對應包裝類。

image-20220519215702289

BasicDAO

問題分析:

dbutils和druid簡化了JDBC開發,但有不足

1.Sql語句是固定,不能通過參數傳入,通用性不好,需要進行該進,更方便執行crud

2.對於select操作,如果有返回值,返回值類型不能固定,需要使用泛型

3.將來的表很多,業務需求複雜,不可能只靠一個Java類完成

設計理念:各司其職。一張表對應一個DAO

DAO:訪問數據的對象

image-20220519222455869

image-20220519223149361

BasicDAO將所有的DAO共有的部分提取出來,讓子類去繼承簡化代碼。

設計步驟:

1.第一個包 放utils工具類

2.第二個包 javaBean/domain

3.第三個包 放xxxDAO和BasicDAO

4.第四個包 寫測試類

目錄結構:

image-20220519231821656

代碼:

BasicDAO:

package com.basic_.dao;

import com.basic_.utils.DruidUtils;
import org.apache.commons.dbutils.QueryRunner;
import org.apache.commons.dbutils.handlers.BeanHandler;
import org.apache.commons.dbutils.handlers.BeanListHandler;
import org.apache.commons.dbutils.handlers.ScalarHandler;

import javax.management.Query;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;

//開發basicDAO
//添加泛型將來是操作domain的
public class BasicDAO<T> {
    private QueryRunner qr=new QueryRunner();

    //開發通用的dml方法,針對任意的表
    public int update(String sql,Object...prameters) throws SQLException {
        Connection conn=null;
        conn= DruidUtils.getConnection();
        int affectedRows = qr.update(conn, sql, prameters);
        return affectedRows;
    }
    //返回多個對象(即查詢的結果是多行的),針對任意表
    //sql語句可以有問號,占位符
    //clazz傳入一個類的Class對象,底層通過反射實現domain 比如Actor.class
    //prameters傳入問號具體的值。
    public List<T> queryMulti(String sql, Class<T> clazz, Object...prameters) throws SQLException {
        Connection conn=null;
        conn=DruidUtils.getConnection();
        List<T> list = qr.query(conn, sql, new BeanListHandler<>(clazz), prameters);
        DruidUtils.close(null,conn,null);
        return list;
    }
    //查詢單行結果通用方法   T可能是Actor..不同的表。
    public T querySingle(String sql,Class<T> clazz,Object...prameters) throws SQLException {
        Connection conn=null;
        conn= DruidUtils.getConnection();
        T rs = qr.query(conn, sql, new BeanHandler<>(clazz), prameters);
        DruidUtils.close(null,conn,null);
        return rs;
    }
    //查詢單行單列即返回單值的方法
    public Object queryScalar(String sql,Object...prameters) throws SQLException {
        Connection conn=null;
        conn= DruidUtils.getConnection();
        Object rs = qr.query(conn, sql, new ScalarHandler(), prameters);
        DruidUtils.close(null,conn,null);
        return rs;
    }

}

ActorDAO

public class ActorDAO extends BasicDAO<Actor>{
    //Actor有BasicDAO 的方法
    //根據業務需求,可以編寫特有的方法
}

domain

public class Actor {//POJO
    //和表的欄位相對應
    //細節建議用Integer裝箱
    private Integer id;
    private String name;
    private String sex;
    //Date用util包下的。
    private Date bornDate;
    private String phone;
    //一定要給一個無參構造器[反射會需要]
    public Actor(){

    }

    public Actor(Integer id, String name, String sex, Date bornDate, String phone) {
        this.id = id;
        this.name = name;
        this.sex = sex;
        this.bornDate = bornDate;
        this.phone = phone;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public void setName(String name) {
        this.name = name;
    }

    public void setSex(String sex) {
        this.sex = sex;
    }

    public void setBornDate(Date bornDate) {
        this.bornDate = bornDate;
    }

    public void setPhone(String phone) {
        this.phone = phone;
    }

    public Integer getId() {
        return id;
    }

    public String getName() {
        return name;
    }

    public String getSex() {
        return sex;
    }

    public Date getBornDate() {
        return bornDate;
    }

    public String getPhone() {
        return phone;
    }
}

utils

    private static DataSource ds;
    //在靜態代碼塊完成ds初始化
    static{
        Properties properties = new Properties();
        try {
            properties.load(new FileInputStream("src\\druid.properties"));
            ds= DruidDataSourceFactory.createDataSource(properties);
        } catch (Exception e) {
            new RuntimeException(e);
        }
    }
    //得到連接方法
    public static Connection getConnection() throws SQLException {
        return ds.getConnection();
    }
    //釋放資源,Connection放回連接池,此是的close是資料庫連接池實現的close方法。
    public static void close(ResultSet rs, Connection conn, Statement st) throws SQLException {
            if(rs!=null){
                rs.close();
            }
            if (conn!=null){
                conn.close();
            }
            if(st!=null){
                st.close();
            }
    }

測試使用

//測試ActorDao對actor表crud操作
    public void testActorDao() throws SQLException {
        ActorDAO actorDAO = new ActorDAO();
        //1.查詢
        List<Actor> actors = actorDAO.queryMulti("select *from actor where id>?", Actor.class, 1);
        //2.查詢單行
        Actor actor01 = actorDAO.querySingle("select *from actor where id=?", Actor.class, 1);
        //3.查詢單行單列
        Object o = actorDAO.queryScalar("select name from actor where id=?", 6);
        //4.dml操作
        //添加數據是按照什麼順序呢?
        int afftedRows = actorDAO.update("insert into actor values(null,?,?)", "nihao", "niuma");

    }

作者:程式員包子,轉載請註明原文鏈接:https://www.cnblogs.com/coder-baozi/p/16290759.html

coder-baozi一位菜鳥碼農


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 轉自:https://www.evanlin.com/maglev/ 2016 年 6 月 2 日 前言(為什麼想讀這一篇論文) 這一篇論文吸引我註意的原因是,Consistent Hashing本來的特性就是作為分散式緩存之用。谷歌將他們的負載均衡器(代號:Maglev)發佈他的實作方式,裡面將一 ...
  • 文章已收錄至https://lichong.work,轉載請註明原文鏈接。 ps:歡迎關註公眾號“Fun肆編程”或添加我的私人微信交流經驗🤝 一.Nginx安裝配置及常用命令 1.環境搭建 首先在linux系統下安裝gcc編譯環境,執行: yum install gcc-c++ -y 確保當前系統 ...
  • 來源:blog.csdn.net/qq_15371293/article/details/117090780 項目場景: ClickHouse 操作基於 Mybatis-puls源碼擴展開發。解決ClickHouse的修改和刪除 SQL操作與Mysql不相同。 基於 Mybatis-puls: up ...
  • 個人學習筆記總結。Basic Types、Strings、Arrays, Slices, and Maps、Control Statements、Declarations & Types、Formatted & File I/O、 Functions, Parameters、Defer、Closur... ...
  • #一、概述 ##1.1、Ribbon Spring Cloud Ribbon是基於Netflix Ribbon實現的一套客戶端負載均衡的工具。簡單的說,Ribbon是Netflix發佈的開源項目,主要功能是提供客戶端的軟體負載均衡演算法和服務調用。Spring Cloud Ribbon雖然只是一個工具 ...
  • ##基本介紹 ###1.變數 定義:可以變化的量 ###2.變數聲明 Java是一種強制類型語言,每一個變數必須聲明類型 ###3.變數名,變數類型和作用域 Java變數是程式中最基本的存儲單元,其要素包括變數名,變數類型和作用域 type varName [=value] [{,varName[= ...
  • ​ 概述 服務編排是Fizz網關提供的一個強大的功能,能夠基於現有的業務微服務通過線上配置的方式快速的生成一個聚合介面,減少中間層膠水代碼以及降低編碼投入。在服務編排中支持使用函數,本進階教程中我們分三篇文章(上篇:列表展開&合併、中篇:列表提取&關聯、下篇:列表欄位重命名&欄位移除)來介紹數據列表 ...
  • 來源:zhihu.com/question/23084473 今天我們聊一個不常見的 Java 面試題:為什麼資料庫連接池不採用 IO 多路復用? 這是一個非常好的問題。IO多路復用被視為是非常好的性能助力器。但是一般我們在使用 DB 時,還是經常性採用c3p0,tomcat connection ...
一周排行
    -Advertisement-
    Play Games
  • 概述:在C#中,++i和i++都是自增運算符,其中++i先增加值再返回,而i++先返回值再增加。應用場景根據需求選擇,首碼適合先增後用,尾碼適合先用後增。詳細示例提供清晰的代碼演示這兩者的操作時機和實際應用。 在C#中,++i 和 i++ 都是自增運算符,但它們在操作上有細微的差異,主要體現在操作的 ...
  • 上次發佈了:Taurus.MVC 性能壓力測試(ap 壓測 和 linux 下wrk 壓測):.NET Core 版本,今天計劃準備壓測一下 .NET 版本,來測試並記錄一下 Taurus.MVC 框架在 .NET 版本的性能,以便後續持續優化改進。 為了方便對比,本文章的電腦環境和測試思路,儘量和... ...
  • .NET WebAPI作為一種構建RESTful服務的強大工具,為開發者提供了便捷的方式來定義、處理HTTP請求並返迴響應。在設計API介面時,正確地接收和解析客戶端發送的數據至關重要。.NET WebAPI提供了一系列特性,如[FromRoute]、[FromQuery]和[FromBody],用 ...
  • 原因:我之所以想做這個項目,是因為在之前查找關於C#/WPF相關資料時,我發現講解圖像濾鏡的資源非常稀缺。此外,我註意到許多現有的開源庫主要基於CPU進行圖像渲染。這種方式在處理大量圖像時,會導致CPU的渲染負擔過重。因此,我將在下文中介紹如何通過GPU渲染來有效實現圖像的各種濾鏡效果。 生成的效果 ...
  • 引言 上一章我們介紹了在xUnit單元測試中用xUnit.DependencyInject來使用依賴註入,上一章我們的Sample.Repository倉儲層有一個批量註入的介面沒有做單元測試,今天用這個示例來演示一下如何用Bogus創建模擬數據 ,和 EFCore 的種子數據生成 Bogus 的優 ...
  • 一、前言 在自己的項目中,涉及到實時心率曲線的繪製,項目上的曲線繪製,一般很難找到能直接用的第三方庫,而且有些還是定製化的功能,所以還是自己繪製比較方便。很多人一聽到自己畫就害怕,感覺很難,今天就分享一個完整的實時心率數據繪製心率曲線圖的例子;之前的博客也分享給DrawingVisual繪製曲線的方 ...
  • 如果你在自定義的 Main 方法中直接使用 App 類並啟動應用程式,但發現 App.xaml 中定義的資源沒有被正確載入,那麼問題可能在於如何正確配置 App.xaml 與你的 App 類的交互。 確保 App.xaml 文件中的 x:Class 屬性正確指向你的 App 類。這樣,當你創建 Ap ...
  • 一:背景 1. 講故事 上個月有個朋友在微信上找到我,說他們的軟體在客戶那邊隔幾天就要崩潰一次,一直都沒有找到原因,讓我幫忙看下怎麼回事,確實工控類的軟體環境複雜難搞,朋友手上有一個崩潰的dump,剛好丟給我來分析一下。 二:WinDbg分析 1. 程式為什麼會崩潰 windbg 有一個厲害之處在於 ...
  • 前言 .NET生態中有許多依賴註入容器。在大多數情況下,微軟提供的內置容器在易用性和性能方面都非常優秀。外加ASP.NET Core預設使用內置容器,使用很方便。 但是筆者在使用中一直有一個頭疼的問題:服務工廠無法提供請求的服務類型相關的信息。這在一般情況下並沒有影響,但是內置容器支持註冊開放泛型服 ...
  • 一、前言 在項目開發過程中,DataGrid是經常使用到的一個數據展示控制項,而通常表格的最後一列是作為操作列存在,比如會有編輯、刪除等功能按鈕。但WPF的原始DataGrid中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...