图片来源:动力节点老杜的JDBC视频讲解
conn.setAutoCommit(false);//关闭自动提交机制
conn.rollback();//出错了回滚
conn.commit();//提交事务
第一种方式
//DriverManager.registerDriver(new com.mysql.cj.jdbc.Driver());
Driver driver = new com.mysql.cj.jdbc.Driver();//多态,父类型引用指向子类型对象
DriverManager.registerDriver(driver);
第二种方式
# properties 文件,文件名为 jdbc.properties
driver=com.mysql.cj.jdbc.Driver
//1.注册驱动器
//以下方式不需要接收返回值,因为我们只想用他类加载动作。
//静态代码块,在类加载是执行(因为数据库实现的Driver类的静态代码块中写了第一种方式的代码)
//Class.forName("com.mysql.cj.jdbc.Driver");
//因为参数是一个字符串,字符串可以写到xxx.properties文件中
ResourceBundle bundle = ResourceBundle.getBundle("jdbc");
String driver = bundle.getString("driver");
Class.forName(diver);
对于 ResourceBundle和Properties 文件的使用参考我都博客 ====> properties 属性配置文件
第二种方式引入Java中的properties文件可以提高程序的扩展性。
第一种方式
/*url是什么?
* 统一资源定位符(网络中某个资源的绝对路径)
* jdbc:mysql://localhost:3306/gdb
* jdbc:mysql://——协议
* 通信协议是通信之前就提前定好的数据传送格式,数据包具体怎么传输据,格式提前定好的。
* localhost IP地址
* 3306 mysql数据库端口号
* gdb 具体的数据库实例名
* 说明localhost和127.0.0.1都是本机IP地址
*/
String url = "jdbc:mysql://127.0.0.1:3306/jdbc"; //"jdbc:mysql://ip:port/数据库名字"
String user = "root";
String password = "123456";
Connection coon = DriverManager.getConnection(url, user, password);
第二种方式
# properties 文件,文件名为 jdbc.properties
driver=com.mysql.cj.jdbc.Driver
url=jdbc:mysql://localhost:3306/gdb
user=root
password=123456
//使用资源绑定器绑定属性配置文件
ResourceBundle bundle = ResourceBundle.getBundle("jdbc");
String url = bundle.getString("url");
String user = bundle.getString("user");
String password = bundle.getString("password");
Connection coon = DriverManager.getConnection(url, user, password);
使用第二种方式的目的也是为了提高程序的扩展性、降低耦合度。
第一种方式
Statement stmt = coon.createStatement();
第二种方式
//上面的方式存在sql注入的问题
/**
* 1.解决sql注入的问题 只要用户的信息不参与sql语句的编译过程,问题就解决了。
* 即使用户提供的信息中含有SQL语句的关键字,但是没有参与编译,不起作用。 必须使用java.sql.PreparedStatement
* PreparedStatement接口继承了java.sql.Statement
* PreparedStatement接口是属于预编译的数据库操作对象。
* PreparedStatement的原理是,预先对SQL语句的框架进行编译,然后再给SQL语句传”值“。
*/
// 3.获取预编译的数据库操作对象
/*SQl语句的框子
* 一个?表示一个站位符,一个?将来接收一个”值“,注意占位符不能用''括起来
*/
String sql = "select * from t_user where loginName = ? and loginPassword = ?";
//程序执行到这里,会发送sql语句框架,然后DBMS进行sql语句的预先编译
PreparedStatement ps = conn.prepareStatement(sql);
//第一个?下标为1,第二个?下标为2
ps.setString(1,userLoginInfo.get("userName"));
ps.setString(2,userLoginInfo.get("password"));
Statement和PreparedStatement对比:
什么情况必须使用Statement?
第一种方式:对应了获取数据库操作对象的方式一
//4.执行sql语句
//JDBC的sql语句不需要提供分号结尾
//String sql = "delete from dept where deptno=50";
String sql = "insert into dept values (50,'人事部','成都')";
//专门执行DML语句的(insert delete update)
//返回值是”影响数据库中的记录条数“
int count = stmt.executeUpdate(sql);
//专门执行DQL语句的方法(select)
String sql = "select empno as a,ename,sal from emp";
ResultSet rs = stmt.executeQuery(sql);
第二种方式:对应了获取数据库操作对象的方式二
ResultSet rs = ps.executeQuery();//注意这里不能再传sql语句了
// 5.处理查询结果集
while(rs.next()) {
//int empno = rs.getInt("a");//列名称不是查询表中的名称,是查询结果集的名称
//String ename = rs.getString("ename");
//double sal = rs.getDouble("sal");
int empno = rs.getInt(1);
String ename = rs.getString(2);
double sal = rs.getDouble(3);
System.out.println(empno + " " + ename + " " + sal);
}
finally {
// 6.释放资源
if (rs != null)
try {
rs.close();
} catch (Exception e) {
e.printStackTrace();
}
if (stmt != null)
try {
stmt.close();
} catch (Exception e) {
e.printStackTrace();
}
if (conn != null)
try {
conn.close();
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* JDBC工具类,简化JDBC编程
*/
public class DBUtil {
public static ResourceBundle bundle = ResourceBundle.getBundle("jdbc/resources/jdbc");
public static String driver = bundle.getString("driver");
public static String url = bundle.getString("url");
public static String user = bundle.getString("user");
public static String password = bundle.getString("password");
//静态代码快在类加载的时候执行,只执行一次
static {
try {
Class.forName(driver);//只让类加载
} catch (ClassNotFoundException e) {
System.out.println("类加载失败");
}
}
/**
* 工具类中的构造方法都是私有的
* 因为工具类当中的方法都是静态的,不需要new对象,直接采用类名.调用
*/
private DBUtil(){}
/**
* 获取数据库连接对象
* @return 连接对象
*/
public static Connection getConnection() throws SQLException{
return DriverManager.getConnection(url,user,password);
}
/**
* 释放资源
* @param conn 连接对象
* @param ps 数据库操作对象
* @param rs 结果集
*/
public static void close(Connection conn, Statement ps, ResultSet rs){//注意这里Statement这里是想想抽象编程,多态
if (rs != null){
try {
rs.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
if (ps != null){
try {
ps.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
if (conn != null){
try {
conn.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
}
/**
* 释放资源
* @param conn 连接对象
* @param ps 数据库操作对象
*/
public static void close(Connection conn, PreparedStatement ps){
if (ps != null){
try {
ps.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
if (conn != null){
try {
conn.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
}
}
package com.gdb.jdbc;
import javax.sql.DataSource;
import java.io.PrintWriter;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.SQLException;
import java.sql.SQLFeatureNotSupportedException;
import java.util.logging.Logger;
public class MyDataSource implements DataSource {
// 添加4个属性
private String driver;
private String url;
private String username;
private String password;
// 提供4个setter方法
public void setDriver(String driver) {
this.driver = driver;
}
public void setUrl(String url) {
this.url = url;
}
public void setUsername(String username) {
this.username = username;
}
public void setPassword(String password) {
this.password = password;
}
// 重点写怎么获取Connection对象就行。其他方法不用管。
@Override
public Connection getConnection() throws SQLException {
try {
Class.forName(driver);
Connection conn = DriverManager.getConnection(url, username, password);
return conn;
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
@Override
public Connection getConnection(String username, String password) throws SQLException {
return null;
}
@Override
public PrintWriter getLogWriter() throws SQLException {
return null;
}
@Override
public void setLogWriter(PrintWriter out) throws SQLException {
}
@Override
public void setLoginTimeout(int seconds) throws SQLException {
}
@Override
public int getLoginTimeout() throws SQLException {
return 0;
}
@Override
public Logger getParentLogger() throws SQLFeatureNotSupportedException {
return null;
}
@Override
public <T> T unwrap(Class<T> iface) throws SQLException {
return null;
}
@Override
public boolean isWrapperFor(Class<?> iface) throws SQLException {
return false;
}
}