JDBC学习2

示例地址:https://github.com/lizhongzhen11/jdbcStudy-1

SQL 注入攻击

SQL 注入是利用某些系统没有对用户输入的数据进行充分的检查,而在用户输入数据中注入非法的 SQL 语句段或命令,从而利用系统的 SQL 引擎完成恶意行为的做法

对于 Java 而言,要防范 SQL 注入,只要用 PreparedStatement 取代 Statement 就可以了

PreparedStatement

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

PreparedStatement 接口是 Statement 的子接口,它表示一条预编译过的 SQL 语句
PreparedStatement 对象所代表的 SQL 语句中的参数用问号(?)来表示,调用 PreparedStatement 对象的 setXXX() 方法来设置这些参数. setXXX() 方法有两个参数,第一个参数是要设置的 SQL 语句中的参数的索引(从 1 开始),第二个是设置的 SQL 语句中的参数的值

PreparedStatement vs Statement

代码的可读性和可维护性.
PreparedStatement 能最大可能提高性能:

  • DBServer会对预编译语句提供性能优化。因为预编译语句有可能被重复调用,所以语句在被DBServer的编译器编译后的执行代码被缓存下来,那么下次调用时只要是相同的预编译语句就不需要编译,只要将参数直接传入编译过的语句执行代码中就会得到执行。
  • 在statement语句中,即使是相同操作但因为数据内容不一样,所以整个语句本身不能匹配,没有缓存语句的意义.事实是没有数据库会对普通语句编译后的执行代码缓存.这样每执行一次都要对传入的语句编译一次.
  • (语法检查,语义检查,翻译成二进制命令,缓存)
    PreparedStatement 可以防止 SQL 注入

Why PreparedStatement?

1.一开始使用的是 Statement,但是需要拼sql语句,非常复杂,容易出错。
2.PreparedStatement:是 Statement 的子接口,可以传入带占位符的SQL语句。并且提供了补充占位符变量的方法。
3.防范SQL注入攻击

例:

1
2
3
4
5
6
7
String sql = "insert into jdbcStudentTest values("+ jdbcStudentTest.getFlowid()
+","+ jdbcStudentTest.getType()
+",'"+ jdbcStudentTest.getIdcard()
+"','" + jdbcStudentTest.getStudentname()
+ "','"+ jdbcStudentTest.getLocation()
+"',"+ jdbcStudentTest.getScore()
+")";

How PreparedStatement?

1.创建 PreparedStatement
2.调用 PreparedStatement 的 setXxx(int index, Object val) 设置占位符的值
3.执行 SQL 语句: executeQuery() 或 executeUpdate().注意: 执行时不再需要传入 SQL 语句

利用反射及JDBC元数据编写通用的查询方法

示例地址:https://github.com/lizhongzhen11/jdbcStudy-1

使用 JDBC 驱动程序处理元数据

Java 通过JDBC获得连接以后,得到一个Connection 对象,可以从这个对象获得有关数据库管理系统的各种信息,包括数据库中的各个表,表中的各个列,数据类型,触发器,存储过程等各方面的信息。根据这些信息,JDBC可以访问一个实现事先并不了解的数据库。
获取这些信息的方法都是在DatabaseMetaData类的对象上实现的,而DataBaseMetaData对象是在Connection对象上获得的。

DatabaseMetaData类

DatabaseMetaData 类中提供了许多方法用于获得数据源的各种信息,通过这些方法可以非常详细的了解数据库的信息:

  • getURL():返回一个String类对象,代表数据库的URL。
  • getUserName():返回连接当前数据库管理系统的用户名。
  • isReadOnly():返回一个boolean值,指示数据库是否只允许读操作。
  • getDatabaseProductName():返回数据库的产品名称。
  • getDatabaseProductVersion():返回数据库的版本号。
  • getDriverName():返回驱动驱动程序的名称。
  • getDriverVersion():返回驱动程序的版本号。

ResultSetMetaData 类

可用于获取关于 ResultSet 对象中列的类型和属性信息的对象:

  • getColumnName(int column):获取指定列的名称
  • getColumnCount():返回当前 ResultSet 对象中的列数。
  • getColumnTypeName(int column):检索指定列的数据库特定的类型名称。
  • getColumnDisplaySize(int column):指示指定列的最大标准宽度,以字符为单位。
  • isNullable(int column):指示指定列中的值是否可以为 null。
  • isAutoIncrement(int column):指示是否自动为指定列进行编号,这样这些列仍然是只读的。

通用的查询方法:可以根据传入的sql还有class对象返回 SQL 对应的记录的对象。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
// 写一个通用的方法,减少代码冗余
public <T> T get(Class<T> clazz, String sql, Object ... args) {
T entity = null;
Connection connection = null;
PreparedStatement preparedStatement = null;
ResultSet resultSet = null;
try{
//1. 得到 ResultSet 对象
connection = JDBCTest.getConnection2();
preparedStatement = connection.prepareStatement(sql);
for(int i = 0; i < args.length; i++) {
preparedStatement.setObject(i+1, args[i]);
}
resultSet = preparedStatement.executeQuery();
//2. 得到ResultSetMetaData对象
ResultSetMetaData data = (ResultSetMetaData) resultSet.getMetaData();
//3. 创建一个Map<String,Object>对象,键:SQL查询的列的别名, 值:列的值
Map<String, Object> values = new HashMap<String, Object>();
//4. 处理结果集.利用 ResultSetMetaData 填充 3 对应的 Map 对象
while(resultSet.next()){
// 打印每一列的列名
for(int i = 0; i < data.getColumnCount(); i++) {
String columnLabel = data.getColumnLabel(i + 1);
Object columnValue = resultSet.getObject(columnLabel);
values.put(columnLabel, columnValue);
}
}
//5. 若 Map 不为空集,利用反射创建 clazz 对应的对象
entity = clazz.newInstance();
//6. 遍历 Map 对象,利用反射为 Class 对象的对应的属性赋值
for(Map.Entry<String, Object> entry:values.entrySet()) {
String fieldName = entry.getKey();
Object fieldValue = entry.getValue();
ReflectionUtils.setFieldValue(entity, fieldName, fieldValue);
}
} catch (Exception e) {
e.printStackTrace();
} finally {
JDBCTest.release(resultSet, preparedStatement, connection);
}
return entity;
}

Newer Post

JDBC学习3--DAO设计模式

DAO: Data Access Object 示例地址:https://github.com/lizhongzhen11/jdbcStudy-1 why? 实现功能的模块化。更有利于代码的维护和升级。DAO 可以被子类继承或直接使用 what? 访问数据信息的类。包含了对数据的 CRUD( …

继续阅读
Older Post

win10系统安装sql server爬坑之旅

经历。。。 一开始电脑上有 Navicat,可以直接连接数据库,本来也够用了,前段时间学习 spring 和 jdbc 就下载了 mysql,心想也差不多够了吧,虽然祥哥早就把 sql server 安装包给我了,之前在公司另一台破旧的win7电脑上也安装成功了,所以也没太在意。直到昨天,组长分配任 …

继续阅读