JDBC(一)

第1章:JDBC概述

1.1 数据的持久化

  • 持久化(persistence):把数据保存到可掉电式存储设备中以供之后使用。大多数情况下,特别是企业级应用,数据持久化意味着将内存中的数据保存到硬盘上**,而持久化的实现过程大多通过各种关系数据库来完成**。

  • 持久化的主要应用是将内存中的数据存储在关系型数据库中,当然也可以存储在磁盘文件、XML数据文件中。

    在这里插入图片描述

1.2 Java中的数据存储技术

  • 在Java中,数据库存取技术可分为如下几类:

    • JDBC直接访问数据库

    • JDO (Java Data Object )技术

    • 第三方O/R工具,如Hibernate, Mybatis 等

  • JDBC是java访问数据库的基石,JDO、Hibernate、MyBatis等只是更好的封装了JDBC。

1.3 JDBC介绍

  • JDBC(Java Database Connectivity)是一个独立于特定数据库管理系统、通用的SQL数据库存取和操作的公共接口(一组API),定义了用来访问数据库的标准Java类库,(java.sql,javax.sql)使用这些类库可以以一种标准的方法、方便地访问数据库资源。
  • JDBC为访问不同的数据库提供了一种统一的途径,为开发者屏蔽了一些细节问题。
  • JDBC的目标是使Java程序员使用JDBC可以连接任何提供了JDBC驱动程序的数据库系统,这样就使得程序员无需对特定的数据库系统的特点有过多的了解,从而大大简化和加快了开发过程。
  • 如果没有JDBC,那么Java程序访问数据库时是这样的:

在这里插入图片描述


  • 有了JDBC,Java程序访问数据库时是这样的:

在这里插入图片描述


  • 总结如下:

在这里插入图片描述

1.4 JDBC体系结构

  • JDBC接口(API)包括两个层次:
    • 面向应用的API:Java API,抽象接口,供应用程序开发人员使用(连接数据库,执行SQL语句,获得结果)。
    • 面向数据库的API:Java Driver API,供开发商开发数据库驱动程序用。

JDBC是sun公司提供一套用于数据库操作的接口,java程序员只需要面向这套接口编程即可。

不同的数据库厂商,需要针对这套接口,提供不同实现。不同的实现的集合,即为不同数据库的驱动。 ————面向接口编程

1.5 JDBC程序编写步骤

在这里插入图片描述

补充:ODBC(Open Database Connectivity,开放式数据库连接),是微软在Windows平台下推出的。使用者在程序中只需要调用ODBC API,由 ODBC 驱动程序将调用转换成为对特定的数据库的调用请求。

第2章:获取数据库连接

2.1 配置

1、将junit4和mysql-cpnnector-java的jar包加入项目依赖中。

注意:mysql-cpnnector版本要和你的mysql数据库版本一致

运行代码需要先在你的电脑上安装mysql,并创建一个数据库。

mysql-connector下载地址:https://downloads.mysql.com/archives/c-j/

在这里插入图片描述

2、导入表数据

在这里插入图片描述

2.2 数据库连接方式举例

代码结构如下:

在这里插入图片描述

2.2.1 连接方式一

connection/ConnectionTest.java

package com.atguigu1.connection;public class ConnectionTest {// 方式一:@Testpublic void testConnection1() throws SQLException {// 获取Driver实现类对象Driver driver = new com.mysql.jdbc.Driver();// url:http://localhost:8080/gmall/keyboard.jpg// jdbc:mysql:协议// localhost:ip地址// 3306:默认mysql的端口号// test:test数据库String url = "jdbc:mysql://localhost:3306/myDB";// 将用户名和密码封装在Properties中Properties info = new Properties();info.setProperty("user", "root");info.setProperty("password", "root");Connection conn = driver.connect(url, info);System.out.println(conn);}}

说明:上述代码中显式出现了第三方数据库的API

注意:这里运行出现了一个错误:java.lang.NoClassDefFoundError: org/hamcrest/SelfDescribing

解决方法:https://blog.csdn.net/tenaguan4461/article/details/83505186

2.2.2 连接方式二

	// 方式二:对方式一的迭代:在如下的程序中不出现第三方的api,使得程序具有更好的可移植性@Testpublic void testConnection2() throws Exception {// 1.获取Driver实现类对象:使用反射Class clazz = Class.forName("com.mysql.cj.jdbc.Driver");Driver driver = (Driver) clazz.newInstance();// 2.提供要连接的数据库String url = "jdbc:mysql://localhost:3306/myDB";// 3.提供连接需要的用户名和密码Properties info = new Properties();info.setProperty("user", "root");info.setProperty("password", "root");// 4.获取连接Connection conn = driver.connect(url, info);System.out.println(conn);}

说明:相较于方式一,这里使用反射实例化Driver,不在代码中体现第三方数据库的API。体现了面向接口编程思想。

2.2.3 连接方式三

	// 方式三:使用DriverManager替换Driver@Testpublic void testConnection3() throws Exception {// 1.获取Driver实现类的对象Class clazz = Class.forName("com.mysql.jdbc.Driver");Driver driver = (Driver) clazz.newInstance();// 2.提供另外三个连接的基本信息:String url = "jdbc:mysql://localhost:3306/myDB";String user = "root";String password = "root";// 注册驱动DriverManager.registerDriver(driver);// 获取连接Connection conn = DriverManager.getConnection(url, user, password);System.out.println(conn);}

说明:使用DriverManager实现数据库的连接。体会获取连接必要的4个基本要素。

2.2.4 连接方式四

	// 方式四:可以只是加载驱动,不用显示的注册驱动过了。@Testpublic void testConnection4() throws Exception {// 1.提供三个连接的基本信息:String url = "jdbc:mysql://localhost:3306/myDB";String user = "root";String password = "root";// 2.加载DriverClass.forName("com.mysql.jdbc.Driver");//相较于方式三,可以省略如下的操作:
//		Driver driver = (Driver) clazz.newInstance();
//		// 注册驱动
//		DriverManager.registerDriver(driver);//为什么可以省略上述操作呢?/** 在mysql的Driver实现类中,声明了如下的操作:* static {try {java.sql.DriverManager.registerDriver(new Driver());} catch (SQLException E) {throw new RuntimeException("Can't register driver!");}}*/// 3.获取连接Connection conn = DriverManager.getConnection(url, user, password);System.out.println(conn);}

说明:不必显式的注册驱动了。因为在DriverManager的源码中已经存在静态代码块,实现了驱动的注册。

2.2.5 连接方式五(最终版)

	//方式五(final版):将数据库连接需要的4个基本信息声明在配置文件中,通过读取配置文件的方式,获取连接/** 此种方式的好处?* 1.实现了数据与代码的分离。实现了解耦* 2.如果需要修改配置文件信息,可以避免程序重新打包。*/@Testpublic void getConnection5() throws Exception{//1.读取配置文件中的4个基本信息InputStream is = ConnectionTest.class.getClassLoader().getResourceAsStream("jdbc.properties");Properties pros = new Properties();pros.load(is);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 conn = DriverManager.getConnection(url, user, password);System.out.println(conn);	}

其中,配置文件声明在工程的src目录下:

jdbc.properties

user=root
password=root
url=jdbc:mysql://localhost:3306/myDB?rewriteBatchedStatements=true
driverClass=com.mysql.cj.jdbc.Driver

说明:使用配置文件的方式保存配置信息,在代码中加载配置文件

使用配置文件的好处:

①实现了代码和数据的分离,如果需要修改配置信息,直接在配置文件中修改,不需要深入代码
②如果修改了配置信息,省去重新编译的过程。

第3章:PreparedStatement实现CRUD操作

3.1 操作和访问数据库

  • 数据库连接被用于向数据库服务器发送命令和 SQL 语句,并接受数据库服务器返回的结果。其实一个数据库连接就是一个Socket连接

  • 在 java.sql 包中有 3 个接口分别定义了对数据库的调用的不同方式:

    • Statement:用于执行静态 SQL 语句并返回它所生成结果的对象。
    • PrepatedStatement:SQL 语句被预编译并存储在此对象中,可以使用此对象多次高效地执行该语句。
    • CallableStatement:用于执行 SQL 存储过程

    在这里插入图片描述

3.2 Statement操作的弊端

  • 通过调用 Connection 对象的 createStatement() 方法创建该对象。该对象用于执行静态的 SQL 语句,并且返回执行结果。

  • Statement 接口中定义了下列方法用于执行 SQL 语句:

    int excuteUpdate(String sql):执行更新操作INSERTUPDATEDELETE
    ResultSet executeQuery(String sql):执行查询操作SELECT
    
  • 但是使用Statement操作数据表存在弊端:

    • 问题一:存在拼串操作,繁琐
    • 问题二:存在SQL注入问题
  • 对于 Java 而言,要防范 SQL 注入,只要用 PreparedStatement(从Statement扩展而来) 取代 Statement 就可以了。

代码结构:

在这里插入图片描述

代码演示:

User.java

package com.atguigu2.statement.crud;public class User {private String user;private String password;
//这里有参构造、无参构造、get、set、toString方法省略了。。。
}

statement/crud/StatementTest.java

package com.atguigu2.statement.crud;public class StatementTest {// 使用Statement的弊端:需要拼写sql语句,并且存在SQL注入的问题//如何避免出现sql注入:只要用 PreparedStatement(从Statement扩展而来) 取代 Statement@Testpublic void testLogin() {Scanner scanner = new Scanner(System.in);System.out.print("请输入用户名:");String user = scanner.nextLine();System.out.print("请输入密码:");String password = scanner.nextLine();//注入  拼串//SELECT user,password FROM user_table WHERE user = '1' or ' AND password = '=1 or '1' = '1'String sql = "SELECT user,password FROM user_table WHERE user = '"+ user +"' AND password = '"+ password +"'";User returnUser = get(sql,User.class);if(returnUser != null){System.out.println("登录成功");}else{System.out.println("用户名不存在或密码错误");}}// 使用Statement实现对数据表的查询操作public <T> T get(String sql, Class<T> clazz) {T t = null;Connection conn = null;Statement st = null;ResultSet rs = null;try {// 1.加载配置文件InputStream is = StatementTest.class.getClassLoader().getResourceAsStream("jdbc.properties");Properties pros = new Properties();pros.load(is);// 2.读取配置信息String user = pros.getProperty("user");String password = pros.getProperty("password");String url = pros.getProperty("url");String driverClass = pros.getProperty("driverClass");// 3.加载驱动Class.forName(driverClass);// 4.获取连接conn = DriverManager.getConnection(url, user, password);st = conn.createStatement();rs = st.executeQuery(sql);// 获取结果集的元数据ResultSetMetaData rsmd = rs.getMetaData();// 获取结果集的列数int columnCount = rsmd.getColumnCount();if (rs.next()) {t = clazz.newInstance();for (int i = 0; i < columnCount; i++) {// //1. 获取列的名称// String columnName = rsmd.getColumnName(i+1);// 1. 获取列的别名String columnName = rsmd.getColumnLabel(i + 1);// 2. 根据列名获取对应数据表中的数据Object columnVal = rs.getObject(columnName);// 3. 将数据表中得到的数据,封装进对象Field field = clazz.getDeclaredField(columnName);field.setAccessible(true);field.set(t, columnVal);}return t;}} catch (Exception e) {e.printStackTrace();} finally {// 关闭资源if (rs != null) {try {rs.close();} catch (SQLException e) {e.printStackTrace();}}if (st != null) {try {st.close();} catch (SQLException e) {e.printStackTrace();}}if (conn != null) {try {conn.close();} catch (SQLException e) {e.printStackTrace();}}}return null;}}

使用PreparedStatement代替:

PreparedStatementTest.java

package com.atguigu2.statement.crud;import com.atguigu3.util.JDBCUtils;/*** * @Description 演示使用PreparedStatement替换Statement,解决SQL注入问题* @author shkstart  Email:shkstart@126.com* @version * @date 上午11:52:37* * 除了解决Statement的拼串、sql问题之外,PreparedStatement还有哪些好处呢?* 1.PreparedStatement操作Blob的数据,而Statement做不到。* 2.PreparedStatement可以实现更高效的批量操作。**/
public class PreparedStatementTest {@Testpublic void testLogin() {Scanner scanner = new Scanner(System.in);System.out.println("请输入用户名:");String user = scanner.nextLine();System.out.println("请输入密码:");String password = scanner.nextLine();//SELECT user,password FROM user_table WHERE user = '1' or ' AND password = '=1 or '1' = '1'String sql = "SELECT user,password FROM user_table WHERE user = ? and password = ?";User returnUser = getInstance(User.class,sql,user,password);if(returnUser != null){System.out.println("登录成功");}else{System.out.println("用户名不存在或密码错误");}}/*** * @Description 针对于不同的表的通用的查询操作,返回表中的一条记录* @author shkstart* @date 上午11:42:23* @param clazz* @param sql* @param args* @return*/public <T> T getInstance(Class<T> clazz,String sql, Object... args) {Connection conn = null;PreparedStatement ps = null;ResultSet rs = null;try {conn = JDBCUtils.getConnection();ps = conn.prepareStatement(sql);for (int i = 0; i < args.length; i++) {ps.setObject(i + 1, args[i]);}rs = ps.executeQuery();// 获取结果集的元数据 :ResultSetMetaDataResultSetMetaData rsmd = rs.getMetaData();// 通过ResultSetMetaData获取结果集中的列数int columnCount = rsmd.getColumnCount();if (rs.next()) {T t = clazz.newInstance();// 处理结果集一行数据中的每一个列for (int i = 0; i < columnCount; i++) {// 获取列值Object columValue = rs.getObject(i + 1);// 获取每个列的列名// String columnName = rsmd.getColumnName(i + 1);String columnLabel = rsmd.getColumnLabel(i + 1);// 给t对象指定的columnName属性,赋值为columValue:通过反射Field field = clazz.getDeclaredField(columnLabel);field.setAccessible(true);field.set(t, columValue);}return t;}} catch (Exception e) {e.printStackTrace();} finally {JDBCUtils.closeResource(conn, ps, rs);}return null;}
}

综上:

在这里插入图片描述

3.3 PreparedStatement的使用

3.3.1 PreparedStatement介绍

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

  • PreparedStatement 接口是 Statement 的子接口,它表示一条预编译过的 SQL 语句

  • PreparedStatement 对象所代表的 SQL 语句中的参数用问号?来表示,调用 PreparedStatement 对象的 setXxx() 方法来设置这些参数.

  • setXxx() 方法有两个参数,第一个参数是要设置的 SQL 语句中的参数的索引(从 1 开始),第二个是设置的 SQL 语句中的参数的值

3.3.2 PreparedStatement vs Statement

  • 代码的可读性和可维护性。
  • PreparedStatement 能提高性能:
    • DBServer会对预编译语句提供性能优化。因为预编译语句有可能被重复调用,所以语句在被DBServer的编译器编译后的执行代码被缓存下来,那么下次调用时只要是相同的预编译语句就不需要编译,只要将参数直接传入编译过的语句执行代码中就会得到执行。
  • PreparedStatement 可以防止 SQL 注入

3.3.3 Java与SQL对应数据类型转换表

Java类型SQL类型
booleanBIT
byteTINYINT
shortSMALLINT
intINTEGER
longBIGINT
StringCHAR,VARCHAR,LONGVARCHAR
byte arrayBINARY , VAR BINARY
java.sql.DateDATE
java.sql.TimeTIME
java.sql.TimestampTIMESTAMP

3.3.4 使用PreparedStatement实现增、删、改操作

PreparedStatementTest.java

	//通用的增、删、改操作(体现一:增、删、改 ; 体现二:针对于不同的表)public void update(String sql,Object ... args){Connection conn = null;PreparedStatement ps = null;try {//1.获取数据库的连接conn = JDBCUtils.getConnection();//2.获取PreparedStatement的实例 (或:预编译sql语句)ps = conn.prepareStatement(sql);//3.填充占位符for(int i = 0;i < args.length;i++){ps.setObject(i + 1, args[i]);}//4.执行sql语句ps.execute();} catch (Exception e) {e.printStackTrace();}finally{//5.关闭资源JDBCUtils.closeResource(conn, ps);}}

3.3.5 使用PreparedStatement实现查询操作

	// 通用的针对于不同表的查询:返回一个对象 (version 1.0)public <T> T getInstance(Class<T> clazz, String sql, Object... args) {Connection conn = null;PreparedStatement ps = null;ResultSet rs = 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.执行executeQuery(),得到结果集:ResultSetrs = ps.executeQuery();// 5.得到结果集的元数据:ResultSetMetaDataResultSetMetaData rsmd = rs.getMetaData();// 6.1通过ResultSetMetaData得到columnCount,columnLabel;通过ResultSet得到列值int columnCount = rsmd.getColumnCount();if (rs.next()) {T t = clazz.newInstance();for (int i = 0; i < columnCount; i++) {// 遍历每一个列// 获取列值Object columnVal = rs.getObject(i + 1);// 获取列的别名:列的别名,使用类的属性名充当String columnLabel = rsmd.getColumnLabel(i + 1);// 6.2使用反射,给对象的相应属性赋值Field field = clazz.getDeclaredField(columnLabel);field.setAccessible(true);field.set(t, columnVal);}return t;}} catch (Exception e) {e.printStackTrace();} finally {// 7.关闭资源JDBCUtils.closeResource(conn, ps, rs);}return null;}

3.4 ResultSet与ResultSetMetaData

3.4.1 ResultSet

  • 查询需要调用PreparedStatement 的 executeQuery() 方法,查询结果是一个ResultSet 对象

  • ResultSet 对象以逻辑表格的形式封装了执行数据库操作的结果集,ResultSet 接口由数据库厂商提供实现

  • ResultSet 返回的实际上就是一张数据表。有一个指针指向数据表的第一条记录的前面。

  • ResultSet 对象维护了一个指向当前数据行的游标,初始的时候,游标在第一行之前,可以通过 ResultSet 对象的 next() 方法移动到下一行。调用 next()方法检测下一行是否有效。若有效,该方法返回 true,且指针下移。相当于Iterator对象的 hasNext() 和 next() 方法的结合体。

  • 当指针指向一行时, 可以通过调用 getXxx(int index) 或 getXxx(int columnName) 获取每一列的值。

    • 例如: getInt(1), getString(“name”)
    • 注意:Java与数据库交互涉及到的相关Java API中的索引都从1开始。
  • ResultSet 接口的常用方法:

    • boolean next()

    • getString()

    在这里插入图片描述

3.4.2 ResultSetMetaData

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

  • ResultSetMetaData meta = rs.getMetaData();

    • getColumnName(int column):获取指定列的名称

    • getColumnLabel(int column):获取指定列的别名

    • getColumnCount():返回当前 ResultSet 对象中的列数。

    • getColumnTypeName(int column):检索指定列的数据库特定的类型名称。

    • getColumnDisplaySize(int column):指示指定列的最大标准宽度,以字符为单位。

    • isNullable(int column):指示指定列中的值是否可以为 null。

    • isAutoIncrement(int column):指示是否自动为指定列进行编号,这样这些列仍然是只读的。

在这里插入图片描述
问题1:得到结果集后, 如何知道该结果集中有哪些列 ? 列名是什么?

​ 需要使用一个描述 ResultSet 的对象, 即 ResultSetMetaData

问题2:关于ResultSetMetaData

  1. 如何获取 ResultSetMetaData: 调用 ResultSet 的 getMetaData() 方法即可
  2. 获取 ResultSet 中有多少列:调用 ResultSetMetaData 的 getColumnCount() 方法
  3. 获取 ResultSet 每一列的列的别名是什么:调用 ResultSetMetaData 的getColumnLabel() 方法

在这里插入图片描述

3.5 实操

代码结构:

在这里插入图片描述

util/JDBCUtils.java

package com.atguigu3.util;/*** * @Description 操作数据库的工具类* @author shkstart Email:shkstart@126.com* @version* @date 上午9:10:02**/
public class JDBCUtils {/*** * @Description 获取数据库的连接* @author shkstart* @date 上午9:11:23* @return* @throws Exception*/public static Connection getConnection() throws Exception {// 1.读取配置文件中的4个基本信息InputStream is = ClassLoader.getSystemClassLoader().getResourceAsStream("jdbc.properties");Properties pros = new Properties();pros.load(is);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 conn = DriverManager.getConnection(url, user, password);return conn;}/*** * @Description 关闭连接和Statement的操作* @author shkstart* @date 上午9:12:40* @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();}}/*** * @Description 关闭资源操作* @author shkstart* @date 上午10:21:15* @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();}}
}

bean/Customer.java

package com.atguigu3.bean;import java.sql.Date;/** ORM编程思想  (object relational mapping)* 一个数据表对应一个java类* 表中的一条记录对应java类的一个对象* 表中的一个字段对应java类的一个属性* */
public class Customer {private int id;private String name;private String email;private Date birth;//这里有参构造、无参构造、get、set、toString方法省略了。。。
}

bean/Order.java

package com.atguigu3.bean;import java.sql.Date;public class Order {private int orderId;private String orderName;private Date orderDate;//这里有参构造、无参构造、get、set、toString方法省略了。。。	
}

CustomerForQuery.java

package com.atguigu3.preparedstatement.crud;import com.atguigu3.bean.Customer;
import com.atguigu3.util.JDBCUtils;/*** * @Description 针对于Customers表的查询操作* @author shkstart  Email:shkstart@126.com* @version * @date 上午10:04:55**/
public class CustomerForQuery {@Testpublic void testQueryForCustomers(){String sql = "select id,name,birth,email from customers where id = ?";Customer customer = queryForCustomers(sql, 13);System.out.println(customer);sql = "select name,email from customers where name = ?";Customer customer1 = queryForCustomers(sql,"周杰伦");System.out.println(customer1);}/*** * @Description 针对于customers表的通用的查询操作* @author shkstart* @throws Exception * @date 上午10:23:40*/public Customer queryForCustomers(String sql,Object...args){Connection conn = null;PreparedStatement ps = null;ResultSet rs = null;try {conn = JDBCUtils.getConnection();ps = conn.prepareStatement(sql);for(int i = 0;i < args.length;i++){ps.setObject(i + 1, args[i]);}rs = ps.executeQuery();//获取结果集的元数据 :ResultSetMetaDataResultSetMetaData rsmd = rs.getMetaData();//通过ResultSetMetaData获取结果集中的列数int columnCount = rsmd.getColumnCount();if(rs.next()){Customer cust = new Customer();//处理结果集一行数据中的每一个列for(int i = 0;i <columnCount;i++){//获取列值Object columValue = rs.getObject(i + 1);//获取每个列的列名
//					String columnName = rsmd.getColumnName(i + 1);String columnLabel = rsmd.getColumnLabel(i + 1);//给cust对象指定的columnName属性,赋值为columValue:通过反射Field field = Customer.class.getDeclaredField(columnLabel);field.setAccessible(true);field.set(cust, columValue);}return cust;}} catch (Exception e) {e.printStackTrace();}finally{JDBCUtils.closeResource(conn, ps, rs);}return null;}@Testpublic void testQuery1() {Connection conn = null;PreparedStatement ps = null;ResultSet resultSet = null;try {conn = JDBCUtils.getConnection();String sql = "select id,name,email,birth from customers where id = ?";ps = conn.prepareStatement(sql);ps.setObject(1, 1);//执行,并返回结果集resultSet = ps.executeQuery();//处理结果集if(resultSet.next()){//next():判断结果集的下一条是否有数据,如果有数据返回true,并指针下移;如果返回false,指针不会下移。//获取当前这条数据的各个字段值int id = resultSet.getInt(1);String name = resultSet.getString(2);String email = resultSet.getString(3);Date birth = resultSet.getDate(4);//方式一:
//			System.out.println("id = " + id + ",name = " + name + ",email = " + email + ",birth = " + birth);//方式二:
//			Object[] data = new Object[]{id,name,email,birth};//方式三:将数据封装为一个对象(推荐)Customer customer = new Customer(id, name, email, birth);System.out.println(customer);}} catch (Exception e) {e.printStackTrace();}finally{//关闭资源JDBCUtils.closeResource(conn, ps, resultSet);}}}

OrderForQuery.java

package com.atguigu3.preparedstatement.crud;import com.atguigu3.bean.Order;
import com.atguigu3.util.JDBCUtils;/*** * @Description 针对于Order表的通用的查询操作* @author shkstart  Email:shkstart@126.com* @version * @date 上午10:43:58**/
public class OrderForQuery {/** 针对于表的字段名与类的属性名不相同的情况:* 1. 必须声明sql时,使用类的属性名来命名字段的别名* 2. 使用ResultSetMetaData时,需要使用getColumnLabel()来替换getColumnName(),*    获取列的别名。*  说明:如果sql中没有给字段其别名,getColumnLabel()获取的就是列名* * */@Testpublic void testOrderForQuery(){String sql = "select order_id orderId,order_name orderName,order_date orderDate from `order` where order_id = ?";Order order = orderForQuery(sql,1);System.out.println(order);}/*** * @Description 通用的针对于order表的查询操作* @author shkstart* @date 上午10:51:12* @return* @throws Exception */public Order orderForQuery(String sql,Object...args){Connection conn = null;PreparedStatement ps = null;ResultSet rs = null;try {conn = JDBCUtils.getConnection();ps = conn.prepareStatement(sql);for(int i = 0;i < args.length;i++){ps.setObject(i + 1, args[i]);}//执行,获取结果集rs = ps.executeQuery();//获取结果集的元数据ResultSetMetaData rsmd = rs.getMetaData();//获取列数int columnCount = rsmd.getColumnCount();if(rs.next()){Order order = new Order();for(int i = 0;i < columnCount;i++){//获取每个列的列值:通过ResultSetObject columnValue = rs.getObject(i + 1);//通过ResultSetMetaData//获取列的列名:getColumnName() --不推荐使用//获取列的别名:getColumnLabel()
//					String columnName = rsmd.getColumnName(i + 1);String columnLabel = rsmd.getColumnLabel(i + 1);//通过反射,将对象指定名columnName的属性赋值为指定的值columnValueField field = Order.class.getDeclaredField(columnLabel);field.setAccessible(true);field.set(order, columnValue);}return order;}} catch (Exception e) {e.printStackTrace();}finally{JDBCUtils.closeResource(conn, ps, rs);}return null;	}@Testpublic void testQuery1(){Connection conn = null;PreparedStatement ps = null;ResultSet rs = null;try {conn = JDBCUtils.getConnection();String sql = "select order_id,order_name,order_date from `order` where order_id = ?";ps = conn.prepareStatement(sql);ps.setObject(1, 1);rs = ps.executeQuery();if(rs.next()){int id = (int) rs.getObject(1);String name = (String) rs.getObject(2);Date date = (Date) rs.getObject(3);Order order = new Order(id, name, date);System.out.println(order);}} catch (Exception e) {e.printStackTrace();}finally{JDBCUtils.closeResource(conn, ps, rs);}}}

PreparedStatementQueryTest.java

package com.atguigu3.preparedstatement.crud;import com.atguigu3.bean.Customer;
import com.atguigu3.bean.Order;
import com.atguigu3.util.JDBCUtils;/*** * @Description 使用PreparedStatement实现针对于不同表的通用的查询操作* @author shkstart Email:shkstart@126.com* @version* @date 上午11:32:55**/
public class PreparedStatementQueryTest {@Testpublic void testGetForList(){String sql = "select id,name,email from customers where id < ?";List<Customer> list = getForList(Customer.class,sql,12);list.forEach(System.out::println);String sql1 = "select order_id orderId,order_name orderName from `order`";List<Order> orderList = getForList(Order.class, sql1);orderList.forEach(System.out::println);}public <T> List<T> getForList(Class<T> clazz,String sql, Object... args){Connection conn = null;PreparedStatement ps = null;ResultSet rs = null;try {conn = JDBCUtils.getConnection();ps = conn.prepareStatement(sql);for (int i = 0; i < args.length; i++) {ps.setObject(i + 1, args[i]);}rs = ps.executeQuery();// 获取结果集的元数据 :ResultSetMetaDataResultSetMetaData rsmd = rs.getMetaData();// 通过ResultSetMetaData获取结果集中的列数int columnCount = rsmd.getColumnCount();//创建集合对象ArrayList<T> list = new ArrayList<T>();while (rs.next()) {T t = clazz.newInstance();// 处理结果集一行数据中的每一个列:给t对象指定的属性赋值for (int i = 0; i < columnCount; i++) {// 获取列值Object columValue = rs.getObject(i + 1);// 获取每个列的列名// String columnName = rsmd.getColumnName(i + 1);String columnLabel = rsmd.getColumnLabel(i + 1);// 给t对象指定的columnName属性,赋值为columValue:通过反射Field field = clazz.getDeclaredField(columnLabel);field.setAccessible(true);field.set(t, columValue);}list.add(t);}return list;} catch (Exception e) {e.printStackTrace();} finally {JDBCUtils.closeResource(conn, ps, rs);}return null;}@Testpublic void testGetInstance(){String sql = "select id,name,email from customers where id = ?";Customer customer = getInstance(Customer.class,sql,12);System.out.println(customer);String sql1 = "select order_id orderId,order_name orderName from `order` where order_id = ?";Order order = getInstance(Order.class, sql1, 1);System.out.println(order);}/*** * @Description 针对于不同的表的通用的查询操作,返回表中的一条记录* @author shkstart* @date 上午11:42:23* @param clazz* @param sql* @param args* @return*/public <T> T getInstance(Class<T> clazz,String sql, Object... args) {Connection conn = null;PreparedStatement ps = null;ResultSet rs = null;try {conn = JDBCUtils.getConnection();ps = conn.prepareStatement(sql);for (int i = 0; i < args.length; i++) {ps.setObject(i + 1, args[i]);}rs = ps.executeQuery();// 获取结果集的元数据 :ResultSetMetaDataResultSetMetaData rsmd = rs.getMetaData();// 通过ResultSetMetaData获取结果集中的列数int columnCount = rsmd.getColumnCount();if (rs.next()) {T t = clazz.newInstance();// 处理结果集一行数据中的每一个列for (int i = 0; i < columnCount; i++) {// 获取列值Object columValue = rs.getObject(i + 1);// 获取每个列的列名// String columnName = rsmd.getColumnName(i + 1);String columnLabel = rsmd.getColumnLabel(i + 1);// 给t对象指定的columnName属性,赋值为columValue:通过反射Field field = clazz.getDeclaredField(columnLabel);field.setAccessible(true);field.set(t, columValue);}return t;}} catch (Exception e) {e.printStackTrace();} finally {JDBCUtils.closeResource(conn, ps, rs);}return null;}}

PreparedStatementUpdateTest.java

package com.atguigu3.preparedstatement.crud;import com.atguigu1.connection.ConnectionTest;
import com.atguigu3.util.JDBCUtils;/** 使用PreparedStatement来替换Statement,实现对数据表的增删改操作* * 增删改;查* * */
public class PreparedStatementUpdateTest {@Testpublic void testCommonUpdate(){
//		String sql = "delete from customers where id = ?";
//		update(sql,3);String sql = "update `order` set order_name = ? where order_id = ?";update(sql,"DD","2");}//通用的增删改操作public void update(String sql,Object ...args){//sql中占位符的个数与可变形参的长度相同!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);}}//修改customers表的一条记录@Testpublic void testUpdate(){Connection conn = null;PreparedStatement ps = null;try {//1.获取数据库的连接conn = JDBCUtils.getConnection();//2.预编译sql语句,返回PreparedStatement的实例String sql = "update customers set name = ? where id = ?";ps = conn.prepareStatement(sql);//3.填充占位符ps.setObject(1,"莫扎特");ps.setObject(2, 18);//4.执行ps.execute();} catch (Exception e) {e.printStackTrace();}finally{//5.资源的关闭JDBCUtils.closeResource(conn, ps);}}// 向customers表中添加一条记录@Testpublic void testInsert() {Connection conn = null;PreparedStatement ps = null;try {// 1.读取配置文件中的4个基本信息InputStream is = ClassLoader.getSystemClassLoader().getResourceAsStream("jdbc.properties");Properties pros = new Properties();pros.load(is);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.获取连接conn = DriverManager.getConnection(url, user, password);//		System.out.println(conn);//4.预编译sql语句,返回PreparedStatement的实例String sql = "insert into customers(name,email,birth)values(?,?,?)";//?:占位符ps = conn.prepareStatement(sql);//5.填充占位符ps.setString(1, "哪吒");ps.setString(2, "nezha@gmail.com");SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");java.util.Date date = sdf.parse("1000-01-01");ps.setDate(3, new Date(date.getTime()));//6.执行操作ps.execute();} catch (Exception e) {e.printStackTrace();}finally{//7.资源的关闭try {if(ps != null)ps.close();} catch (SQLException e) {e.printStackTrace();}try {if(conn != null)conn.close();} catch (SQLException e) {e.printStackTrace();}}}}

练习

练习题1:从控制台向数据库的表customers中插入一条数据,表结构如下:

在这里插入图片描述

代码结构如下:

在这里插入图片描述

Exer1Test.java

package com.atguigu4.exer;import com.atguigu3.util.JDBCUtils;
//课后练习1
public class Exer1Test {@Testpublic void testInsert(){Scanner scanner = new Scanner(System.in);System.out.print("请输入用户名:");String name = scanner.next();System.out.print("请输入邮箱:");String email = scanner.next();System.out.print("请输入生日:");String birthday = scanner.next();//'1992-09-08'String sql = "insert into customers(name,email,birth)values(?,?,?)";int insertCount = update(sql,name,email,birthday);if(insertCount > 0){System.out.println("添加成功");}else{System.out.println("添加失败");}}// 通用的增删改操作public int update(String sql, Object... args) {// sql中占位符的个数与可变形参的长度相同!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():* 如果执行的是查询操作,有返回结果,则此方法返回true;* 如果执行的是增、删、改操作,没有返回结果,则此方法返回false.*///方式一:
//			return ps.execute();//方式二:return ps.executeUpdate();} catch (Exception e) {e.printStackTrace();} finally {// 5.资源的关闭JDBCUtils.closeResource(conn, ps);}return 0;}}

练习题2:创立数据库表 examstudent,表结构如下:

在这里插入图片描述

向数据表中添加如下数据:

在这里插入图片描述

代码实现1:插入一个新的student 信息

请输入考生的详细信息

Type:
IDCard:
ExamCard:
StudentName:
Location:
Grade:

信息录入成功!

Student.java

package com.atguigu4.exer;/** Type: 
IDCard:
ExamCard:
StudentName:
Location:
Grade:*/
public class Student {private int flowID;//流水号private int type;//考试类型private String IDCard;//身份证号private String examCard;//准考证号private String name;//学生姓名private String location;//所在城市//这里有参构造、无参构造、get、set、toString方法省略了。。。
}

Exer2Test.java

package com.atguigu4.exer;import com.atguigu3.util.JDBCUtils;//课后练习2
public class Exer2Test {// 问题1:向examstudent表中添加一条记录/**  Type: IDCard:ExamCard:StudentName:Location:Grade:*/@Testpublic void testInsert(){Scanner scanner = new Scanner(System.in);System.out.print("四级/六级:");int type = scanner.nextInt();System.out.print("身份证号:");String IDCard = scanner.next();System.out.print("准考证号:");String examCard = scanner.next();System.out.print("学生姓名:");String studentName = scanner.next();System.out.print("所在城市:");String location = scanner.next();System.out.print("考试成绩:");int grade = scanner.nextInt();String sql = "insert into examstudent(type,IDCard,examCard,studentName,location,grade)values(?,?,?,?,?,?)";int insertCount = update(sql,type,IDCard,examCard,studentName,location,grade);if(insertCount > 0){System.out.println("添加成功");}else{System.out.println("添加失败");}}// 通用的增删改操作public int update(String sql, Object... args) {// sql中占位符的个数与可变形参的长度相同!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(): * 如果执行的是查询操作,有返回结果,则此方法返回true;* 如果执行的是增、删、改操作,没有返回结果,则此方法返回false.*/// 方式一:// return ps.execute();// 方式二:return ps.executeUpdate();} catch (Exception e) {e.printStackTrace();} finally {// 5.资源的关闭JDBCUtils.closeResource(conn, ps);}return 0;}}

代码实现2:在 eclipse中建立 java 程序:输入身份证号或准考证号可以查询到学生的基本信息。结果如下:

在这里插入图片描述

	//问题2:根据身份证号或者准考证号查询学生成绩信息@Testpublic void queryWithIDCardOrExamCard(){System.out.println("请选择您要输入的类型:");System.out.println("a.准考证号");System.out.println("b.身份证号");Scanner scanner = new Scanner(System.in);String selection = scanner.next();if("a".equalsIgnoreCase(selection)){//if(selection.equalsIgnoreCase("a")){System.out.println("请输入准考证号:");String examCard = scanner.next();String sql = "select FlowID flowID,Type type,IDCard,ExamCard examCard,StudentName name,Location location,Grade grade from examstudent where examCard = ?";Student student = getInstance(Student.class,sql,examCard);if(student != null){System.out.println(student);}else{System.out.println("输入的准考证号有误!");}}else if("b".equalsIgnoreCase(selection)){System.out.println("请输入身份证号:");String IDCard = scanner.next();String sql = "select FlowID flowID,Type type,IDCard,ExamCard examCard,StudentName name,Location location,Grade grade from examstudent where IDCard = ?";Student student = getInstance(Student.class,sql,IDCard);if(student != null){System.out.println(student);}else{System.out.println("输入的身份证号有误!");}}else{System.out.println("您的输入有误,请重新进入程序。");}}public <T> T getInstance(Class<T> clazz,String sql, Object... args) {Connection conn = null;PreparedStatement ps = null;ResultSet rs = null;try {conn = JDBCUtils.getConnection();ps = conn.prepareStatement(sql);for (int i = 0; i < args.length; i++) {ps.setObject(i + 1, args[i]);}rs = ps.executeQuery();// 获取结果集的元数据 :ResultSetMetaDataResultSetMetaData rsmd = rs.getMetaData();// 通过ResultSetMetaData获取结果集中的列数int columnCount = rsmd.getColumnCount();if (rs.next()) {T t = clazz.newInstance();// 处理结果集一行数据中的每一个列for (int i = 0; i < columnCount; i++) {// 获取列值Object columValue = rs.getObject(i + 1);// 获取每个列的列名// String columnName = rsmd.getColumnName(i + 1);String columnLabel = rsmd.getColumnLabel(i + 1);// 给t对象指定的columnName属性,赋值为columValue:通过反射Field field = clazz.getDeclaredField(columnLabel);field.setAccessible(true);field.set(t, columValue);}return t;}} catch (Exception e) {e.printStackTrace();} finally {JDBCUtils.closeResource(conn, ps, rs);}return null;}

代码实现3:完成学生信息的删除功能

在这里插入图片描述

	//问题3:删除指定的学生信息@Testpublic void testDeleteByExamCard(){System.out.println("请输入学生的考号:");Scanner scanner = new Scanner(System.in);String examCard = scanner.next();//查询指定准考证号的学生String sql = "select FlowID flowID,Type type,IDCard,ExamCard examCard,StudentName name,Location location,Grade grade from examstudent where examCard = ?";Student student = getInstance(Student.class,sql,examCard);if(student == null){System.out.println("查无此人,请重新输入");}else{String sql1 = "delete from examstudent where examCard = ?";int deleteCount = update(sql1, examCard);if(deleteCount > 0){System.out.println("删除成功");}}}//优化以后的操作:@Testpublic void testDeleteByExamCard1(){System.out.println("请输入学生的考号:");Scanner scanner = new Scanner(System.in);String examCard = scanner.next();String sql = "delete from examstudent where examCard = ?";int deleteCount = update(sql, examCard);if(deleteCount > 0){System.out.println("删除成功");}else{System.out.println("查无此人,请重新输入");}}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/165003.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Linux】:初识git || centos下安装git || 创建本地仓库 || 配置本地仓库 || 认识工作区/暂存区(索引)以及版本库

&#x1f4ee;1.初识git Git 原理与使用 课程⽬标 • 技术⽬标:掌握Git企业级应⽤&#xff0c;深刻理解Git操作过程与操作原理&#xff0c;理解⼯作区&#xff0c;暂存区&#xff0c;版本库的含义 • 技术⽬标:掌握Git版本管理&#xff0c;⾃由进⾏版本回退、撤销、修改等Git操…

纯干货:赝势的选择 | VASP计算入门教程,真的超级有用

VASP软件是基于贋势和平面波基组的第一性原理密度泛函计算程序。VASP使用的是平面波基组&#xff0c;电子与原子核之间的相互作用使用投影缀加波贋势&#xff08;Projector Augmented Wave&#xff0c;PAW&#xff09;方法描述&#xff0c;从而进行量子力学计算。VASP采用PAW贋…

【ReID】1、行人重识别模型

文章目录 一、概念二、实现方案三、代码 一、概念 ReID&#xff0c;也就是 Re-identification&#xff0c;其定义是利用算法&#xff0c;在图像库中找到要搜索的目标的技术&#xff0c;所以它是属于图像检索的一个子问题。 说白了&#xff0c;在监控拍不到人脸的情况下&#…

手机玻璃盖板为什么需要透光率检测

手机盖板&#xff0c;也称为手机壳或保护套&#xff0c;是一种用于保护手机外观和延长使用寿命的装置。它们通常由塑料、硅胶、玻璃或金属等材料制成&#xff0c;并固定在手机外壳上,其中任何一个工序出现差错&#xff0c;都有可能导致手机盖板产生缺陷&#xff0c;例如漏油、透…

《持续交付:发布可靠软件的系统方法》- 读书笔记(十二)

持续交付&#xff1a;发布可靠软件的系统方法&#xff08;十二&#xff09; 第 12 章 数据管理12.1 引言12.2 数据库脚本化12.3 增量式修改12.3.1 对数据库进行版本控制12.3.2 联合环境中的变更管理 12.4 数据库回滚和无停机发布12.4.1 保留数据的回滚12.4.2 将应用程序部署与数…

Bean的四种实例化方式以及BeanFactory和FactoryBean的区别

2023.11.8 Spring为Bean提供了多种实例化方式&#xff0c;通常包括4种方式。 第一种&#xff1a;通过构造方法实例化第二种&#xff1a;通过简单工厂模式实例化第三种&#xff1a;通过factory-bean实例化第四种&#xff1a;通过FactoryBean接口实例化 通过构造方法实例化 创…

一文掌握 Apache SkyWalking

Apache SkyWalking SkyWalking是一个开源可观测平台&#xff0c;用于收集、分析、聚合和可视化来自服务和云原生基础设施的数据。SkyWalking 提供了一种简单的方法来保持分布式系统的清晰视图&#xff0c;甚至跨云。它是一种现代APM&#xff0c;专为云原生、基于容器的分布式系…

大厂面试题-b树和b+树的理解

为了更清晰的解答这个问题&#xff0c;从三个方面来回答&#xff1a; a.了解二叉树、AVL树、B树的概念 b.B树和B树的应用场景 1.B树是一种多路平衡查找树&#xff0c;为了更形象的理解&#xff0c;我们来看这张图。 二叉树&#xff0c;每个节点支持两个分支的树结构&#xff…

2023年11月数据库流行度最新排名

点击查看最新数据库流行度最新排名&#xff08;每月更新&#xff09; 2023年11月数据库流行度最新排名 TOP DB顶级数据库索引是通过分析在谷歌上搜索数据库名称的频率来创建的 一个数据库被搜索的次数越多&#xff0c;这个数据库就被认为越受欢迎。这是一个领先指标。原始数…

四种常见分布式限流算法实现!

转载&#xff1a;四种常见分布式限流算法实现&#xff01; - 知乎 大家好&#xff0c;我是老三&#xff0c;最近公司在搞年终大促&#xff0c;随着各种营销活动“组合拳”打出&#xff0c;进站流量时不时会有一个小波峰&#xff0c;一般情况下&#xff0c;当然是流量越多越好&…

为什么HTTP用得很好的,开始普及HTTPS呢?

显而易见&#xff0c;现在的HTTP早已不安全&#xff0c;当我们在浏览各个网站时会发现HTTP前面都会显示不安全&#xff0c;因为HTTP是明文传输&#xff0c;一旦电脑被植入了木马&#xff0c;木马程序就会主动周期性发消息给Internet的控制终端&#xff0c;这样NAT小洞会一直敞开…

前端瀑布流怎么布局

假设你需要把页面搭建成这样&#xff0c;有两列元素&#xff0c;每个元素宽度一定&#xff0c;高度不一定。如何实现&#xff1f; 这种瀑布流布局有很多实现方式&#xff0c;比如Grid布局。 什么是Grid布局 Grid布局是一种很灵活的布局方式&#xff0c;他把你的页面划分成了很…