• 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 {
    
    	// 方式一:
    	@Test
    	public 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);
    	}
    
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28

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

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

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

    2.2.2 连接方式二

    	// 方式二:对方式一的迭代:在如下的程序中不出现第三方的api,使得程序具有更好的可移植性
    	@Test
    	public 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);
    	}
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19

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

    2.2.3 连接方式三

    	// 方式三:使用DriverManager替换Driver
    	@Test
    	public 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);
    	}
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19

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

    2.2.4 连接方式四

    	// 方式四:可以只是加载驱动,不用显示的注册驱动过了。
    	@Test
    	public void testConnection4() throws Exception {
    		// 1.提供三个连接的基本信息:
    		String url = "jdbc:mysql://localhost:3306/myDB";
    		String user = "root";
    		String password = "root";
    		
    		// 2.加载Driver
    		Class.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);
    	}
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30

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

    2.2.5 连接方式五(最终版)

    	//方式五(final版):将数据库连接需要的4个基本信息声明在配置文件中,通过读取配置文件的方式,获取连接
    	/*
    	 * 此种方式的好处?
    	 * 1.实现了数据与代码的分离。实现了解耦
    	 * 2.如果需要修改配置文件信息,可以避免程序重新打包。
    	 */
    	@Test
    	public 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);	
    		
    	}
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28

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

    jdbc.properties

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

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

    使用配置文件的好处:

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

    第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
      
      • 1
      • 2
    • 但是使用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方法省略了。。。
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    statement/crud/StatementTest.java

    package com.atguigu2.statement.crud;
    
    public class StatementTest {
    
    	// 使用Statement的弊端:需要拼写sql语句,并且存在SQL注入的问题
    	//如何避免出现sql注入:只要用 PreparedStatement(从Statement扩展而来) 取代 Statement
    	@Test
    	public 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;
    	}
    
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91
    • 92
    • 93
    • 94
    • 95
    • 96
    • 97
    • 98
    • 99
    • 100
    • 101
    • 102
    • 103
    • 104
    • 105
    • 106
    • 107
    • 108
    • 109
    • 110
    • 111
    • 112
    • 113
    • 114

    使用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 {
    	@Test
    	public 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();
    			// 获取结果集的元数据 :ResultSetMetaData
    			ResultSetMetaData 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;
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91

    综上:

    在这里插入图片描述

    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);
    			
    		}
    	}
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26

    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(),得到结果集:ResultSet
    			rs = ps.executeQuery();
    
    			// 5.得到结果集的元数据:ResultSetMetaData
    			ResultSetMetaData 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;
    
    	}
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55

    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();
    		}
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91

    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方法省略了。。。
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19

    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方法省略了。。。	
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    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 {
    	
    	@Test
    	public 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();
    			//获取结果集的元数据 :ResultSetMetaData
    			ResultSetMetaData 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;
    	}
    	
    	
    	@Test
    	public 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);
    			
    		}
    	}
    	
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91
    • 92
    • 93
    • 94
    • 95
    • 96
    • 97
    • 98
    • 99
    • 100
    • 101
    • 102
    • 103
    • 104
    • 105
    • 106
    • 107
    • 108
    • 109
    • 110
    • 111
    • 112
    • 113
    • 114
    • 115
    • 116
    • 117
    • 118
    • 119
    • 120
    • 121
    • 122

    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()获取的就是列名
    	 * 
    	 * 
    	 */
    	@Test
    	public 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++){
    					//获取每个列的列值:通过ResultSet
    					Object columnValue = rs.getObject(i + 1);
    					//通过ResultSetMetaData
    					//获取列的列名:getColumnName() --不推荐使用
    					//获取列的别名:getColumnLabel()
    //					String columnName = rsmd.getColumnName(i + 1);
    					String columnLabel = rsmd.getColumnLabel(i + 1);
    					
    					//通过反射,将对象指定名columnName的属性赋值为指定的值columnValue
    					Field 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;	
    	}
    	
    	
    	@Test
    	public 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);
    		}
    		
    	}
    	
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91
    • 92
    • 93
    • 94
    • 95
    • 96
    • 97
    • 98
    • 99
    • 100
    • 101
    • 102
    • 103
    • 104
    • 105
    • 106
    • 107
    • 108
    • 109
    • 110
    • 111
    • 112
    • 113
    • 114
    • 115
    • 116
    • 117

    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 {
    	
    	@Test
    	public 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();
    			// 获取结果集的元数据 :ResultSetMetaData
    			ResultSetMetaData 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;
    	}
    	
    	@Test
    	public 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();
    			// 获取结果集的元数据 :ResultSetMetaData
    			ResultSetMetaData 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;
    	}
    
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91
    • 92
    • 93
    • 94
    • 95
    • 96
    • 97
    • 98
    • 99
    • 100
    • 101
    • 102
    • 103
    • 104
    • 105
    • 106
    • 107
    • 108
    • 109
    • 110
    • 111
    • 112
    • 113
    • 114
    • 115
    • 116
    • 117
    • 118
    • 119
    • 120
    • 121
    • 122
    • 123
    • 124
    • 125
    • 126
    • 127
    • 128
    • 129
    • 130
    • 131
    • 132
    • 133
    • 134
    • 135
    • 136
    • 137
    • 138
    • 139
    • 140
    • 141
    • 142
    • 143
    • 144

    PreparedStatementUpdateTest.java

    package com.atguigu3.preparedstatement.crud;
    
    import com.atguigu1.connection.ConnectionTest;
    import com.atguigu3.util.JDBCUtils;
    
    /*
     * 使用PreparedStatement来替换Statement,实现对数据表的增删改操作
     * 
     * 增删改;查
     * 
     * 
     */
    public class PreparedStatementUpdateTest {
    	
    	@Test
    	public 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表的一条记录
    	@Test
    	public 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表中添加一条记录
    	@Test
    	public 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
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91
    • 92
    • 93
    • 94
    • 95
    • 96
    • 97
    • 98
    • 99
    • 100
    • 101
    • 102
    • 103
    • 104
    • 105
    • 106
    • 107
    • 108
    • 109
    • 110
    • 111
    • 112
    • 113
    • 114
    • 115
    • 116
    • 117
    • 118
    • 119
    • 120
    • 121
    • 122
    • 123
    • 124
    • 125
    • 126
    • 127
    • 128
    • 129
    • 130
    • 131
    • 132
    • 133
    • 134
    • 135

    练习

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

    在这里插入图片描述

    代码结构如下:

    在这里插入图片描述

    Exer1Test.java

    package com.atguigu4.exer;
    
    import com.atguigu3.util.JDBCUtils;
    //课后练习1
    public class Exer1Test {
    	
    	@Test
    	public 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;
    	}
    
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62

    练习题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方法省略了。。。
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20

    Exer2Test.java

    package com.atguigu4.exer;
    
    import com.atguigu3.util.JDBCUtils;
    
    //课后练习2
    public class Exer2Test {
    
    	// 问题1:向examstudent表中添加一条记录
    	/*
    	 *  Type: 
    		IDCard:
    		ExamCard:
    		StudentName:
    		Location:
    		Grade:
    	 */
    	@Test
    	public 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;
    	}
    
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77

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

    在这里插入图片描述

    	//问题2:根据身份证号或者准考证号查询学生成绩信息
    	@Test
    	public 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();
    			// 获取结果集的元数据 :ResultSetMetaData
    			ResultSetMetaData 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;
    	}
    	
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85

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

    在这里插入图片描述

    	//问题3:删除指定的学生信息
    	@Test
    	public 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("删除成功");
    			}
    		}
    	}
    
    	//优化以后的操作:
    	@Test
    	public 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("查无此人,请重新输入");
    		}
    	}
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
  • 相关阅读:
    AD9361寄存器功能笔记之本振频率设定
    Qt与MQTT交互通信
    计算机的计算单位
    微信小程序的事件绑定方式
    [附源码]计算机毕业设计基于springboot校园帮平台管理系统
    指令重排以及案例
    智能文件夹改名助手,秒级恢复原始文件夹名称,避免繁琐操作!
    STM32第十课:串口发送
    声音好听,颜值能打,基于PaddleGAN给人工智能AI语音模型配上动态画面(Python3.10)
    uniapp打包微信小程序。报错:https://api.weixin.qq.com 不在以下 request 合法域名列表
  • 原文地址:https://blog.csdn.net/qq_39236499/article/details/134298762