浅谈JDBC

前言

我们知道,在进行后端开发的时候,既要精通PHP、Java(JSP)等语言,也要能熟练使用MySQL、SQL Server等数据库系统,那怎样能将两者结合起来呢?JDBC就是其中的一种方式。

简介

JDBC全称为Java Data Base Connectivity,是一种可用于执行SQL语句的Java API(Application Programming Interface,应用程序接口),是连接数据库和Java应用程序的纽带。方便起见,本文使用的数据库为MySQL的5.7版本。

基本操作

首先,我们登录MySQL,创建数据库t1和数据表student_info:

创建数据库和数据表

然后再向表中插入一些数据:

插入数据
连接数据库:

要对数据表中的数据进行操作,应该首先建立与数据库的连接。访问数据库的时候,首先要加载数据库的驱动程序(只需要在第一次访问数据库时加载一次),然后每次访问的时候创建一个Connection类对象,接着执行操作数据库的SQL语句,完成对数据库的操作后再销毁Connection类对象,释放与数据库的连接。下面举个例子:

package test;
import java.sql.*;//导入包
public class Connect { // 创建类Connect
    Connection con; // 声明Connection对象
    public Connection getConnection() {// 定义返回值为Connection的方法
        String driver = "com.mysql.jdbc.Driver"; //driver指向数据库的驱动程序
        String url = "jdbc:mysql://127.0.0.1:3306/t1?framework?characterEncoding=utf8&useSSL=true";
        // url指向要连接的数据库,这里我们访问数据库t1           
        String user = "root";// MySQL配置时的用户名
        String password = "root"; // MySQL配置时的密码  
        try {// 加载数据库驱动类
            Class.forName(driver);
            System.out.println("数据库驱动加载成功");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
        try {// 通过访问数据库的URL获取数据库连接对象
            con = DriverManager.getConnection(url,user,password);
            System.out.println("数据库连接成功");
        } catch (SQLException e) {
            e.printStackTrace();
        }
        return con;
    }
    public static void main(String[] args){
        Connect c = new Conn(); // 创建本类对象
        c.getConnection(); // 调用连接数据库方法
    }
}

jdbc:mysql://127.0.0.1:3306/t2这段代码中,127.0.0.1是本机地址,也可以用localhost代替;t1是要访问的数据库名;而后面的framewor?characterEncoding=utf8&useSSL=true则是新版本MySQL的特性,即需要指明字符的编码格式和是否进行SSL连接,如果不指明的话程序就会报错。rootroot分别是我登录MySQL时的用户名和密码。
这里最关键的是两个方法:Class.forName(驱动程序)DriverManager.getConnection(数据库,用户名,密码) 作用就是加载数据库驱动类和获取数据库连接对象,我们知道Java是面向对象的语言,因此就把数据库的操作转化为了对类和对象的操作,
还有一点就是,这里把连接数据库这一操作定义Connect类,并以Connection对象作为直接调用Connect类的getConnection()方法或者修改方法的代码(要连接的的数据库名)再调用方法就行了,提高效率。
至于是否已成功连接数据库t1呢?我们可以用接下来的几个操作进行检验。

向数据库发送SQL语句:

连接数据库之后,要向数据库发送SQL语句(即指令),才能对数据库中的数据进行一系列操作。要执行SQL语句首先要获得Statement类对象,通过上面创建的Connect类对象concreatedStatement方法可获得Statement类对象:

try{
    Statement sql = con.createStatement();
}catch(SQL Exception e){
  e.printStackTrace();
}
处理查询结果集:

有了Statement类对象后,可以调用相应的方法实现对数据库中数据库的查询和修改,并将查询的结果集存放在ResultSet类对象中:

ResulySet result = sql.executeQuery(SELECT * FROM student_info);

SELECT * FROM student_info这条指令的含义是查询数据表student_info中的所有数据。
ResultSet类对象一次只能显示结果集中的一行数据(即数据表中的一条记录),使用该类的next()方法可将查询位置移向下一行。next()方法的返回值为boolean类型的数据,当查询位置移动到最后一行之后再调用next()方法会返回false.

查询数据:

下面我们来查询数据表student_info中的数据:

package test;
import java.sql.*;
public class Query { // 创建类
    static Connection con; // 声明Connection类对象
    static Statement sql; // 声明Statement类对象
    static ResultSet result; // 声明ResultSet类对象
    public Connection getConnection() { // 连接数据库方法
    /*****连接数据库的代码,此处省略*****/
    return con; // 返回Connection类对象
    }
    public static void main(String[] args) { 
        Query q = new Query(); // 创建本类对象
        con = q.getConnection(); // 连接数据库
          try {
              sql = con.createStatement(); // 实例化Statement类对象
              result = sql.executeQuery("SELECT * FROM student_info");// 执行SQL语句,返回结果集
            while (result.next()) { 
                String id = result.getString("id"); // 获取列名是"id"的字段值
                String name = result.getString("name");// 获取列名是"name"的字段值
                String sex = result.getString("sex");// 获取列名是"sex"的字段值
                System.out.print("编号:" + id); // 将数据输出
                System.out.print(" 姓名:" + name);
                System.out.println(" 性别:" + sex);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

控制台显示的效果为:

控制台显示

可见,通过JDBC查询到的数据和直接登录MySQL查询到的数据是完全相同的。
这个例子是顺序查询数据,在SQL语句中,也可以用LIKE操作符进行模糊查询,即使用%来代替0个或多个字符,使用_来代替一个字符。在使用模糊查询之前,先插入几条数据:

插入数据

如果我们想要查询张姓同学的信息,就可以使用模糊查询,改动如下代码即可:

result = sql.executeQuery("SELECT * FROM student_info WHERE name LIKE '张%'");

控制台显示的效果为:

控制台显示
预处理语句:

向数据库发送一个SQL语句,数据库中的SQL解释器负责把SQL语句生成底层的内部命令然后执行,完成对数据的操作。如果不断地向数据库提交SQL语句,肯定会增加数据库SQL解释器的负担,影响执行的速度。
在JDBC中,可以通过Connection类对象的prepareStatement()方法对SQL语句进行预处理,生成数据库底层的内部命令,并将该命令封装在PreparedStatement类对象中,通过调用该对象的相应方法执行底层数据库命令。这样能够减轻数据库的负担,提高访问数据库的速度。
对SQL进行预处理时可以使用通配符?来代替任何的字段值:

sql = con.prepareStatement("SELECT * FROM student_info WHERE id = ?");

在执行预处理语句恰安,必须用相应方法来设置通配符所表示的值:

 sql.setInt(1,2);

上面语句中的1表示左数第一个通配符,2表示设置的通配符的值。将通配符的值设置为2后,功能等同于:

sql = con.prepareStatement("SELECT * FROM student_info WHERE id = 2");

需要注意的是,通过setXXX()方法为通配符设置值的时候,建议使用与通配符所代表参数匹配的方法,即上面的例子中字段id的类型为int,因此使用setInt()方法。当然也可以使用setObject()为各种类型的参数(通配符)赋值。
下面使用预处理语句查找id为3的同学信息:

 package test;
 import java.sql.*;
 public class Prepare { // 创建类
     static Connection con; // 声明Connection类对象
     static PrepareStatement ps; // 声明PrepareStatement类对象
     static ResultSet result; // 声明ResultSet类对象
     public Connection getConnection() { // 连接数据库方法
     /*****连接数据库的代码,此处省略*****/
     return con; // 返回Connection类对象
     }
     public static void main(String[] args) { 
         Prepare p = new Prepare(); // 创建本类对象
         con = p.getConnection(); // 连接数据库
         try {
                 ps = con.prepareStatement("SELECT * FROM student_info WHERE id = ?");
                 ps.setInt(1, 3); // 设置参数
                 result = ps.executeQuery(); // 执行预处理语句
                 while (result.next()) { 
                     /*****查询和输出数据的代码,此处省略*****/
                 }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

控制台显示的效果为:

控制台显示

尽管使用预处理语句看起来更加繁琐了,但却能使程序动态地改变SQL语句中关于字段值条件的设定。

添加、修改、删除数据:

通过SQL语句可以对数据执行添加、修改和删除操作,可通过PrepareStatement类的指定参数动态地对数据表中的数据进行操作,并使用executeUpdate()方法执行更新语句的操作。
下面对数据表student_info中的数据进行添加、修改和删除:

package test;
import java.sql.*;
public class Test { // 创建类
    static Connection con; // 声明Connection类对象
    static Statement sql; // 声明Statement类对象
    static ResultSet result; // 声明ResultSet类对象
    public Connection getConnection() { // 连接数据库方法
    /*****连接数据库的代码,此处省略*****/
    return con; // 返回Connection类对象
    }
    public static void main(String[] args) { 
        Test t = new Test(); // 创建本类对象
        con = t.getConnection(); // 连接数据库
        try {
            ps = con.prepareStatement("INSERT student_info VALUES(?,?,?)");// 预处理添加数据
            ps.setInt(1, 06); 
            ps.setString(2, "王红");
            ps.setString(3, "女");
            ps.executeUpdate();
            ps = con.prepareStatement("UPDATE student_info SET name = ? WHERE id = 5");
            // 预处理修改数据
            ps.setString(1, "张小娜");
            ps.executeUpdate();
            ps = con.prepareStatement("DELETE FROM student_info WHERE sex = ?");// 预处理删除数据
            ps.setString(1, "男"); 
            ps.executeUpdate();
            result = ps.executeQuery("SELECT * FROM student_info");
            while (result.next()) { 
                /*****查询和输出数据的代码,此处省略*****/
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

控制台显示的效果为:

控制台显示

可见,所有对数据的操作都是正确无误的。

后记

关于JDBC的应用,举个很简单的例子,当我们用JSP做Web后台开发比如一个登录系统的时候,可以将已经注册过的用户信息(用户名,密码,手机号码等)保存到数据库中,也可以判断正在登录用户是否已注册或者用户名或密码是否正确。JSP是基于Java的语言,要在JSP中访问数据库,自然就要用到JDBC了。
可以看到,JDBC其实就是在Java程序中操作数据库,核心就在调用于类中的各种方法,这需要Java的基础,同时,在对数据的各种操作中也利用到了SQL语句,这就要求掌握一定的数据库知识。
总而言之,JDBC并不难,但要熟练使用各种Java方法和SQL语句却不是一件易事,这就需要我们查阅官方文档和平时的积累。
当然,除了JDBC还有其他的后端语言和数据库的连接方式,学习其他语言或者有兴趣的小伙伴可以自行学习。

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

推荐阅读更多精彩内容