《Java SQL 操作指南:深入理解 Statement 用法与优化》

在 Java 数据库编程中,Statement 是用于执行 SQL 语句的接口,允许程序与数据库进行交互。本文将详细介绍 Statement 的基本概念、常见用法以及 PreparedStatementCallableStatement 等相关接口。

1. Statement 基本介绍

Statement 接口继承了 AutoCloseableWrapper 接口,使其具备自动关闭和封装功能。

  • AutoCloseable:提供 close() 方法,确保 Statement 在不再使用时释放资源。

  • Wrapper:提供 isWrapperFor(Class<?>)unwrap(Class<T>) 方法,允许 Statement 作为其他数据库 API 的封装。

此外,Statement 还包含多个重要属性,如 cursorNamepoolableconnection 等。

Statement 是 JDBC 提供的用于执行 SQL 语句的接口,主要特点如下:

  1. 适用于执行静态 SQL 语句,每次执行都需要编译。

  2. 通过 executeQuery() 执行查询语句,返回 ResultSet

  3. 通过 executeUpdate() 执行更新语句,返回影响的行数。

  4. 存在 SQL 注入风险,建议使用 PreparedStatement 代替。

2. Statement 的基本用法

java 复制代码
package JDBC;

import java.io.FileInputStream;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.Statement;
import java.util.Properties;
import java.util.Scanner;

//JDBC注入机制演示

public class Statement_ {
    public static void main(String[] args) throws Exception {

//        Class.forName("com.mysql.jdbc.Driver");
        Scanner scanner = new Scanner(System.in);
        System.out.println("Enter SQL nameusr statement");
        String username =scanner.nextLine();
        System.out.println("Enter SQL pwd statement");
        String pwd =scanner.nextLine();

        Properties properties = new Properties();
        properties.load(new FileInputStream("src\\JDBC\\mysql.properties"));
        String url = (String) properties.get("url");
        String user =(String) properties.get("user");
        String password =(String) properties.get("password");
        String driver =(String) properties.get("driver");
//        实例化
        Class.forName(driver);

        Connection connection = DriverManager.getConnection(url, user, password);

//        得到statement
        Statement statement = connection.createStatement();

//        组织查询语句
        String sql = "select * from sql_injection where NAME='"+username+"'and  pwd = '"+pwd+"';" ;
        System.out.printf(sql+"\n");

        ResultSet resultSet = statement.executeQuery(sql);
        if (resultSet.next()) {
            System.out.println("查询成功");

        }
        else {
            System.out.println("查询失败");

        }

//        关闭连接,安全操作
        resultSet.close();
        statement.close();
        connection.close();



    }
}

2.1 主要方法

  • execute(String sql): 执行 SQL 语句,返回 boolean,如果执行的是查询语句,返回 true,否则返回 false

  • executeQuery(String sql): 仅用于 SELECT 语句,返回 ResultSet

  • executeUpdate(String sql): 仅用于 INSERTUPDATEDELETE 语句,返回受影响的行数。

  • addBatch(String sql): 添加 SQL 语句到批处理命令中。

  • executeBatch(): 执行批处理命令,返回每条 SQL 语句影响的行数。

  • clearBatch(): 清除已添加的批处理命令。

  • setQueryTimeout(int seconds): 设置查询超时时间。

  • getMoreResults(): 检查是否存在多个 ResultSet

  • close(): 关闭 Statement,释放资源。

2.2 执行更新操作

可以使用 executeUpdate() 进行 INSERTUPDATEDELETE 操作,例如:

java 复制代码
Statement stmt = conn.createStatement();
int rowsAffected = stmt.executeUpdate("UPDATE users SET age = 30 WHERE id = 1");
System.out.println("更新影响的行数: " + rowsAffected);

3. Statement 的问题与优化

3.1 SQL 注入风险

使用 Statement 直接拼接 SQL 语句可能会导致 SQL 注入攻击,例如:

sql 复制代码
USE mydatabase;
SHOW DATABASES;

SHOW TABLES;
CREATE TABLE sql_injection  ( -- 管理表
	NAME VARCHAR(32) NOT NULL UNIQUE,
	pwd VARCHAR(32) NOT NULL DEFAULT ''
)CHARACTER SET utf8;


DROP TABLE sql_injection;
DESC sql_injection;

INSERT INTO sql_injection VALUES( 'wangya' ,'1314520'
);

SELECT * FROM sql_injection;

SELECT *  FROM  sql_injection
	WHERE NAME='wangya' AND pwd='1314520';

-- sql 注入演示
-- 用户输入密码为: 1' or
-- 输入密码为:or '1'= '1
SELECT *  FROM  sql_injection
	WHERE NAME='1' OR' or AND pwd='OR '1'= '1';

SELECT *  FROM  sql_injection
	WHERE NAME='wangya' OR pwd='1314520'  OR '1'='1' ;

以上代码可能被攻击者利用,绕过密码验证。

总结

Statement 是 Java 数据库操作的基本接口,但由于其存在 SQL 注入风险,在实际开发中推荐使用 PreparedStatement。此外,CallableStatement 适用于调用存储过程,提高数据库访问效率。了解并正确使用这些接口,可以提升数据库操作的安全性和性能。

相关推荐
铭毅天下5 分钟前
Easysearch 版本进化全图——从 ES 国产替代到 AI Native 搜索数据库
大数据·数据库·人工智能·elasticsearch·搜索引擎
muddjsv12 分钟前
SQL 最常用技能详解与实战示例
数据库·sql·mysql
砍材农夫27 分钟前
物联网 基于netty构建mqtt协议规范(遗嘱与保留消息)
java·开发语言·物联网·netty
DFT计算杂谈30 分钟前
KPROJ编译教程
java·前端·python·算法·conda
重生之我是Java开发战士41 分钟前
【笔试强训】Week5:空调遥控, kotor和气球,走迷宫,主持人调度II,体操队形,二叉树的最大路径和,排序子序列,消减整数
java·算法·动态规划
郑重其事,鹏程万里1 小时前
表达式计算器(mvel2)
java
其实防守也摸鱼1 小时前
软件安全与漏洞--软件安全编码
java·前端·网络·安全·网络安全·web·工具
888CC++1 小时前
栈上分配 VS 堆分配 核心区别
java·开发语言·jvm
艾利克斯冰1 小时前
Java面试题汇总
java
muddjsv2 小时前
大中小型企业数据配置年度成本估算分析
数据库·企业运营