亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java中JDBC連接數(shù)據(jù)庫詳解

 更新時間:2017年03月16日 11:35:08   作者:Dmego  
本文主要介紹了JDBC連接數(shù)據(jù)庫的相關(guān)知識,具有很好的參考價值。下面跟著小編一起來看下吧

一、概念

  1. 為了能讓程序操作數(shù)據(jù)庫,對數(shù)據(jù)庫中的表進行操作,每一種數(shù)據(jù)庫都會提供一套連接和操作該數(shù)據(jù)庫的驅(qū)動,而且每種數(shù)據(jù)庫的驅(qū)動都各不相同,例如mysql數(shù)據(jù)庫使用mysql驅(qū)動,oracle數(shù)據(jù)庫使用oracle驅(qū)動,這樣假如我們編寫的程序哪一天想要換數(shù)據(jù)庫,那樣就會很不方便,因為所有連接數(shù)據(jù)庫的代碼都要從新編寫。SUN公司為了簡化。統(tǒng)一對數(shù)據(jù)庫的操作,定義了一套java操作數(shù)據(jù)庫的標(biāo)準(zhǔn)或者規(guī)范,這個規(guī)范就是JDBC。

  2.JDBC全稱為:Java Data Base Connectivity(java數(shù)據(jù)庫連接),它主要由接口組成。我們在開發(fā)過程中,只要實現(xiàn)它相應(yīng)的接口就可以非常進行連接?!   ?/p>

  3.我們在開發(fā)JDBC應(yīng)用時,還需要導(dǎo)入相應(yīng)的數(shù)據(jù)庫的驅(qū)動jar包,這些驅(qū)動jar包是由數(shù)據(jù)庫公司自己編寫的。

二、編寫JDBC應(yīng)用程序(需要連接數(shù)據(jù)庫的程序)的前提準(zhǔn)備

1.首先要確定連接的是哪個數(shù)據(jù)庫實例,例如在mysql中,我們可以先創(chuàng)建一個庫,然后在庫中新建一張表,在表中插入一些數(shù)據(jù),我在這里提供一段在mysql數(shù)據(jù)庫中創(chuàng)建一個庫,以及表和數(shù)據(jù)的sql語句,這也是下面連接數(shù)據(jù)庫后操作的庫和表。

create database test ; /*創(chuàng)建一個名為Test的數(shù)據(jù)庫*/
use test;        /*使用該數(shù)據(jù)庫或者說切換到該數(shù)據(jù)庫*/
create table book (
  id int primary key auto_increment, /*列:id ,類型:int,從0開始,自動增加, 備注:主鍵*/
  name varchar(40) NOT NULL,    /*列:name ,類型:varchar, 備注:非空*/
  author varchar(40)NOT NULL, /*列:author ,類型:varchar, 備注:非空*/
  prices double NOT NULL  /*列:prices ,類型:double, 備注:非空*/
); /*新建一張名為book的表*/
/*插入四大名著的數(shù)據(jù)*/
insert into book(id,name,author,prices) values (null,'西游記','吳承恩',25.00);
insert into book(id,name,author,prices) values (null,'水滸傳','施耐庵',30.00);
insert into book(id,name,author,prices) values (null,'紅樓夢','曹雪芹',35.00);
insert into book(id,name,author,prices) values (null,'三國演義','羅貫中',40.00);

2.新建一個java項目,然后把mysql的驅(qū)動jar包導(dǎo)入進來,即添加到程序運行的庫中,具體的驅(qū)動jar包,我們可以在數(shù)據(jù)庫的安裝目錄下找到,或者都網(wǎng)上自己下載相對應(yīng)的數(shù)據(jù)庫驅(qū)動jar包

三、連接數(shù)據(jù)庫操作的步驟解析

(1)注冊數(shù)據(jù)庫驅(qū)動

雖然我們剛才在新建java項目的時候?qū)ysql數(shù)據(jù)庫的驅(qū)動jar包導(dǎo)入進來了,但是JBDC不知道這里有一個驅(qū)動包,此時我們就需要將這個驅(qū)動包交給JBDC去管理,我們可以使用java.sql包下的DriverManager 工具類 提供的registerDriver(Driver driver) 方法來在JDBC中注冊這個數(shù)據(jù)驅(qū)動,這個registerDriver(Driver driver)方法需要一個Driver對象,而這個Driver類本身是JDBC提供的一個接口,我們的驅(qū)動里面已經(jīng)實現(xiàn)了這個接口,所以我們只需要寫如下代碼就可以實現(xiàn)注冊數(shù)據(jù)庫驅(qū)動的功能

import java.sql.DriverManager; //需要導(dǎo)入的是接口類包
DriverManager.registerDriver(new Driver());

(2)獲取(創(chuàng)建)數(shù)據(jù)庫的連接

我們注冊好數(shù)據(jù)庫驅(qū)動后,并沒有連接上數(shù)據(jù)庫,以往,我們不管在CMD窗口下,通過可視化數(shù)據(jù)庫管理工具操作數(shù)據(jù)庫時,我們都需要先連接數(shù)據(jù)庫服務(wù)器,java程序連接數(shù)據(jù)庫也不例外,這里的java程序就相當(dāng)于客戶端,只有先連接上數(shù)據(jù)庫服務(wù),才能對數(shù)據(jù)庫進行操作

客戶端與數(shù)據(jù)庫所有交互都是通過connection對象完成的,這個對象的常用方法:

createStatement():創(chuàng)建向數(shù)據(jù)庫發(fā)送sql的statement對象。

prepareStatement(sql) :創(chuàng)建向數(shù)據(jù)庫發(fā)送預(yù)編譯sql的

這里我們可以通過DriverManager 工具類里的getConnection(url,user,password)方法來創(chuàng)建數(shù)據(jù)庫連接對象,此方法需要傳入三個參數(shù):

User: 數(shù)據(jù)庫的用戶名

Password:用戶密碼

URL:數(shù)據(jù)庫服務(wù)器地址,不同的數(shù)據(jù)庫的URL寫法不同,我在這里提供三種主流數(shù)據(jù)庫的URL地址寫法:

Oracle寫法:jdbc:oracle:thin:@localhost:1521:sid

SqlServe寫法:jdbc:microsoft:sqlserver://localhost:1433; DatabaseName=sid

MySql寫法:jdbc:mysql://localhost:3306/sid

Mysql的url地址的簡寫形式: jdbc:mysql:///sid

注:后面的sid就是數(shù)據(jù)庫的實例名稱(使用的數(shù)據(jù)庫名)

import java.sql.Connection; //導(dǎo)入的是接口類包
Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/test", "root", "root"); //這里使用的是一個名為test的mysql數(shù)據(jù)庫,用戶名和密碼都是root 

(3)創(chuàng)建傳輸器對象

上面我們已經(jīng)創(chuàng)建了數(shù)據(jù)庫的連接,已經(jīng)連上數(shù)據(jù)庫了,但是如果我們想要操作該數(shù)據(jù)庫,我們需要用到sql語句,而我們怎樣使用在java程序中使用sql語句來操作數(shù)據(jù)庫呢,這里我們就需要一個傳輸器對象來傳輸sql語句到數(shù)據(jù)庫中去執(zhí)行。上文提到在Connection 類中就有一個createStatement()的方法可以創(chuàng)建一個傳輸器對象

import java.sql.Statement; //導(dǎo)入的是接口類包
Statement stat = conn.createStatement();

(4)利用傳輸器對象傳輸sql語句到數(shù)據(jù)庫中執(zhí)行操作,將結(jié)果用結(jié)果集返回

java.sql.Statement身上有許多傳輸sql語句的方法:其中用的最多的是

executeQuery(String sql) :用于向數(shù)據(jù)發(fā)送查詢語句。

executeUpdate(String sql):用于向數(shù)據(jù)庫發(fā)送insert、update或delete語句

execute(String sql):用于向數(shù)據(jù)庫發(fā)送任意sql語句

import java.sql.ResultSet; //需要導(dǎo)入的接口類包
ResultSet rs = stat.executeQuery("select * from book"); //傳輸一條查詢語句,查詢book表中所有的元組數(shù)據(jù)

(5) 遍歷結(jié)果集,并獲取查詢對象

Jdbc程序中的ResultSet用于代表Sql語句的執(zhí)行結(jié)果。Resultset封裝執(zhí)行結(jié)果時,采用的類似于表格的方式。ResultSet 對象維護了一個指向表格數(shù)據(jù)行的游標(biāo),初始的時候,游標(biāo)在第一行之前,調(diào)用ResultSet.next() 方法,可以使游標(biāo)指向具體的數(shù)據(jù)行,進行調(diào)用方法獲取該行的數(shù)據(jù)。

ResultSet既然用于封裝執(zhí)行結(jié)果的,所以該對象提供的都是用于獲取數(shù)據(jù)的get方法:

獲取指定類型的數(shù)據(jù),例如:

getString(int index)

getString(String columnName)

ResultSet還提供了對結(jié)果集進行滾動的方法:

next():移動到下一行

Previous():移動到前一行

absolute(int row):移動到指定行

beforeFirst():移動resultSet的最前面。

afterLast() :移動到resultSet的最后面。

while(rs.next())
 {
 String name = rs.getString("name");
 System.out.println(name);
}

(6)關(guān)閉連接(先創(chuàng)建的后關(guān)閉)

Jdbc程序運行完后,切記要釋放程序在運行過程中,創(chuàng)建的那些與數(shù)據(jù)庫進行交互的對象,這些對象通常是ResultSet, Statement和Connection對象。

特別是Connection對象,它是非常稀有的資源,用完后必須馬上釋放,如果Connection不能及時、正確的關(guān)閉,極易導(dǎo)致系統(tǒng)宕機。Connection的使用原則是盡量晚創(chuàng)建,盡量早的釋放。

rs.close(); 
stat.close();
conn.close(); 

初期完整的源代碼

package jdbcDemo;
/****************************
 * 初版連接數(shù)據(jù)庫程序
 **************************/
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
//不能導(dǎo)入 java.sql 中的 Driver 接口,要導(dǎo)入驅(qū)動jar包中實現(xiàn)該接口的類,只有這要才能注冊相對應(yīng)的數(shù)據(jù)庫驅(qū)動
import com.mysql.jdbc.Driver; 
public class JDBCTest {
 public static void main(String[] args) throws SQLException {
 //1.注冊數(shù)據(jù)庫驅(qū)動
 DriverManager.registerDriver(new Driver());
 //2.獲取數(shù)據(jù)庫的連接 
 Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/test?useSSL=false", "root", "root");
 //3.創(chuàng)建傳輸器對象
 Statement stat = conn.createStatement();
 //4.利用傳輸器對象傳輸sql語句到數(shù)據(jù)庫中執(zhí)行操作,將結(jié)果用結(jié)果集返回
  ResultSet rs = stat.executeQuery("select * from book");
 //5.遍歷結(jié)果集,并獲取查詢結(jié)果
 while(rs.next()) {
  String name = rs.getString("name");
  System.out.println(name);
 }
 //6.關(guān)閉連接(后開先關(guān))
 rs.close(); 
 stat.close();
 conn.close();
 } 
}

數(shù)據(jù)表視圖和運行結(jié)果:

四、初期連接數(shù)據(jù)庫程序中出現(xiàn)的問題

1--注冊數(shù)據(jù)庫驅(qū)動方法不當(dāng)導(dǎo)致出現(xiàn)了兩次注冊,程序通用性低

我們在查看Driver類的源碼中可以看到如下代碼,從第7行代碼中我們可以看到,mysql在Driver類的實現(xiàn)中自己注冊了一次,而我們在程序中又注冊了一次,導(dǎo)致注冊兩次

我們在注冊驅(qū)動時,需要導(dǎo)入mysql驅(qū)動jar包中已經(jīng)實現(xiàn)的Driver類,這樣程序就和具體的數(shù)據(jù)庫綁定在一起了,程序的通用性就降低了,如果我們想要切換數(shù)據(jù)庫,還得改動源碼

public class Driver extends NonRegisteringDriver implements java.sql.Driver {
 //
 // Register ourselves with the DriverManager
 //
 static {
 try {
  java.sql.DriverManager.registerDriver(new Driver());
 } catch (SQLException E) {
  throw new RuntimeException("Can't register driver!");
 }
 }

修復(fù)方法:

使用Class.forname() 方法將mysql中已經(jīng)實現(xiàn)的Driver類加載到程序中來,由于Driver類在實現(xiàn)接口時使用的是靜態(tài)代碼塊,而靜態(tài)代碼塊只會在類加載的時候執(zhí)行一次,即保證了數(shù)據(jù)庫驅(qū)動只會被注冊一次,同時不用導(dǎo)入mysql驅(qū)驅(qū)動里的類包,程序通用性提高

Class.forName("com.mysql.jdbc.Driver");

2--忽略了程序中可能會拋出的異常(最大的問題)

我們在執(zhí)行程序時,它的許多方法的調(diào)用都會拋出異常,如果它拋出異常后,沒有做相應(yīng)的處理(catch 這個異常)那么程序就會中斷執(zhí)行,Statement對象和Connection對象就沒有被關(guān)閉,而我們知道Connection對象,它是非常稀有的資源,用完后必須馬上釋放,如果Connection不能及時、正確的關(guān)閉,極易導(dǎo)致系統(tǒng)宕機,所以我們需要保證無論程序中哪一步出現(xiàn)了異常導(dǎo)致程序中斷,連接關(guān)閉的代碼都會被執(zhí)行,此時我們就會想到異常處理中的finally代碼塊,我們可以把異常向上拋出,而是先 try 住然后 catch 異常,最后執(zhí)行 finally 代碼塊

修改之后,我們發(fā)現(xiàn)每個close() 都提示有異常要處理,此時我們也直接 try/catch 每個異常

修改后的源代碼:

package jdbcDemo;
/****************************
 * 修改版連接數(shù)據(jù)庫程序
 **************************/
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
public class JDBCTest {
 public static void main(String[] args) {
 Connection conn = null;
 Statement stat = null;
 ResultSet rs = null;
 try {
  //1.注冊數(shù)據(jù)庫驅(qū)動
  Class.forName("com.mysql.jdbc.Driver");
  //2.獲取數(shù)據(jù)庫的連接 
   conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/test?useSSL=false", "root", "root");
  //3.創(chuàng)建傳輸器對象
   stat = conn.createStatement();
  //4.利用傳輸器對象傳輸sql語句到數(shù)據(jù)庫中執(zhí)行操作,將結(jié)果用結(jié)果集返回
   rs = stat.executeQuery("select * from book");
  //5.遍歷結(jié)果集,并獲取查詢結(jié)果
  while(rs.next()) {
   String name = rs.getString("name");
   System.out.println(name);
  }
 }catch(Exception e) {
  e.printStackTrace();
 }finally {
  //6.關(guān)閉連接(后開先關(guān))
  try {
  rs.close();
  } catch (SQLException e) {
  e.printStackTrace();
  } 
  try {
  stat.close();
  } catch (SQLException e) {
  e.printStackTrace();
  }
  try {
  conn.close();
  } catch (SQLException e) {
  e.printStackTrace();
  }
 }
 } 
}

五、修改后程序中被忽略的異常

異常問題

1.由于我們在程序開頭先聲明了三個對象的引用,并且都賦值為null,假如程序在執(zhí)行到注冊數(shù)據(jù)庫這一步時就拋出了異常,此時catch 到這個異常 后執(zhí)行finally 代碼塊,結(jié)果發(fā)現(xiàn)ResultSet 對象的引用,Connection對象的引用以及Statement對象的引用都是空值,調(diào)用這個對象上的方法就會拋出空指針異常

2.close()這個方法身上也有異常,如果我們不做相應(yīng)的異常處理,那些對象還是不能被正常關(guān)閉

解決辦法

1---為了防止出現(xiàn)空指針異常,我們可以先判斷哪些對象的引用是否為null,如果不為null,則執(zhí)行異常處理代碼

2---在每個close()異常處理后在加上一個finally靜態(tài)代碼塊,將每個相應(yīng)對象的引用值置為null,原理是:如果程序執(zhí)行到close() 方法并拋出了異常,那么最后finally代碼塊執(zhí)行,給該對象的應(yīng)用值置為null,由于這個對象沒有任何引用指向它,它就成為了垃圾對象,JVM垃圾回收器就會回收這個對象資源,這個對象也就關(guān)閉了

異常處理完后最終的源代碼:

package jdbcDemo;
/****************************
 * 無異常版連接數(shù)據(jù)庫程序
 **************************/
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
public class JDBCTest {
 public static void main(String[] args) {
 Connection conn = null;
 Statement stat = null;
 ResultSet rs = null;
 try {
  //1.注冊數(shù)據(jù)庫驅(qū)動
  Class.forName("com.mysql.jdbc.Driver");
  //2.獲取數(shù)據(jù)庫的連接 
   conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/test?useSSL=false", "root", "root");
  //3.創(chuàng)建傳輸器對象
   stat = conn.createStatement();
  //4.利用傳輸器對象傳輸sql語句到數(shù)據(jù)庫中執(zhí)行操作,將結(jié)果用結(jié)果集返回
   rs = stat.executeQuery("select * from book");
  //5.遍歷結(jié)果集,并獲取查詢結(jié)果
  while(rs.next()) {
   String name = rs.getString("name");
   System.out.println(name);
  }
 }catch(Exception e) {
  e.printStackTrace();
 }finally {
  //6.關(guān)閉連接(后開先關(guān))
  if(rs != null) {
   try {
   rs.close();
   } catch (SQLException e) {
   e.printStackTrace();
   } finally {
   rs = null;
   }
  }
  if(stat != null) {
   try {
   stat.close();
   } catch (SQLException e) {
   e.printStackTrace();
   }finally {
   stat = null;
   }
  }
  if(conn != null) {
   try {
   conn.close();
   } catch (SQLException e) {
   e.printStackTrace();
   }finally {
   conn = null;
   }
  } 
 } //--finally
 } //--main 
}//--class

總結(jié):

這里我只是把所有的異常處理完,但是程序的通用性還不是特別高,因為連接數(shù)據(jù)庫用到的Driver類名、URL、user以及password 都寫在程序中,我們其實可以寫在一個文本文件中,通過對文件的讀取來獲得每種數(shù)據(jù)庫特有的連接參數(shù)

還有就是在實際開發(fā)過程中,連接數(shù)據(jù)庫的程序代碼一般會寫在一個工具類中,我們想要對數(shù)據(jù)庫中的數(shù)據(jù)進行操作時,只需要調(diào)用這個工具類就可以了,不用每次都寫那么多代碼

接下來我還會更新一篇如何將數(shù)據(jù)庫的連接信息保存在文本文件中,然后讀取這個文件來實現(xiàn)連接數(shù)據(jù)庫的操作,同時我也會將這個程序修改為連接數(shù)據(jù)庫的一個工具類

以上就是本文的全部內(nèi)容,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,同時也希望多多支持腳本之家!

相關(guān)文章

  • Java包機制及javadoc詳解

    Java包機制及javadoc詳解

    為了更好地組織類,Java提供了包機制,用于區(qū)別類名的命名空間,一般利用公司域名倒置作為包名,這篇文章主要介紹了Java包機制以及javadoc,需要的朋友可以參考下
    2022-10-10
  • eclipse項目在IDEA中打開并運行的詳細圖文教程

    eclipse項目在IDEA中打開并運行的詳細圖文教程

    這篇文章主要給大家介紹了關(guān)于eclipse項目在IDEA中打開并運行的詳細圖文教程,至從使用IDEA開發(fā)工具以來,不少次有使用IDEA運行Eclipse項目或非Maven項目,所以這里給大家總結(jié)下,需要的朋友可以參考下
    2023-09-09
  • spring boot如何使用POI讀取Excel文件

    spring boot如何使用POI讀取Excel文件

    本文主要介紹使用POI進行Excel文件的相關(guān)操作,涉及讀取文件,獲取sheet表格,對單元格內(nèi)容進行讀寫操作,以及合并單元格的操作
    2021-08-08
  • Flowable歷史查詢實例分析

    Flowable歷史查詢實例分析

    這篇文章主要介紹了Flowable歷史查詢實例分析,歷史是記錄流程執(zhí)行過程中發(fā)生的事情,并將其永久存儲的組件,與運行時數(shù)據(jù)不同,歷史數(shù)據(jù)在流程實例完成以后仍保存在數(shù)據(jù)庫中,下面我們來深入了解
    2023-10-10
  • 集成apollo動態(tài)日志取締logback-spring.xml配置

    集成apollo動態(tài)日志取締logback-spring.xml配置

    這篇文章主要為大家介紹了集成apollo動態(tài)日志取締logback-spring.xml配置的過程內(nèi)容詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助
    2022-02-02
  • 詳解用Kotlin寫一個基于Spring Boot的RESTful服務(wù)

    詳解用Kotlin寫一個基于Spring Boot的RESTful服務(wù)

    這篇文章主要介紹了詳解用Kotlin寫一個基于Spring Boot的RESTful服務(wù) ,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-05-05
  • Java Swing組件復(fù)選框JCheckBox用法示例

    Java Swing組件復(fù)選框JCheckBox用法示例

    這篇文章主要介紹了Java Swing組件復(fù)選框JCheckBox用法,結(jié)合具體實例形式分析了Swing復(fù)選框JCheckBox簡單用法與相關(guān)操作注意事項,需要的朋友可以參考下
    2017-11-11
  • Java中List.of()和Arrays.asList()的區(qū)別及原因分析

    Java中List.of()和Arrays.asList()的區(qū)別及原因分析

    這篇文章主要介紹了Java中List.of()和Arrays.asList()的區(qū)別及原因分析,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-09-09
  • 一文帶你搞懂Java類加載機制

    一文帶你搞懂Java類加載機制

    Java?類加載機制是?Java?運行時的核心組成部分,負(fù)責(zé)在程序運行過程中動態(tài)加載和連接類文件,并將其轉(zhuǎn)換為可執(zhí)行代碼,接下來小編就來帶大家搞懂面試官老問的?Java?類加載機制,需要的朋友可以參考下
    2023-08-08
  • Spring使用IOC與DI實現(xiàn)完全注解開發(fā)

    Spring使用IOC與DI實現(xiàn)完全注解開發(fā)

    IOC也是Spring的核心之一了,之前學(xué)的時候是采用xml配置文件的方式去實現(xiàn)的,后來其中也多少穿插了幾個注解,但是沒有說完全采用注解實現(xiàn)。那么這篇文章就和大家分享一下,全部采用注解來實現(xiàn)IOC + DI
    2022-09-09

最新評論