JDBC
1.介绍
JDBC(Java Database Connectivity)是一个独立于特定数据库管理系统、通用的SQL数据库存取和操作的公共接口(一组API),定义了用来访问数据库的标准Java类库,(java.sql,javax.sql) 使用这些类库可以以一种标准的方法、方便地访问数据库资源。
JDBC为访问不同的数据库提供了一种统一的途径,为开发者屏蔽了一些细节问题
JDBC接口(API) 包括两个层次 :
- 面向应用的APl: Java API,抽象接口,供应用程序开发人员使用( 连接数据库,执行SOL语句,获得结果)。
- 面向数据库的API:Java Driver API,供开发商开发数据库驱动程序用
2.JDBC程序编写步骤
3.获取数据库连接
java.sql.Driver 接口是所有JDBC驱动程序需要实现的接口。这个接口是提供给数据库厂商使用的,不同数据库厂商提供不同的实现。
在程序中不需要直接去访问实现了 Driver 接口的类,而是由驱动程序管理器类(java.sql.DriverManager)去调用这些Driver实现。
- Oracle的驱动 :oracle.jdbc.driver.OracleDriver
- mysql的驱动: com.mysql.cj.jdbc.Driver
下面是把数据源信息放在Properties文件中读取,获取数据库连接:
//读取配置文件
InputStream inputStream = ConnectionTest.class.getClassLoader().getResourceAsStream("jdbc.properties");
Properties pros = new Properties();
pros.load(inputStream);
String user = pros.getProperty("user");
String password = pros.getProperty("password");
String url = pros.getProperty("url");
String driverClass = pros.getProperty("driverClass");
//加载驱动
Class.forName(driverClass);
//获取连接
Connection connection = DriverManager.getConnection(url, user, password);
System.out.println(connection);
user=root
password=root
url=jdbc:mysql://localhost:3306/exercise
driverClass=com.mysql.cj.jdbc.Driver
因为一般由 DriverManager.getConnection来获取连接,获取Driver的实现类之后要向DriverManager注册驱动,这一步在源码中已经写好了,我们只需要用Class.forName(driverClass)加载驱动就可以了
4.操作和访问数据库
数据库连接被用于向数据库服务器发送命令和 SQL 语句,并接受数据库服务器返回的结果。其实一个数据库连接就是一个Socket连接
在java.sql包中有 3 个接口分别定义了对数据库的调用的不同方式:
- Statement: 用于执行静态 SOL 语句并返回它所生成结果的对象。
- PreparedStatement: SQL 语句被预编译并存储在此对象中,可以使用此对象多次高效地执行该语句
- CallableStatement: 用于执行 SQL 存储过程
4.1 使用Statement操作数据库存在弊端
问题一: 存在拼串操作,繁琐
问题二: 存在SQL注入问题
SQL注入是利用某些系统没有对用户输入的数据进行充分的检查,而在用户输入数据中注入非法的 SQL 语句段或命令(如 : SELECT user, password FROM user_table WHERE user=‘a’ OR 1=‘AND password =’ OR ‘1’=‘1’),从而利用系统的 SQL引擎完成恶意行为的做法。
对于Java 而言,要防范SQL注入,只要用 PreparedStatement(从Statement扩展而来)取代 Statement就可以了。
4.2 使用PreparedStatement操作数据库
4.2.1 插入
首先有一个customers表,表结构如下
插入一条数据
Connection connection = null;
PreparedStatement preparedStatement = null;
try {
//1.读取配置文件
InputStream inputStream =ClassLoader.getSystemClassLoader().getResourceAsStream("jdbc.properties");
Properties pros = new Properties();
pros.load(inputStream);
String user = pros.getProperty("user");
String password = pros.getProperty("password");
String url = pros.getProperty("url");
String driverClass = pros.getProperty("driverClass");
//2.加载驱动
Class.forName(driverClass);
//3.获取连接
connection = DriverManager.getConnection(url, user, password);
//4.预编译sql语句,返回PreparedStatement的实例
String sql = "insert into customers(name,email,birth) values(?,?,?)"; //? 表示占位符
preparedStatement = connection.prepareStatement(sql);
//5.填充占位符
preparedStatement.setString(1,"小智");
preparedStatement.setString(2,"123@qq.com");
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
Date date = sdf.parse("1999-01-01");
preparedStatement.setDate(3,new java.sql.Date(date.getTime()));
//6.执行操作
preparedStatement.execute();
} catch (Exception e) {
e.printStackTrace();
}finally {
//7.关闭资源
try {
if(preparedStatement != null)
preparedStatement.close();
} catch (SQLException e) {
e.printStackTrace();
}
try {
if(connection != null)
connection.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
结果:
4.2.2 封装JDBCUtils
作用:获取连接,关闭连接。这些操作都是一样的,所以封装到一个类里,方便使用
public class JDBCUtils {
/**
* 获取连接
* @return
* @throws Exception
*/
public static Connection getConnection() throws Exception{
//读取配置文件
InputStream inputStream = ClassLoader.getSystemClassLoader().getResourceAsStream("jdbc.properties");
Properties pros = new Properties();
pros.load(inputStream);
String user = pros.getProperty("user");
String password = pros.getProperty("password");
String url = pros.getProperty("url");
String driverClass = pros.getProperty("driverClass");
//加载驱动
Class.forName(driverClass);
//获取连接
Connection connection = DriverManager.getConnection(url, user, password);
return connection;
}
/**
* 关闭资源
* @param conn
* @param ps
*/
public static void closeResource(Connection conn, Statement ps){
//关闭资源
try {
if(ps != null)
ps.close();
} catch (SQLException e) {
e.printStackTrace();
}
try {
if(conn != null)
conn.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
/**
* 关闭资源
* @param conn
* @param ps
* @param rs
*/
public static void closeResource(Connection conn, Statement ps,ResultSet rs){
//关闭资源
try {
if(ps != null)
ps.close();
} catch (SQLException e) {
e.printStackTrace();
}
try {
if(conn != null)
conn.close();
} catch (SQLException e) {
e.printStackTrace();
}
try {
if(rs != null)
rs.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
}
4.2.3 修改
@Test
public void test2(){
String sql = "update customers set name = ? where id = ?";
update(sql,"小兰",1);
}
public void update(String sql,Object ...args){
Connection conn = null;
PreparedStatement ps = null;
try {
//1.获取数据库的连接
conn = JDBCUtils.getConnection();
//2.预编译sql语句,返回PreparedStatement的实例
ps = conn.prepareStatement(sql);
//3.填充占位符
for (int i = 0; i < args.length; i++) {
ps.setObject(i+1,args[i]);
}
//4.执行
ps.execute();
}catch (Exception e){
e.printStackTrace();
}finally {
//5.关闭资源
JDBCUtils.closeResource(conn,ps);
}
}
结果:
4.2.4 查询
ResultSet称为结果集,用来接收执行查询操作返回的结果
ResultSetMetaData,结果集的元数据,可用于获取有关ResultSet对象中列的类型和属性的信息。
返回单个结果:
@Test
public void test(){
String sql = "select id,name,email,birth from customers where id = ?";
Customer customer = query(Customer.class,sql, 1);
System.out.println(customer);
}
public <T> T query(Class<T> clazz,String sql,Object ...args){
Connection conn = null;
PreparedStatement ps = null;
ResultSet resultSet = null;
try {
conn = JDBCUtils.getConnection();
ps = conn.prepareStatement(sql);
for (int i = 0; i < args.length; i++) {
ps.setObject(i+1,args[i]);
}
resultSet = ps.executeQuery();
//获取结果集的元数据: ResultSetMetaData
ResultSetMetaData metaData = resultSet.getMetaData();
//通过ResultSetMetaData获取结果集中的列数
int columnCount = metaData.getColumnCount();
if(resultSet.next()){
T instance = clazz.newInstance();
//处理结果集一行数据中的每一个列
for (int i = 0; i < columnCount; i++) {
//获取每个列的列名
// String columnName = metaData.getColumnName(i + 1);
//获取每个列的别名,没有别名就是列名
String columnName = metaData.getColumnLabel(i + 1);
//获取每个列的值
Object columnValue = resultSet.getObject(i + 1);
//通过反射给customer对象指定的columnName属性,赋值为columnValue
Field field = clazz.getDeclaredField(columnName);
field.setAccessible(true);
field.set(instance,columnValue);
}
return instance;
}
} catch (Exception e) {
e.printStackTrace();
}finally {
JDBCUtils.closeResource(conn,ps,resultSet);
}
return null;
}
public class Customer {
private Integer id;
private String name;
private String email;
private Date birth;
public Customer(Integer id, String name, String email, Date birth) {
this.id = id;
this.name = name;
this.email = email;
this.birth = birth;
}
public Customer() {
}
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getEmail() {
return email;
}
public void setEmail(String email) {
this.email = email;
}
public Date getBirth() {
return birth;
}
public void setBirth(Date birth) {
this.birth = birth;
}
@Override
public String toString() {
return "Customer{" +
"id=" + id +
", name='" + name + '\'' +
", email='" + email + '\'' +
", birth=" + birth +
'}';
}
}
Customer{id=1, name='小兰', email='123@qq.com', birth=1999-01-01}
返回多个结果:
@Test
public void test2(){
String sql = "select id,name,email,birth from customers where id <= ?";
List<Customer> customers = queryList(Customer.class, sql, 2);
for (Customer customer : customers) {
System.out.println(customer);
}
}
public <T> List<T> queryList(Class<T> clazz, String sql, Object ...args){
Connection conn = null;
PreparedStatement ps = null;
ResultSet resultSet = null;
try {
conn = JDBCUtils.getConnection();
ps = conn.prepareStatement(sql);
for (int i = 0; i < args.length; i++) {
ps.setObject(i+1,args[i]);
}
resultSet = ps.executeQuery();
//获取结果集的元数据: ResultSetMetaData
ResultSetMetaData metaData = resultSet.getMetaData();
//通过ResultSetMetaData获取结果集中的列数
int columnCount = metaData.getColumnCount();
//创建集合对象
ArrayList<T> list = new ArrayList<>();
while (resultSet.next()){
T instance = clazz.newInstance();
//处理结果集一行数据中的每一个列
for (int i = 0; i < columnCount; i++) {
//获取每个列的列名
// String columnName = metaData.getColumnName(i + 1);
//获取每个列的别名
String columnName = metaData.getColumnLabel(i + 1);
//获取每个列的值
Object columnValue = resultSet.getObject(i + 1);
//通过反射给customer对象指定的columnName属性,赋值为columnValue
Field field = clazz.getDeclaredField(columnName);
field.setAccessible(true);
field.set(instance,columnValue);
}
list.add(instance);
}
return list;
} catch (Exception e) {
e.printStackTrace();
}finally {
JDBCUtils.closeResource(conn,ps,resultSet);
}
return null;
}
Customer{id=1, name='小兰', email='123@qq.com', birth=1999-01-01}
Customer{id=2, name='新一', email='777@qq.com', birth=2000-01-02}