JDBCMYSQL学习笔记(一)JDBC基本使用_MySQL
1、JDBC简介SUN公司为了简化、统一对数据库的操作,定义了一套Java操作数据库的规范,称之为JDBC。 JDBC全称为:Java Data Base Connectivity(java数据库连接),它主要由接口组成。 组成JDBC的2个包:java.sql javax.sql 开发JDBC应用需要以上2个包的支持外,还需要导入相应JDBC的数据库实现(即数据库驱动)。 2、使用JDBC的步骤——第一个JDBC程序需求:编程从user表中读取数据,并打印在命令行窗口中。 a、方式一:DriverManager. registerDriver(Driver driver) b、方式二:Class.forName(“com.mysql.jdbc.Driver”);(3) 建立连接(Connection) Connection conn = DriverManager.getConnection(url,user,pass); (4) 创建用于向数据库发送SQL的Statement对象,并发送sql Statement st = conn.createStatement(); ResultSet rs = st.excuteQuery(sql); (5) 从代表结果集的ResultSet中取出数据,打印到命令行窗口 (6) 断开与数据库的连接,并释放相关资源 dome: import java.sql.*;import com.sun.org.apache.regexp.internal.recompile;public class dome { /** * @param args */ public static void main(String[] args) throws SQLException,ClassNotFoundException{ // TODO Auto-generated method stub String url = "jdbc:mysql://localhost:3306/skyfin"; String username = "root"; String password = "skyfin"; //1.加载驱动 //DriverManager.registerDriver(new com.mysql.jdbc.Driver()); Class.forName("com.mysql.jdbc.Driver"); //2获取连接Connection connection = DriverManager.getConnection(url,username,password); //3.获取向数据库发sql语句的statament对象Statement stat = connection.createStatement();//4.向数据库发送sql,获取数据库返回的结果集ResultSet rsResultSet = stat.executeQuery("select * from user");//5.从结果集中获取数据while (rsResultSet.next()) {System.out.println("id = "+ rsResultSet.getObject("id"));System.out.println("name = "+ rsResultSet.getObject("name"));System.out.println("password = "+ rsResultSet.getObject("password"));} //6.释放资源(释放链接)rsResultSet.close();stat.close();connection.close();}} 3、DriverManager ——加载数据库驱动Jdbc程序中的DriverManager用于加载驱动,并创建与数据库的链接,这个API的常用方法: DriverManager.registerDriver(new Driver()); DriverManager.getConnection(url, user, password);注意:在实际开发中并不推荐采用registerDriver方法注册驱动。原因有二: 一、查看Driver的源代码可以看到,如果采用此种方式,会导致驱动程序注册两次,也就是在内存中会有两个Driver对象。 二、程序依赖mysql的api,脱离mysql的jar包,程序将无法编译,将来程序切换底层数据库将会非常麻烦。 推荐方式: Class.forName(“com.mysql.jdbc.Driver”);采用此种方式不会导致驱动对象在内存中重复出现,并且采用此种方式,程序仅仅只需要一个字符串,不需要依赖具体的驱动,使程序的灵活性更高。 同样,在开发中也不建议采用具体的驱动类型指向getConnection方法返回的connection对象。 4、数据库URL ——标识数据库的位置URL用于标识数据库的位置,程序员通过URL地址告诉JDBC程序连接哪个数据库, MySql 数据库的URL写法为: jdbc:mysql:[]//localhost:3306/test ?参数名:参数值 jdbc:mysql://localhost:3306/test?user=root&password=&useUnicode=true&characterEncoding=gbk&autoReconnect=true&failOverReadOnly 5、Connection ——代表数据库的链接Jdbc程序中的Connection,它用于代表数据库的链接。Connection 是数据库编程中最重要的一个对象,客户端与数据库所有交互都是通过Connection 对象完成的,这个对象的常用方法:(1) createStatement():创建向数据库发送sql的statement对象。 (2) prepareStatement(sql) :创建向数据库发送预编译sql的PrepareSatement对象。 (3) prepareCall(sql):创建执行存储过程的callableStatement对象。 (4) setAutoCommit(boolean autoCommit):设置事务是否自动提交。 (5) commit() :在链接上提交事务。 (6) rollback() :在此链接上回滚事务。 6、Statement ——向数据库发送SQL语句Jdbc程序中的Statement对象用于向数据库发送SQL语句, Statement对象常用方法:(1) executeQuery(String sql) :用于向数据库发送查询语句。 (2) executeUpdate(String sql):用于向数据库发送insert、update或delete语句 (3) execute(String sql):用于向数据库发送任意sql语句 (4) addBatch(String sql) :把多条sql语句放到一个批处理中。 (5) executeBatch():向数据库发送一批sql语句执行。 (6) clearBatch() :清空此 Statement 对象的当前 SQL 命令列表。 7、ResultSet ——代表Sql语句的执行结果Jdbc程序中的ResultSet用于代表Sql语句的执行结果。Resultset封装执行结果时,采用的类似于表格的方式。ResultSet 对象维护了一个指向表格数据行的游标,初始的时候,游标在第一行之前,调用ResultSet.next() 方法,可以使游标指向具体的数据行,进行调用方法获取该行的数据。(1) ResultSet提供了对结果集进行滚动的方法:a、next():移动到下一行b、Previous():移动到前一行 c、absolute(int row):移动到指定行 d、beforeFirst():移动resultSet的最前面。 e、 afterLast() :移动到resultSet的最后面。 (2) ResultSet既然用于封装执行结果的,所以该对象提供了用于获取数据的get方法:获取任意类型的数据getObject(int index) getObject(string columnName) 获取指定类型的数据,例如: getString(int index) getString(String columnName) 其他获取指定类型数据的方法见下表: 常用数据类型转换表:
8、释放资源 ——释放与数据库进行交互的对象Jdbc程序运行完后,切记要释放程序在运行过程中,创建的那些与数据库进行交互的对象,这些对象通常是ResultSet, Statement和Connection对象。特别是Connection对象,它是非常稀有的资源,用完后必须马上释放,如果Connection不能及时、正确的关闭,极易导致系统宕机。Connection的使用原则是尽量晚创建,尽量早的释放。 为确保资源释放代码能运行,资源释放代码也一定要放在finally语句中。 9、用JDBC对数据库进行CRUDjdbc中的statement对象用于向数据库发送SQL语句,想完成对数据库的增删改查,只需要通过这个对象向数据库发送增删改查语句即可。Statement对象的executeUpdate方法,用于向数据库发送增、删、改的sql语句,executeUpdate执行完后,将会返回一个int整数(即增删改语句导致了数据库几行数据发生了变化)。 Statement.executeQuery方法用于向数据库发送查询语句,executeQuery方法返回代表查询结果的ResultSet对象。
public static void main(String[] args) throws SQLException,ClassNotFoundException{ // TODO Auto-generated method stub String url = "jdbc:mysql://localhost:3306/skyfin"; String username = "root"; String password = "skyfin"; //DriverManager.registerDriver(new com.mysql.jdbc.Driver()); Class.forName("com.mysql.jdbc.Driver"); Connection connection = DriverManager.getConnection(url,username,password); Statement stat = connection.createStatement(); /* * 执行查找操作 */ ResultSet rsResultSet = stat.executeQuery("select * from user"); while (rsResultSet.next()) { System.out.println("id = "+ rsResultSet.getObject("id")); System.out.println("name = "+ rsResultSet.getObject("name")); System.out.println("password = "+ rsResultSet.getObject("password")); } /* * 执行插入操作 */ String sql = "insert into user(id,name,password) value(6,"+"'staff'"+","+"'staff'"+")"; System.out.println(sql); int statentnum = stat.executeUpdate(sql); if (statentnum>0) { System.out.println("insert ok"); } /* * 执行更新操作 */ sql = "update user set name = 'skstaff' where name = 'staff'"; System.out.println(sql); statentnum = stat.executeUpdate(sql); if (statentnum>0) { System.out.println("update ok"); } /* * 执行删除操作 */ sql = "delete from user where name = 'skstaff'"; System.out.println(sql); statentnum = stat.executeUpdate(sql); if (statentnum>0) { System.out.println("delete ok"); } /* * 资源的释放 */ rsResultSet.close(); stat.close(); connection.close(); } 10、用JDBC的PreparedStatement启动事务使用批处理executeBatch()JDBC使用MySQL处理大数据的时候,自然而然的想到要使用批处理,普通的执行过程是:每处理一条数据,就访问一次数据库; 而批处理是:累积到一定数量,再一次性提交到数据库,减少了与数据库的交互次数,所以效率会大大提高 至于事务:事务指逻辑上的一组操作,组成这组操作的各个单元,要不全部成功,要不全部不成功,默认是关闭事务的。 1. PreparedStatement使用批处理 executeBatch()1.1. 不使用executeBatch(),而使用executeUpdate(),速度很慢
public static void main(String[] args) throws SQLException,ClassNotFoundException{ // TODO Auto-generated method stub String url = "jdbc:mysql://localhost:3306/skyfin"; String username = "root"; String password = "skyfin"; //DriverManager.registerDriver(new com.mysql.jdbc.Driver()); Class.forName("com.mysql.jdbc.Driver"); Connection connection = DriverManager.getConnection(url,username,password); String sql = "insert into user1(id,name) value(?,?)"; PreparedStatement preparedStatement = connection.prepareStatement(sql); for(int i = 0;i<10000;i++){ preparedStatement.setInt(1, i); preparedStatement.setString(2, "skyfin"+i); preparedStatement.executeUpdate(); } }
1.2. 而使用executeBatch()public static void main(String[] args) throws SQLException,ClassNotFoundException{ // TODO Auto-generated method stub String url = "jdbc:mysql://localhost:3306/skyfin"; String username = "root"; String password = "skyfin"; //DriverManager.registerDriver(new com.mysql.jdbc.Driver()); Class.forName("com.mysql.jdbc.Driver"); Connection connection = DriverManager.getConnection(url,username,password); String sql = "insert into user1(id,name) value(?,?)"; PreparedStatement preparedStatement = connection.prepareStatement(sql); for(int i = 0;i<10000;i++){ preparedStatement.setInt(1, i); preparedStatement.setString(2, "skyfin"+i); //preparedStatement.executeUpdate(); /* * 使用executeBatch() */ preparedStatement.addBatch(); } //执行批处理 preparedStatement.executeBatch(); } 注意:1. 如果使用了 addBatch() -> executeBatch() 还是很慢,那就得使用到这个参数了rewriteBatchedStatements=true (启动批处理操作) 在数据库连接URL后面加上这个参数: String dbUrl = "jdbc:mysql://localhost:3306/User? rewriteBatchedStatements=true"; 2. 在代码中,pstmt的位置不能乱放,必须放在循环体外 2. 启用事务处理
public static void main(String[] args) throws SQLException,ClassNotFoundException{ // TODO Auto-generated method stub String url = "jdbc:mysql://localhost:3306/skyfin"; String username = "root"; String password = "skyfin"; //DriverManager.registerDriver(new com.mysql.jdbc.Driver()); Class.forName("com.mysql.jdbc.Driver"); Connection connection = DriverManager.getConnection(url,username,password); //关闭自动提交 connection.setAutoCommit(false); String sql = "update user1 set name = ?where id = ?"; PreparedStatement preparedStatement = connection.prepareStatement(sql); for(int i = 0;i<10000;i++){ preparedStatement.setString(1, "loco"+i); preparedStatement.setInt(2, i); //preparedStatement.executeUpdate(); /* * 使用executeBatch() */ preparedStatement.addBatch(); } //执行批处理 preparedStatement.executeBatch(); preparedStatement.close(); //执行完后手动提交事务 connection.commit(); //打开自动提交 connection.setAutoCommit(true); connection.close(); }
3. 事务和批处理混合使用public static void main(String[] args) throws SQLException,ClassNotFoundException{ // TODO Auto-generated method stub String url = "jdbc:mysql://localhost:3306/skyfin"; String username = "root"; String password = "skyfin"; //DriverManager.registerDriver(new com.mysql.jdbc.Driver()); Class.forName("com.mysql.jdbc.Driver"); Connection connection = DriverManager.getConnection(url,username,password); //关闭自动提交 connection.setAutoCommit(false); String sql = "update user1 set name = ?where id = ?"; PreparedStatement preparedStatement = connection.prepareStatement(sql); for(int i = 0;i<10000;i++){ preparedStatement.setString(1, "skyfin"+i); preparedStatement.setInt(2, i); //preparedStatement.executeUpdate(); /* * 使用executeBatch() */ if (i>0&&i%500 == 0) { preparedStatement.executeBatch(); //如果不想出错后,完全没保留数据,则可以没执行一次提交一次,但得保证数据不会重复 connection.commit(); } preparedStatement.addBatch(); } //执行批处理 preparedStatement.executeBatch(); preparedStatement.close(); //执行完后手动提交事务 connection.commit(); //打开自动提交 connection.setAutoCommit(true); connection.close(); } 11、SQL 注入的防范SQL 注入是用户利用某些系统没有对输入数据进行充分的检查,从而进行恶意破坏的行为。1、statement存在sql注入攻击问题,例如登陆用户名采用' or 1=1 or username=‘ 2、对于防范 SQL 注入,可以采用PreparedStatement取代Statement。 备注:本例只是最基本的防止SQL注入方式,其他情况还请查阅资料。 12、PreparedStatementPreperedStatement是Statement的孩子,它的实例对象可以通过调用Connection.preparedStatement()方法获得,相对于Statement对象而言的优势:(1) 防止SQL注入:PreperedStatement可以避免SQL注入的问题。 (2) 预编译SQL语句:Statement会使数据库频繁编译SQL,可能造成数据库缓冲区溢出。PreparedStatement 可对SQL进行预编译,从而提高数据库的执行效率。 (3) 使用占位符简化语句:并且PreperedStatement对于sql中的参数,允许使用占位符的形式进行替换,简化sql语句的编写。 (例如多次循环插入数据)
public List |
- 上一篇:mysql运维之
- 下一篇:eclipse连接mysql_MySQL