JDBC的使用

1.介绍

JDBC(Java Database Connectivity)是一个独立于特定数据库管理系统、通用的SQL数据库存取和操作的公共接口(一组API),定义了用来访问数据库的标准Java类库,(java.sql,javax.sql) 使用这些类库可以以一种标准的方法、方便地访问数据库资源。
JDBC为访问不同的数据库提供了一种统一的途径,为开发者屏蔽了一些细节问题

JDBC接口(API) 包括两个层次 :

  • 面向应用的APl: Java API,抽象接口,供应用程序开发人员使用( 连接数据库,执行SOL语句,获得结果)。
  • 面向数据库的API:Java Driver API,供开发商开发数据库驱动程序用

2.JDBC程序编写步骤

在这里插入图片描述

3.获取数据库连接

java.sql.Driver 接口是所有JDBC驱动程序需要实现的接口。这个接口是提供给数据库厂商使用的,不同数据库厂商提供不同的实现。
在程序中不需要直接去访问实现了 Driver 接口的类,而是由驱动程序管理器类(java.sql.DriverManager)去调用这些Driver实现。

  • Oracle的驱动 :oracle.jdbc.driver.OracleDriver
  • mysql的驱动: com.mysql.cj.jdbc.Driver

下面是把数据源信息放在Properties文件中读取,获取数据库连接:

//读取配置文件
InputStream inputStream = ConnectionTest.class.getClassLoader().getResourceAsStream("jdbc.properties");
Properties pros = new Properties();
pros.load(inputStream);

String user = pros.getProperty("user");
String password = pros.getProperty("password");
String url = pros.getProperty("url");
String driverClass = pros.getProperty("driverClass");

//加载驱动
Class.forName(driverClass);

//获取连接
Connection connection = DriverManager.getConnection(url, user, password);
System.out.println(connection);
user=root
password=root
url=jdbc:mysql://localhost:3306/exercise
driverClass=com.mysql.cj.jdbc.Driver

因为一般由 DriverManager.getConnection来获取连接,获取Driver的实现类之后要向DriverManager注册驱动,这一步在源码中已经写好了,我们只需要用Class.forName(driverClass)加载驱动就可以了

在这里插入图片描述

4.操作和访问数据库

数据库连接被用于向数据库服务器发送命令和 SQL 语句,并接受数据库服务器返回的结果。其实一个数据库连接就是一个Socket连接
在java.sql包中有 3 个接口分别定义了对数据库的调用的不同方式:

  • Statement: 用于执行静态 SOL 语句并返回它所生成结果的对象。
  • PreparedStatement: SQL 语句被预编译并存储在此对象中,可以使用此对象多次高效地执行该语句
  • CallableStatement: 用于执行 SQL 存储过程

4.1 使用Statement操作数据库存在弊端

问题一: 存在拼串操作,繁琐
问题二: 存在SQL注入问题

SQL注入是利用某些系统没有对用户输入的数据进行充分的检查,而在用户输入数据中注入非法的 SQL 语句段或命令(如 : SELECT user, password FROM user_table WHERE user=‘a’ OR 1=‘AND password =’ OR ‘1’=‘1’),从而利用系统的 SQL引擎完成恶意行为的做法。

对于Java 而言,要防范SQL注入,只要用 PreparedStatement(从Statement扩展而来)取代 Statement就可以了。

4.2 使用PreparedStatement操作数据库

4.2.1 插入

首先有一个customers表,表结构如下

在这里插入图片描述

插入一条数据

Connection connection = null;
PreparedStatement preparedStatement = null;
try {
    //1.读取配置文件
    InputStream inputStream =ClassLoader.getSystemClassLoader().getResourceAsStream("jdbc.properties");
    Properties pros = new Properties();
    pros.load(inputStream);

    String user = pros.getProperty("user");
    String password = pros.getProperty("password");
    String url = pros.getProperty("url");
    String driverClass = pros.getProperty("driverClass");

    //2.加载驱动
    Class.forName(driverClass);

    //3.获取连接
    connection = DriverManager.getConnection(url, user, password);

    //4.预编译sql语句,返回PreparedStatement的实例
    String sql = "insert into customers(name,email,birth) values(?,?,?)";	//? 表示占位符
    preparedStatement = connection.prepareStatement(sql);

    //5.填充占位符
    preparedStatement.setString(1,"小智");
    preparedStatement.setString(2,"123@qq.com");
    SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
    Date date = sdf.parse("1999-01-01");
    preparedStatement.setDate(3,new java.sql.Date(date.getTime()));

    //6.执行操作
    preparedStatement.execute();
} catch (Exception e) {
    e.printStackTrace();
}finally {
    //7.关闭资源
    try {
        if(preparedStatement != null)
            preparedStatement.close();
    } catch (SQLException e) {
        e.printStackTrace();
    }
    try {
        if(connection != null)
            connection.close();
    } catch (SQLException e) {
        e.printStackTrace();
    }
}

结果:

在这里插入图片描述

4.2.2 封装JDBCUtils

作用:获取连接,关闭连接。这些操作都是一样的,所以封装到一个类里,方便使用

public class JDBCUtils {

    /**
     * 获取连接
     * @return
     * @throws Exception
     */
    public static Connection getConnection() throws Exception{
        //读取配置文件
        InputStream inputStream = ClassLoader.getSystemClassLoader().getResourceAsStream("jdbc.properties");
        Properties pros = new Properties();
        pros.load(inputStream);

        String user = pros.getProperty("user");
        String password = pros.getProperty("password");
        String url = pros.getProperty("url");
        String driverClass = pros.getProperty("driverClass");

        //加载驱动
        Class.forName(driverClass);

        //获取连接
        Connection connection = DriverManager.getConnection(url, user, password);
        return connection;
    }

    /**
     * 关闭资源
     * @param conn
     * @param ps
     */
    public static void closeResource(Connection conn, Statement ps){
        //关闭资源
        try {
            if(ps != null)
                ps.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
        try {
            if(conn != null)
                conn.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }

    /**
     * 关闭资源
     * @param conn
     * @param ps
     * @param rs
     */
    public static void closeResource(Connection conn, Statement ps,ResultSet rs){
        //关闭资源
        try {
            if(ps != null)
                ps.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
        try {
            if(conn != null)
                conn.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
        try {
            if(rs != null)
                rs.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
}
4.2.3 修改
@Test
public void test2(){
    String sql = "update customers set name = ? where id = ?";
    update(sql,"小兰",1);
}

public void update(String sql,Object ...args){
    Connection conn = null;
    PreparedStatement ps = null;
    try {
        //1.获取数据库的连接
        conn = JDBCUtils.getConnection();
        //2.预编译sql语句,返回PreparedStatement的实例
        ps = conn.prepareStatement(sql);
        //3.填充占位符
        for (int i = 0; i < args.length; i++) {
            ps.setObject(i+1,args[i]);
        }
        //4.执行
        ps.execute();
    }catch (Exception e){
        e.printStackTrace();
    }finally {
        //5.关闭资源
        JDBCUtils.closeResource(conn,ps);
    }
}

结果:

在这里插入图片描述

4.2.4 查询

ResultSet称为结果集,用来接收执行查询操作返回的结果

ResultSetMetaData,结果集的元数据,可用于获取有关ResultSet对象中列的类型和属性的信息。

返回单个结果:

@Test
public void test(){
    String sql = "select id,name,email,birth from customers where id = ?";
    Customer customer = query(Customer.class,sql, 1);
    System.out.println(customer);
}

public <T> T query(Class<T> clazz,String sql,Object ...args){
    Connection conn = null;
    PreparedStatement ps = null;
    ResultSet resultSet = null;
    try {
        conn = JDBCUtils.getConnection();
        ps = conn.prepareStatement(sql);
        for (int i = 0; i < args.length; i++) {
            ps.setObject(i+1,args[i]);
        }
        resultSet = ps.executeQuery();
        //获取结果集的元数据: ResultSetMetaData
        ResultSetMetaData metaData = resultSet.getMetaData();
        //通过ResultSetMetaData获取结果集中的列数
        int columnCount = metaData.getColumnCount();
        if(resultSet.next()){
            T instance = clazz.newInstance();
            //处理结果集一行数据中的每一个列
            for (int i = 0; i < columnCount; i++) {
                //获取每个列的列名
//              String columnName = metaData.getColumnName(i + 1);
                //获取每个列的别名,没有别名就是列名
                String columnName = metaData.getColumnLabel(i + 1);
                //获取每个列的值
                Object columnValue = resultSet.getObject(i + 1);

                //通过反射给customer对象指定的columnName属性,赋值为columnValue
                Field field = clazz.getDeclaredField(columnName);
                field.setAccessible(true);
                field.set(instance,columnValue);
            }
           return instance;
        }
    } catch (Exception e) {
        e.printStackTrace();
    }finally {
        JDBCUtils.closeResource(conn,ps,resultSet);
    }
    return null;
}
public class Customer {

    private Integer id;
    private String name;
    private String email;
    private Date birth;

    public Customer(Integer id, String name, String email, Date birth) {
        this.id = id;
        this.name = name;
        this.email = email;
        this.birth = birth;
    }

    public Customer() {
    }

    public Integer getId() {
        return id;
    }

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

    public String getName() {
        return name;
    }

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

    public String getEmail() {
        return email;
    }

    public void setEmail(String email) {
        this.email = email;
    }

    public Date getBirth() {
        return birth;
    }

    public void setBirth(Date birth) {
        this.birth = birth;
    }

    @Override
    public String toString() {
        return "Customer{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", email='" + email + '\'' +
                ", birth=" + birth +
                '}';
    }
}
Customer{id=1, name='小兰', email='123@qq.com', birth=1999-01-01}

返回多个结果:

@Test
public void test2(){
    String sql = "select id,name,email,birth from customers where id <= ?";
    List<Customer> customers = queryList(Customer.class, sql, 2);
    for (Customer customer : customers) {
        System.out.println(customer);
    }
}


public <T> List<T> queryList(Class<T> clazz, String sql, Object ...args){
    Connection conn = null;
    PreparedStatement ps = null;
    ResultSet resultSet = null;
    try {
        conn = JDBCUtils.getConnection();
        ps = conn.prepareStatement(sql);
        for (int i = 0; i < args.length; i++) {
            ps.setObject(i+1,args[i]);
        }
        resultSet = ps.executeQuery();
        //获取结果集的元数据: ResultSetMetaData
        ResultSetMetaData metaData = resultSet.getMetaData();
        //通过ResultSetMetaData获取结果集中的列数
        int columnCount = metaData.getColumnCount();
        //创建集合对象
        ArrayList<T> list = new ArrayList<>();
        while (resultSet.next()){
            T instance = clazz.newInstance();
            //处理结果集一行数据中的每一个列
            for (int i = 0; i < columnCount; i++) {
                //获取每个列的列名
//              String columnName = metaData.getColumnName(i + 1);
                //获取每个列的别名
                String columnName = metaData.getColumnLabel(i + 1);
                //获取每个列的值
                Object columnValue = resultSet.getObject(i + 1);

                //通过反射给customer对象指定的columnName属性,赋值为columnValue
                Field field = clazz.getDeclaredField(columnName);
                field.setAccessible(true);
                field.set(instance,columnValue);
            }
            list.add(instance);
        }
        return list;
    } catch (Exception e) {
        e.printStackTrace();
    }finally {
        JDBCUtils.closeResource(conn,ps,resultSet);
    }
    return null;
}
Customer{id=1, name='小兰', email='123@qq.com', birth=1999-01-01}
Customer{id=2, name='新一', email='777@qq.com', birth=2000-01-02}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值