文学起点网
当前位置: 首页 文学百科

jdbc详细教程(JDBC技术知识PreparedStatement)

时间:2023-05-23 作者: 小编 阅读量: 1 栏目名: 文学百科

对于java而言,要防范SQL注入,只要用PreparedStatement取代Statement就可以了。插入BLOB类型的数据必须使用PreparedStatement,因为BLOB类型的数据无法使用字符串拼接写的。PreparedStatement可以防止SQL注入PreparedStatement可以处理Blob类型的数据PreparedStatement能最大可能提高性能:DBServer会对预编译语句提供性能优化。

1、PreparedStatement概述

可以通过调用 Connection 对象的 preparedStatement(String sql) 方法获取 PreparedStatement 对象

PreparedStatement 接口是 Statement 的子接口,它表示一条预编译过的 SQL 语句

l PreparedStatement 对象所代表的 SQL 语句中的参数用问号(?)来表示,调用 PreparedStatement 对象的 setXxx() 方法来设置这些参数. setXxx() 方法有两个参数,第一个参数是要设置的 SQL 语句中的参数的索引(从 1 开始),第二个是设置的 SQL 语句中的参数的值

l ResultSet executeQuery()执行查询,并返回该查询生成的 ResultSet 对象。

l int executeUpdate():执行更新,包括增、删、该

2、Statement的不足

(1)SQL拼接

(2)SQL注入

SQL 注入是利用某些系统没有对用户输入的数据进行充分的检查,而在用户输入数据中注入非法的 sql 语句段或命令,从而利用系统的 SQL 引擎完成恶意行为的做法。对于 java 而言,要防范 SQL 注入,只要用 PreparedStatement 取代 Statement 就可以了。

(3)处理BLOB类型的数据

BLOB (binary large object),二进制大对象,BLOB常常是数据库中用来存储二进制文件的字段类型。

插入BLOB类型的数据必须使用PreparedStatement,因为BLOB类型的数据无法使用字符串拼接写的。

MySQL的四种BLOB类型(除了在存储的最大信息量上不同外,他们是等同的)

如果还是报错:xxx too large,那么在mysql的安装目录下,找my.ini文件加上如下的配置参数:max_allowed_packet=16M注意:修改了my.ini文件,一定要重新启动服务

实际使用中根据需要存入的数据大小定义不同的BLOB类型。需要注意的是:如果存储的文件过大,数据库的性能会下降。

CREATE TABLE `user` (`id` int(11) NOT NULL AUTO_INCREMENT,`username` varchar(20) COLLATE utf8_unicode_ci DEFAULT NULL,`head_picture` mediumblob,PRIMARY KEY (`id`)) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;

package com.JDBC;import java.io.FileInputStream;import java.sql.Connection;import java.sql.DriverManager;import java.sql.PreparedStatement;import java.util.Properties;public class TestBlob {public static void main(String[] args) throws Exception{//加载jdbc.properties资源配置文件Properties pro = new Properties();pro.load(ClassLoader.getSystemResourceAsStream("jdbc.properties"));//1、加载与注册驱动Class.forName(pro.getProperty("driver"));//2、获取数据库连接Connection conn = DriverManager.getConnection(pro.getProperty("url"), pro);//3、访问数据库//(1)准备带参数(?)的SQL//PreparedStatement 接口是 Statement 的子接口,它表示一条预编译过的 SQL 语句//PreparedStatement 对象所代表的 SQL 语句中的参数用问号(?)来表示String sql ="insert into user(username,head_picture) value(?,?)";//(2)通过调用 Connection 对象的 preparedStatement(String sql) 方法获取 PreparedStatement 对象PreparedStatement pst = conn.prepareStatement(sql);//(3)调用 PreparedStatement 对象的 setXxx(int parameterIndex,XX value) 方法来设置这些参数pst.setString(1, "lily");pst.setBlob(2, new FileInputStream("head/girl.jpg"));//(4)调用PreparedStatement的executeUpdate()执行sql语句进行插入//注意此处不能在传sql,否则?就白设置了int len = pst.executeUpdate();//(5)处理结果if (len > 0) {System.out.println("添加成功");} else {System.out.println("添加失败");}//4、释放资源pst.close();conn.close();}}

3、PreparedStatement vs Statement

代码的可读性和可维护性. Statement的sql拼接是个难题。

PreparedStatement 可以防止 SQL 注入

PreparedStatement 可以处理Blob类型的数据

PreparedStatement 能最大可能提高性能:(Oracle和PostgreSQL8是这样,但是对于MySQL不一定比Statement高)

DBServer会对预编译语句提供性能优化。因为预编译语句有可能被重复调用,所以语句在被DBServer的编译器编译后的执行代码被缓存下来,那么下次调用时只要是相同的预编译语句就不需要编译,只要将参数直接传入编译过的语句执行代码中就会得到执行。

在statement语句中,即使是相同操作但因为数据内容不一样,所以整个语句本身不能匹配,没有缓存语句的意义.事实是没有数据库会对普通语句编译后的执行代码缓存.这样每执行一次都要对传入的语句编译一次.

(语法检查,语义检查,翻译成二进制命令,缓存)

4、示例代码

(1)使用Statement

package com.atguigu.statement;import java.sql.Connection;import java.sql.ResultSet;import java.sql.Statement;import java.util.Scanner;import org.junit.Test;import com.atguigu.utils.JDBCUtils;/** Statement:* 1、SQL拼接* 2、SQL注入* 3、处理不了Blob类型的数据*/public class TestStatementProblem {@Testpublic void add() throws Exception{Scanner input = new Scanner(System.in);System.out.println("请输入姓名:");String name = input.nextLine();System.out.println("请输入领导编号:");int mid = input.nextInt();System.out.println("请输入部门编号:");int did = input.nextInt();//1、获取连接Connection conn = JDBCUtils.getConnection();//2、创建Statement对象Statement st = conn.createStatement();//3、编写sqlString sql = "INSERT INTO emp (ename,`mid`,did) VALUES('"name "',"mid","did")";//4、执行sqlint update = st.executeUpdate(sql);System.out.println(update>0?"添加成功":"添加失败");//5、释放资源JDBCUtils.closeQuietly(st, conn);}@Testpublic void select()throws Exception{Scanner input = new Scanner(System.in);System.out.println("请输入姓名:");String name = input.nextLine();//1、获取连接Connection conn = JDBCUtils.getConnection();//2、写sql//孙红雷 ' or '1' = '1String sql = "SELECT eid,ename,tel,gender,salary FROM t_employee WHERE ename = '"name"'";System.out.println(sql);// SELECT eid,ename,tel,gender,salary FROM t_employee WHERE ename = '孙红雷 ' or '1' = '1'//3、用Statement执行Statement st = conn.createStatement();//4、执行查询sqlResultSet rs = st.executeQuery(sql);while(rs.next()){int id = rs.getInt(1);String ename = rs.getString(2);String tel = rs.getString(3);String gender =rs.getString(4);double salary = rs.getDouble(5);System.out.println(id "\t"ename"\t"tel"\t"gender"\t"salary);}//5、释放资源JDBCUtils.closeQuietly(rs, st, conn);}@Testpublic void testAddBlob(){String sql = "INSERT INTO `user` (username,`password`,photo)VALUES('chai','123',没法在String中处理Blob类型的数据);";}}

(2)使用PreparedStatement

package com.atguigu.preparedstatement;import java.io.FileInputStream;import java.sql.Connection;import java.sql.PreparedStatement;import java.sql.ResultSet;import java.util.Scanner;import org.junit.Test;import com.atguigu.utils.JDBCUtils;/** PreparedStatement:是Statement子接口* 1、SQL不需要拼接* 2、SQL不会出现注入* 3、可以处理Blob类型的数据* tinyblob:255字节以内* blob:65K以内* mediumblob:16M以内* longblob:4G以内** 如果还是报错:xxx too large,那么在mysql的安装目录下,找my.ini文件加上如下的配置参数:* max_allowed_packet=16M* 注意:修改了my.ini文件,一定要重新启动服务**/public class TestPreparedStatement {@Testpublic void add() throws Exception {Scanner input = new Scanner(System.in);System.out.println("请输入姓名:");String name = input.nextLine();System.out.println("请输入性别:");String gender = input.nextLine();System.out.println("请输入领导编号:");int mid = input.nextInt();System.out.println("请输入部门编号:");int did = input.nextInt();String sql = "INSERT INTO emp VALUES(NULL,?,?,?,?)";// 参数,占位符,通配符,表示这个地方需要设置值// 2、获取连接Connection conn = JDBCUtils.getConnection();// 3、准备一个PreparedStatement:预编译sqlPreparedStatement pst = conn.prepareStatement(sql);// 对带?的sql进行预编译// 4、把?用具体的值进行代替pst.setString(1, name);pst.setString(2, gender);pst.setInt(3, mid);pst.setInt(4, did);// 5、执行sqlint len = pst.executeUpdate();// 6、释放资源JDBCUtils.closeQuietly(pst, conn);}@Testpublic void select() throws Exception {// 3、写sqlScanner input = new Scanner(System.in);System.out.println("请输入姓名:");String name = input.nextLine();// 孙红雷 ' or '1' = '1String sql = "SELECT eid,ename,tel,gender,salary FROM t_employee WHERE ename = ?";// 1、注册驱动,注册过了// 2、获取连接Connection conn = JDBCUtils.getConnection();// 3、把带?的sql语句进行预编译PreparedStatement pst = conn.prepareStatement(sql);// 4、把?用具体的变量的赋值pst.setString(1, name);// 5、执行sqlResultSet rs = pst.executeQuery();while (rs.next()) {int id = rs.getInt("eid");String ename = rs.getString("ename");String tel = rs.getString("tel");String gender = rs.getString("gender");double salary = rs.getDouble("salary");System.out.println(id"\t"ename"\t"tel"\t"gender"\t"salary);}// 6、释放资源JDBCUtils.closeQuietly(rs, pst, conn);}@Testpublic void addBlob() throws Exception {Scanner input = new Scanner(System.in);System.out.println("请输入用户名:");String username = input.nextLine();System.out.println("请输入密码:");String password = input.nextLine();System.out.println("请指定照片的路径:");String photoPath = input.nextLine();// INSERT INTO `user` VALUES(NULL,用户名,密码,照片)String sql = "INSERT INTO `user` VALUES(NULL,?,?,?)";// 1、注册驱动,注册过了// 2、获取连接Connection conn = JDBCUtils.getConnection();// 3、准备一个PreparedStatement:预编译sqlPreparedStatement pst = conn.prepareStatement(sql);// 对带?的sql进行预编译// 4、对?进行设置pst.setString(1, username);pst.setString(2, password);pst.setBlob(3, new FileInputStream(photoPath));// 5、执行sqlint len = pst.executeUpdate();System.out.println(len > 0 ? "添加成功" : "添加失败");// 6、释放资源JDBCUtils.closeQuietly(pst, conn);}}


相关阅读:

JDBC工具类

如何使用JDBC API

java编程技术JDBC

如何使用JDBC API操作数据库

数据库连接池之DBCP数据源

    推荐阅读
  • 供电运维队长职责(奋进供电人杨晶晶)

    2013年入职南方电网深圳供电局以来,她从事新闻宣传工作已经9个年头。从小就有着“新闻梦”的杨晶晶,以笔为媒,做深圳电网与公众间的“桥梁”“窗口”。打开杨晶晶的电脑文件夹,她“码”下的每一篇稿件按“年度-季度”分类存档。自2018年来到深圳供电局新闻中心以来,她累计写稿超过了36万字。杨晶晶记得,2020年,新冠肺炎疫情突然袭来,她因居家办公无法到现场采访。

  • 中秋月圆夜难眠(中秋月圆之夜有荞麦)

    众所周知,荞麦是一种健康的食物。赤峰荞麦文化协会成立于2017年8月3日,是以发掘荞麦文化产业为宗旨,不断的创新研发荞麦文化及产品。近期还研发了荞麦壳床垫,一面使用羊毛毡垫,另一面使用荞麦壳填充。荞麦全身都是宝,一方水土养一方荞麦。将会有更多的荞麦品牌相继推出,将赤峰荞麦这张名片推向全国!

  • 早餐什么时候吃最好(吃早餐的最佳时间是几点)

    早餐最好在早上7点后吃。医学专家指出,人在睡眠时,绝大部分器官都得到了充分休息,而消化器官却仍在消化吸收晚餐存留在胃肠道中的食物,到早晨才渐渐进入休息状态。一旦吃早餐太早,势必会干扰胃肠的休息,使消化系统长期处于疲劳应战的状态,扰乱肠胃的蠕动节奏。另外,早餐与午餐以间隔4至5小时左右为好,也就是说早餐7至8点之间为好,如果早餐过早,那么数量应该相应增加或者将午餐相应提前。

  • 死亡笔记中硫克最后在笔记本上写夜神月的名字的真正原因终于揭晓(死亡笔记中硫克最后在笔记本上写夜神月的名字的真正原因终于揭晓)

    死亡笔记中硫克最后在笔记本上写夜神月的名字的真正原因终于揭晓?夜神月被捕,肯定会被判死刑,寿命本来就已经到尽头了,夜神月让硫克用笔记杀死其他人,硫克如果照办,就等于是帮助月延长了生命,硫克就会死。另外,死亡笔记有规则,当死亡笔记在人间的第一个拥有者死亡时,死神必须要在自己的那本笔记上写上此拥有者的名字,相当于一个契约,拿到笔记本成为拥有者契约开始,最后第一个拥有者死亡时,死神写上名字契约结束。

  • 玻璃钢与碳钢哪个更好(林森为大家讲述玻璃钢是玻璃)

    玻璃钢也称作SMC,即纤维强化塑料,一般指用玻璃纤维增强不饱和聚脂、环氧树脂与酚醛树脂基体。以玻璃纤维或其制品作增强材料的增强塑料,称为玻璃纤维增强塑料,或称玻璃钢。无机玻璃钢由于所使用的树脂品种不同,因此有聚酯玻璃钢、环氧玻璃钢、酚醛玻璃钢之称。基体相对于纤维来说,强度、模量都要低很多,但可以经受住大的应变,往往具有粘弹性和弹塑性,是韧性材料。

  • 主动脉硬化是怎么回事(主动脉硬化是怎么回事怎么治疗)

    动脉硬化是随着年龄增长而出现的血管疾病,其规律通常是在青少年时期发生,至中老年时期加重、发病。引起主动脉硬化的病因中最重要的是高血压、高脂血症、抽烟。因此说服患者耐心接受长期的防治措施至关重要。若血脂持续增高,应食用低胆固醇、低动物性脂肪食物,如:各种瘦肉,鸡、鸭、鱼肉,蛋白,豆制品等。参加一定的体力劳动和体育活动,对预防肥胖、锻炼循环系统的功能和调整血脂代谢均有禆益,是预防本病的一项积极措施。

  • 失业金领取条件及标准(失业金领取条件及标准是什么)

    我们一起去了解并探讨一下这个问题吧!失业金领取条件及标准失业保险金领取条件有:具备下列条件的失业人员,可以领取失业保险金。按照规定参加失业保险,所在单位和本人已按照规定履行缴费义务满1年的;非因本人意愿中断就业的;已依法定程序办理失业登记的;有求职要求,愿意接受职业培训、职业介绍的。

  • 如何做爽口萝卜(爽口萝卜制作步骤)

    如何做爽口萝卜材料:白萝卜600g,白醋40,盐10g,糖50g,小米椒5个。米醋和糖的比例可以按自己对酸甜的喜好来配比调整。撒上10g盐搅拌均匀,腌制大约半小时,使得白萝卜软化脱水去辛辣。加入小米椒、糖、白醋。萝卜倒进密封的保鲜盒或玻璃罐。放进冰箱冷藏保存,腌几个小时就可以吃了,隔天后的白萝卜更好吃。每次吃的时候取出一部分,然后放回冰箱,随着腌制时间延长,吃到最后越来越好吃。

  • 酸汤肥牛金针菇的做法(金汤肥牛)

    金汤肥牛酸辣爽口,非常开胃,将肥牛煮在酸爽辛辣的金汤里,用金针菇做配菜非常适合。

  • 出去游玩是清明节的习俗吗(清明节古人玩什么吃什么)

    清明,不仅是季春三月的一个节气,还是传统民俗文化中一个重要节日。清明是扫墓节,因为融合了寒食节。古代寒食节在冬至后一百零五天,恰与汉代以后出现的清明节气贴近。清明扫墓、缅怀先人遂成风俗沿袭至今。清明又是踏青节,因为合并了上巳节。魏晋以后,将上巳节定为三月初三。元朝宫中更是盛行,每逢清明,宫女们穿上“蹴秋千之服”,双双对蹴。秋千是女人和儿童玩的,成年男子则以蹴鞠为戏。佳节有佳肴,清明要吃桃花饭。