时间:2021-07-01 10:21:17 帮助过:38人阅读
上述代码的业务非常简单,连接数据库后依次打印Student表里的ID和Name信息。但请大家关注一下这段代码带给我们的启示。
第一,在短短的业务逻辑里,我们分别在第8到第15行,第18到第44行,用了两块try…catch,而没有图省事用一块try…catch代码包含所有的业务方法,这遵循了“尽量缩小检测范围”的原则。
第二,在第35和36行的catch从句里,没有简单地抛出异常了事,而是输出了一些信息,根据这些信息,调试程序的开发人员能很快从中得到提示,从而很快地找到原因。
此外,在catch从句里,也可以输出一些面向使用者的提示信息,比如让使用者重启程序,总之一句话,需要把面向Java的异常翻译成让程序员或使用者能理解的提示信息。
第三,在第37到第44行的finally从句里,关闭了连接,因为不论发生了什么异常,或者是否发生异常,finally从句一定会被执行到,所以可以把关闭连接的代码放入其中。如果不关闭连接,这个数据库连接对象是无法被回收的(Java的垃圾回收机制也无法回收)。
这里对应的问题有:你有没有用过PreparedStatement对象?PreparedStatement和Statement有什么差别?
请大家说清楚两点,1 提升效率,2 避免SQL注入,从而保证系统的安全。
什么叫SQL注入?例如有下图所示的登录界面。
我们一般用如下SQL来验证身份:
Select userName from users where username = ‘输入的用户名’ and pwd = ‘输入的密码’
一般来说,如果用户名和密码不匹配,是无法通过验证的,但有人可以在User Name里输入1,在User Password部分输入:1’ and pwd = ‘1’ or ‘1’=’1
那么整个SQL语句就会变成:
Select userName from users where username = ‘1’ and pwd = ‘1’ or ‘1’=’1’
这样就能绕过验证。
而处理对象PrepareStatement能有效防止这个现象的发生,因为一个“?”就是一个占位符,无法扩展。下面来看一个预处理和批处理结合的例子,同样用到刚才提到的Student表,这次来批量插入数据。
import java.sql.Connection; import java.sql.DriverManager; import java.sql.PreparedStatement; import java.sql.SQLException; import java.sql.Statement; public class JDBCBatch { public static void main(String[] args) { try { Class.forName("com.mysql.jdbc.Driver"); } catch (ClassNotFoundException e) { System.out.println("Where is your MySQL JDBC Driver?"); e.printStackTrace(); return; } Connection connection = null; PrepareStatement pstmt; try { //这里是连接字符串 connection = DriverManager.getConnection( "jdbc:mysql://localhost:3306/class3", "root", "123456"); if (connection != null) { String query = "insert into student values (?,?)"; pstmt = connection.preparedStatement(query);//开始设置参数 pstmt.setString(1,"1"); pstmt.setString(2,"Peter"); pstmt.addBatch(); //设置第二个参数 pstmt.setString(1,"2"); pstmt.setString(2,"Mike"); pstmt.addBatch(); //执行批处理 pstmt.executeBatch(); } else { System.out.println("Failed to make connection!"); } } catch (SQLException e) { System.out.println("Some of Students were not inserted correctly, please check the student table and insert manually."); e.printStackTrace(); } finally { try { connection.close(); } catch (SQLException e) { e.printStackTrace(); } } } }
这部分的逻辑也比较简单,用insert语句批量插入数据。但请大家注意如下两点:
① 在PrepareStatement里,占位符的编号是从1开始,而不是从0开始。
② 批量操作能提升效率,但一次性操作多少,才能让效率提升到最高?这在不同的数据库里是不同的,一般是每批操作500到1000条语句。但切记,别一次性把所有的insert语句都用addBatch放入,因为如果SQL语句过多,会撑爆缓存,从而出错。
PrepareStatement是个比较重要的JDBC对象。再重复一下,我们在面试的时候,有时会问这个问题:Statement和PrepareStatement有什么差别?答案要点是PrepareStatement能预处理,如果能展开一下,说明能防止SQL注入就更好了。
总结一下,这里大家可以通过叙述代码里的要点来阐释自己在JDBC方面的能力。
1 阐述try...catch...finally的用法。
2 详细阐述PreparedStatement的用法,包括预处理批处理和SQL注入两点。
数据库面试技巧,通过JDBC展示自己专业性,摘自java web轻量级开发面试教程
标签:面试官 failed 类型 .exe 出错 联系人 prepare 语句 student