news 2026/4/3 6:45:17

JDBC和SQL注入学习笔记

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
JDBC和SQL注入学习笔记

JDBC

JDBC全称Java Database Connectivity,即Java数据库连接器,是Java提供的专门执行SQL的API,包含在java.sql和javax.sql包里。

Java程序要连接数据库执行SQL,还需要安装数据库管理软件驱动,比如要连接MySQL,需要导入mysql-connector的jar包。

可以把JDBC的作用,可以与Navivcat等数据库可视化操作工具类比,比如:连接数据库、执行SQL、管理事务等。

JDBC的使用步骤比较固定:

  1. 加载驱动程序:使用Class.forName()加载数据库驱动(JDBC 4.0以后可以自动加载,无需显性编写这个步骤的代码)。
  2. 建立连接:使用DriverManager.getConnection()获取数据库连接,获得Connection对象,代表与数据库的连接对象。
  3. 创建语句:使用Connection对象创建StatementPreparedStatementCallableStatement。这些Statement对象代表一个SQL。
  4. 执行SQL:执行SQL并获取ResultSet。ResultSet对象代表查询结果集。
  5. 处理结果:可以遍历ResultSet获取数据。
  6. 关闭资源:按顺序关闭ResultSetStatementConnection

示例代码:

importjava.sql.Connection;importjava.sql.DriverManager;importjava.sql.ResultSet;importjava.sql.Statement;// 使用JDBC API进行数据库连接并执行SQL语句 public class JdbcDemo1 {publicstatic void main(String[]args){// 定义数据库连接信息:// URL固定格式jdbc:mysql://主机名(ip或域名):端口号/数据库名称?useSSL=true&characterEncoding=utf8&useUnicode=true(参数K:参数V...)String url="jdbc:mysql://localhost:3306/practice_join?useSSL=true&characterEncoding=utf8&useUnicode=true";// 用户名和密码Stringuser="user";String password="password";// 执行SQL语句(注意:需要在Navicat等可视化工具中运行过,确保SQL语法正确)Stringsql="SELECT s.`student_id` AS student_id, s.`student_name` AS student_name, c.`class_name` AS class_name\n"+"FROM `student` AS s\n"+"LEFT JOIN `class` AS c\n"+"ON s.`class_id`=c.`class_id`";// 使用try-catch-resources语句处理异常,便于释放资源try(Connection conn=DriverManager.getConnection(url,user,password);Statement stmt=conn.createStatement();// 创建SQL语句对象ResultSet rs=stmt.executeQuery(sql);// 执行SQL,返回结果集){ System.out.println("数据库连接成功");// 处理结果集,打印出每一列的结果到控制台while(rs.next()){intstudent_id=rs.getInt("student_id");// 该类为整数类型,可以使用getInt()方法获取String student_name=rs.getString("student_name");// 该类为字符串类型,可以使用getString()方法获取String class_name=rs.getString("class_name");System.out.println(student_id+" "+student_name+" "+class_name);} } catch(Exception e){ System.out.println("数据库连接失败");e.printStackTrace();} } }

JDBC核心接口和类

  1. DriverManager类。用来管理JDBC驱动程序的基本服务。最常用的是static Connection getConnection(String url, String user, String password)方法,尝试与给定数据库URL建立连接。注意,不同的数据库的URL变量固定格式是不同的,MySQL的格式为:
// MySQL url固定格式jdbc:mysql://主机名(ip或域名):端口号/数据库名称?useSSL=true&characterEncoding=utf8&useUnicode=true(参数K:参数V...)Stringurl="jdbc:mysql://localhost:3306/practice_join?useSSL=true&characterEncoding=utf8&useUnicode=true";
  1. Connection接口。表示与特定数据库的连接(会话)。常用方法有:
  • Statement createStatement():创建一个Statement对象,用于将SQL语句发送到数据库。
  • PreparedStatement prepareStatement(String sql):创建一个PreparedStatement对象,用于将参数化的SQL语句发送到数据库。
  • CallableStatement prepareCall(String sql):创建一个CallableStatement对象,用于调用数据库存储过程。
  • void setAutoCommit(boolean autoCommit):设置此连接是否自动提交事务。
  • void commit():提交事务,使上一次提交/回滚之后的所有更改成为持久更改。
  • void rollback():回滚事务,撤销当前事务中所做的所有更改。
  • void close():立即释放此Connection对象的数据库和JDBC资源。常用try-catch-resources语句处理异常,以便异常出现后程序自动释放资源。
  1. Statement接口。表示执行SQL语句的对象。常用方法:
  • ResultSet executeQuery(String sql):执行给定的SQL语句,返回一个ResultSet对象(通常用于SELECT查询语句),是最常用的方法。
  • int executeUpdate(String sql):执行给定的SQL语句,可能是INSERT、UPDATE或DELETE语句,或者不返回任何内容的SQL语句(如DDL语句)。
  • boolean execute(String sql):执行给定的SQL语句,可能返回多个结果(用于执行返回多个结果集或更新计数的语句)。
  • void close():释放此Statement对象的数据库和JDBC资源。
  1. PreparedStatement接口。PreparedStatement与Statement相似,都是表示执行SQL语句的对象。但PreparedStatement表示预编译的SQL语句,可以防止SQL注入问题。常见方法:
  • void setInt(int parameterIndex, int x):将指定参数设置为给定的int值。
  • void setString(int parameterIndex, String x):将指定参数设置为给定的String值。
  • void setDate(int parameterIndex, Date x):将指定参数设置为给定的java.sql.Date值。
  • void setObject(int parameterIndex, Object x):使用给定对象设置指定参数的值。
  • ResultSet executeQuery():执行此PreparedStatement对象中的SQL查询,并返回ResultSet对象。
  • int executeUpdate():执行此PreparedStatement对象中的SQL更新(INSERT、UPDATE、DELETE等)。
  1. ResultSet接口。表示数据库结果集的数据表,通常是执行SELECT查询语句生成。常用方法:
  • boolean next():将光标从当前位置向前移动一行,如果新行有效则返回true,否则返回false。最常用的方法,用于循环解析处理数据。
  • getXXX(int columnIndex)getXXX(String columnLabel):获取当前行中某列的值,其中XXX是数据类型(如Int、String、Date等),不同的列字段类型,使用不同的方法解析。
  • void close():释放此ResultSet对象的数据库和JDBC资源。

SQL注入问题

SQL执行时,一般是用用户输入数据+编写的SQL片段拼接成完整的SQL语句,比如要查询一个用户表中的信息,Java代码会编写成:

...Connectionconn=DriverManager.getConnection(url,user,password);// 创建链接对象Statementstmt=conn.createStatement();// 创建SQL语句对象// 编写SQL语句,是从用户名和密码输入框输入的inputUser和inputPassword,拼接成完整的SQL语句// 比如:SELECT * FROM `user` WHERE `user`='张三' AND `password`='123456'Stringsql="SELECT * FROM `user` WHERE `user`='"+inputUser+"' AND `password`='"+inputPassword+"'";ResultSetrs=stmt.executeQuery(sql);...

正常查询结果是:

iduserpassword
1张三123456

假设,一个黑客可以在用户名和密码框输入:

inputUserinputPassword
’ OR ‘1’='1’ OR ‘1’='1

最终SQL会被拼接为:

SELECT*FROM`user`WHERE`user`=''OR'1'='1'AND`password`=''OR'1'='1'

这个SQL在条件中,OR '1'='1'总为真,所以会查出表中所有的数据,最终执行结果如下:

iduserpassword
1张三123456
2李四456789
3王五789123

这就是SQL注入!

SQL注入是一种代码注入技术,利用应用程序对用户输入处理不当的漏洞,使得攻击者能够在应用程序的数据库查询中注入并执行恶意的SQL代码。

SQL注入的原理,就是利用字符串拼接的方式,将用户输入直接嵌入到SQL语句中。如果用户输入中包含了恶意的SQL片段,会在字符串拼接后,成为SQL执行命令的一部分,从而导致数据库遭受攻击,包括但不限于:数据泄露、数据篡改、数据删除、绕过身份验证等。

SQL注入问题解决方案

JDBC提供PreparedStatement用来解决SQL注入问题。PreparedStatement与Statement相似,都是表示执行SQL语句的对象。但PreparedStatement表示预编译的SQL语句,会先使用利用问号?做占位符编写的SQL进行编译,之后再替换参数。

上述存在SQL注入问题的代码,用PreparedStatement可以改为:

...Connectionconn=DriverManager.getConnection(url,user,password);// 创建链接对象Stringsql="SELECT * FROM `user` WHERE `user`=? AND `password`=?"// 使用字符?代表需要替换的参数PreparedStatementps=conn.prepareStatement(sql);// 创建PreparedStatement对象ps.ps.setString(1,inputUser);// 将第一个?占位的参数替换为inputUser字符串ps.setString(2,inputPassword);// 将第二个?占位的参数替换为inputPassword字符串ResultSetrs=ps.executeQuery();// 执行SQL语句...

为什么使用PreparedStatement能解决SQL注入问题?

因为PreparedStatement采用预编译的机制,预编译完成后,数据库知道SQL语句的基本结构,比如它已经知道要查询user表,条件是两个等值比较,然后等待参数传入。后续程序通过setXXX 方法(如 setString, setInt)为占位符设置参数值,这些参数值不会被解释为SQL语句的一部分,而是作为纯数据传递。因此,即使参数值中包含SQL关键词或特殊字符(比如单引号),数据库也会将其视为普通字符串,会对其进行转义处理,而不会改变原有SQL语句的结构。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/26 23:11:04

告别数据标注!RexUniNLU在保险行业的零样本应用案例

告别数据标注!RexUniNLU在保险行业的零样本应用案例 1. 引言:保险业务中的NLU痛点,真的需要标注数据吗? 1.1 一个真实的保险客服场景 “您好,我想查询上个月在杭州投保的车险保单,保单号是ZJ202403XXXX&…

作者头像 李华
网站建设 2026/3/27 18:02:26

Chandra OCR效果展示:老扫描件数学题识别准确率80.3分实测分享

Chandra OCR效果展示:老扫描件数学题识别准确率80.3分实测分享 1. 为什么老扫描件的数学题最难OCR? 你有没有试过把一张泛黄、带折痕、分辨率只有150dpi的初中数学试卷扫描件丢进普通OCR工具?结果往往是:公式变成乱码&#xff0…

作者头像 李华
网站建设 2026/3/29 0:48:37

RexUniNLU零样本NLU:中文专利摘要技术术语与权利要求抽取

RexUniNLU零样本NLU:中文专利摘要技术术语与权利要求抽取 在处理中文专利文档时,工程师和法务人员常常面临一个现实难题:如何从密密麻麻的摘要和权利要求书中,快速、准确地揪出关键技术术语(比如“电致变色薄膜”“多…

作者头像 李华
网站建设 2026/4/3 5:50:34

面向HPC的XDMA驱动开发流程:手把手教程

以下是对您提供的博文内容进行 深度润色与工程化重构后的版本 。本次优化严格遵循您的所有要求: ✅ 彻底去除AI痕迹,语言更贴近一线嵌入式/Linux驱动工程师的实战口吻; ✅ 打破模板化结构(如“引言/概述/核心特性…”),以问题驱动、场景切入、层层递进的方式组织逻辑…

作者头像 李华
网站建设 2026/3/11 6:48:29

ChatGLM-6B一文详解:supervisorctl命令使用大全

ChatGLM-6B一文详解:supervisorctl命令使用大全 你是不是也遇到过这样的情况:ChatGLM-6B服务跑着跑着就卡住了,或者突然没响应了,但又不知道怎么快速恢复?又或者想改个参数、换种运行方式,却不敢轻易重启&…

作者头像 李华
网站建设 2026/3/30 23:47:28

通义千问2.5-7B为何快?>100 tokens/s性能优化揭秘

通义千问2.5-7B为何快?>100 tokens/s性能优化揭秘 你有没有试过在自己的笔记本上跑一个真正能用的大模型?不是那种卡顿半天才蹦出一个字的“玩具”,而是输入问题后,文字像水流一样连续涌出来,每秒稳定输出超过100个…

作者头像 李华