JDBC 概述
客户端操作数据库的方式
方式1: 使用第三方客户端来访问 MySQL:SQLyog
方式2: 使用命令行

3) 我们今天要学习的是通过 Java程序 来访问 MySQL 数据库
什么是JDBC
JDBC( Java Data Base Connectivity) 是 Java 访问数据库的标准规范.是一种用于执行SQL语句的Java API,可以为多种关系数据库提供统一访问,它由一组用Java语言编写的类和接口组成。是Java访问数据库的标准规范.
JDBC 原理
JDBC是接口,驱动是接口的实现,没有驱动将无法完成数据库连接,从而不能操作数据库!每个数据库厂商都需要提供自己的驱动,用来连接自己公司的数据库,也就是说驱动一般都由数据库生成厂商提供。

总结:
JDBC就是由sun公司定义的一套操作所有关系型数据库的规则(接口),而数据库厂商需要实现这套接口,提供数据库驱动jar包, 我们可以使用这套接口编程,真正执行的代码是对应驱动包中的实现类。
JDBC 开发
数据准备
-- 创建 jdbc_user表CREATE TABLE jdbc_user (id INT PRIMARY KEY AUTO_INCREMENT ,username VARCHAR(50), PASSWORD VARCHAR(50),birthday DATE);
-- 添加数据INSERT INTO jdbc_user (username, PASSWORD,birthday) VALUES('admin1', '123','1991/12/24'),('admin2','123','1995/12/24'),('test1', '123','1998/12/24'),('test2', '123','2000/12/24');
MySql驱动包
- 将MySQL驱动包添加到jar包库文件夹中,Myjar文件夹,用于存放当前项目需要的所有jar包

在 idea中 配置jar包库的位置创建一个新的项目jdbc_task01, 配置jar包库


API使用: 1.注册驱动
JDBC规范定义驱动接口: java.sql.Driver
MySql驱动包提供了实现类:
com.mysql.jdbc.Driver
| 加载注册驱动的方式 | 描述 |
|---|---|
| Class.forName(数据库驱动实现类) | 加载和注册数据库驱动,数据库驱动由数据库厂商MySql提供 “com.mysql.jdbc.Driver” |
代码示例
public class JDBCDemo01 {public static void main(String[] args) throws ClassNotFoundException {//1.注册驱动// forName 方法执行将类进行初始化Class.forName("com.mysql.jdbc.Driver");}}
为什么这样可以注册驱动?
我们知道 Class类的forName方法 ,可以将一个类初始化, 现在我们一起Driver类的 看一下源码
// Driver类是MySql提供的数据库驱动类, 实现了JDBC的Driver接口 java.sql.Driverpublic class Driver extends NonRegisteringDriver implements java.sql.Driver {// 空参构造public Driver() throws SQLException {}//静态代码块,Class类的 forName()方法将Driver类 加载到内存, static代码块会自动执行static {try {/*DriverManager 驱动管理类registerDriver(new Driver) 注册驱动的方法注册数据库驱动*/DriverManager.registerDriver(new Driver());} catch (SQLException var1) {throw new RuntimeException("Can't register driver!");}}}
注:
从 JDBC3 开始,目前已经普遍使用的版本。可以不用注册驱动而直接使用。 Class.forName 这句话可以省略。
API使用: 2.获得连接
Connection 接口,代表一个连接对象 ,具体的实现类由数据库的厂商实现使用 DriverManager类的静态方法,getConnection可以获取数据库的连接
| 获取连接的静态方法 | 说明 |
|---|---|
| Connection getConnection(String url, String user, String password) | 通过连接字符串和用户名,密码来获取数据库连接对象 |
getConnection方法 3个 连接参数说明
| 连接参数 | 说明 |
|---|---|
| user | 登录用户名 |
| password | 登录密码 |
| url | mySql URL的格式jdbc:mysql://localhost:3306/db4 |
对URL的详细说明
jdbc:mysql://localhost:3306/db4?characterEncoding=UTF-8
JDBC规定url的格式由三部分组成,每个部分中间使用冒号分隔。 第一部分是协议 jdbc,这是固定的;
第二部分是子协议,就是数据库名称,连接mysql数据库,第二部分当然是mysql了;
第三部分是由数据库厂商规定的,我们需要了解每个数据库厂商的要求,mysql的第三部分分别由数据库服务器的IP地址(localhost)、端口号(3306),以及要使用的 数据库名称 组成。
- 代码示例
public class JDBCDemo02 {public static void main(String[] args) throws Exception {//1.注册驱动Class.forName("com.mysql.jdbc.Driver");//2.获取连接 url,用户名, 密码String url = "jdbc:mysql://localhost:3306/db4";Connection con = DriverManager.getConnection(url, "root", "123456");//com.mysql.jdbc.JDBC4Connection@2e3fc542 System.out.println(con);}}
API 使用: 3.获取语句执行平台
通过Connection 的 createStatement方法 获取sql语句执行对象
| Connection接口中的方法 | 说明 |
|---|---|
| Statement createStatement() | 创建 SQL语句执行对象 |
Statement : 代表一条语句对象,用于发送 SQL 语句给服务器,用于执行静态 SQL 语句并返回它所生成结果的对象。
| Statement类 常用方法 | 说明 |
|---|---|
int executeUpdate(String sql); |
执行insert update delete语句.返回int类型,代表受影响的行数 |
| ResultSet executeQuery(String sql); | 执行select语句, 返回ResultSet结果集对象 |
代码示例
public class JDBCDemo03 {public static void main(String[] args) throws Exception {//1.注册驱动Class.forName("com.mysql.jdbc.Driver");//2.获取连接 url,用户名, 密码String url = "jdbc:mysql://localhost:3306/db4";Connection con = DriverManager.getConnection(url, "root", "123456");//3.获取 Statement对象Statement statement = con.createStatement();//4.执行创建表操作String sql = "create table test01(id int, name varchar(20),age int);";//5.增删改操作 使用executeUpdate,增加一张表int i = statement.executeUpdate(sql);//6.返回值是受影响的函数System.out.println(i);//7.关闭流statement.close();con.close();}}
API 使用: 4.处理结果集
只有在进行查询操作的时候, 才会处理结果集代码示例
public class JDBCDemo04 {public static void main(String[] args) throws SQLException {//1.注册驱动 可以省略//2.获取连接String url = "jdbc:mysql://localhost:3306/db4";Connection con = DriverManager.getConnection(url, "root", "123456");//3.获取 Statement对象Statement statement = con.createStatement();String sql = "select * from jdbc_user";//执行查询操作,返回的是一个 ResultSet 结果对象ResultSet resultSet = statement.executeQuery(sql);//4.处理结果集 resultSet}}
ResultSet接口
作用:封装数据库查询的结果集,对结果集进行遍历,取出每一条记录。
| ResultSet接口方法 | 说明 |
|---|---|
boolean next() |
1. 游标向下一行 1. 返回 boolean 类型,如果还有下一条记录,返回 true,否则返回 false |
xxx getXxx( String or int) |
1. 通过列名,参数是 String 类型。返回不同的类型 1. 通过列号,参数是整数,从 1 开始。返回不同的类型 |

代码示例
public class JDBCDemo04 {public static void main(String[] args) throws SQLException {//1.注册驱动 可以省略//2.获取连接String url = "jdbc:mysql://localhost:3306/db4";Connection con = DriverManager.getConnection(url, "root", "123456");//3.获取 Statement对象Statement statement = con.createStatement();String sql = "select * from jdbc_user";//执行查询操作,返回的是一个 ResultSet 结果对象ResultSet resultSet = statement.executeQuery(sql);//4.处理结果集//next 方法判断是否还有下一条数据//boolean next = resultSet.next(); System.out.println(next);//getXXX 方法获取数据 两种方式//int id = resultSet.getInt("id");//列名//System.out.println(id);//int anInt = resultSet.getInt(1);//列号//System.out.println(anInt);//使用while循环while(resultSet.next()){//获取idint id = resultSet.getInt("id");//获取姓名String username = resultSet.getString("username");//获取生日Date birthday = resultSet.getDate("birthday");System.out.println(id + " = " +username + " : " + birthday);}//关闭连接resultSet.close();statement.close();con.close();}}
API 使用: 5.释放资源
- 需要释放的对象:ResultSet 结果集,Statement 语句,Connection 连接
- 释放原则:先开的后关,后开的先关。ResultSet ==> Statement ==> Connection
- 放在哪个代码块中:finally 块
与IO流一样,使用后的东西都需要关闭!关闭的顺序是先开后关, 先得到的后关闭,后得到的先关闭
代码示例
public class JDBCDemo05 {public static void main(String[] args) {Connection connection = null;Statement statement = null;ResultSet resultSet = null;try {//1.注册驱动(省略)//2.获取连接String url = "jdbc:mysql://localhost:3306/db4";connection = DriverManager.getConnection(url, "root", "123456");//3.获取 Statement对象statement = connection.createStatement();String sql = "select * from jdbc_user";resultSet = statement.executeQuery(sql);} catch (SQLException e) {e.printStackTrace();} finally {/**• 开启顺序: connection ==> statement => resultSet• 关闭顺序: resultSet ==> statement ==> connection*/try {connection.close();resultSet.close();statement.close();} catch (SQLException e) {e.printStackTrace();}}}}
步骤总结
- 获取驱动(可以省略)
- 获取连接
- 获取Statement对象
- 处理结果集(只在查询时处理)
-
JDBC实现增删改查
JDBC工具类
什么时候自己创建工具类?
如果一个功能经常要用到,我们建议把这个功能做成一个工具类,可以在不同的地方重用。
“获得数据库连接”操作,将在以后的增删改查所有功能中都存在,可以封装工具类JDBCUtils。提供获取连接对象的方法,从而达到代码的重复利用。
工具类包含的内容- 可以把几个字符串定义成常量:用户名,密码,URL,驱动类
- 得到数据库的连接:getConnection()
- 关闭所有打开的资源:
代码示例
package com.lagou.utils;import java.sql.*;/*** JDBC工具类*/public class JDBCUtils {//1. 将连接信息定义为 字符串常量public static final String DRIVERNAME = "com.mysql.jdbc.Driver";public static final String URL = "jdbc:mysql://localhost:3306/db4?characterEncoding=UTF-8";public static final String USER = "root";public static final String PASSWORD = "123456";//2.静态代码块static{try {//1.注册驱动Class.forName(DRIVERNAME);} catch (ClassNotFoundException e) {e.printStackTrace();}}//3.获取连接的 静态方法public static Connection getConnection(){try {//获取连接对象 并返回Connection connection = DriverManager.getConnection(URL, USER, PASSWORD);return connection;} catch (SQLException e) {e.printStackTrace();return null;}}//4.关闭资源的方法public static void close(Connection con, Statement statement){if(con != null && statement != null){try {statement.close();con.close();} catch (SQLException e) {e.printStackTrace();}}}public static void close(Connection con, Statement statement, ResultSet resultSet){if(con != null && statement != null){try {resultSet.close();statement.close();con.close();} catch (SQLException e) {e.printStackTrace();}}}}
DML操作
插入记录
解决插入中文乱码问题.
jdbc:mysql://localhost:3306/db4?characterEncoding=UTF-8 characterEncoding=UTF-8 指定字符的编码、解码格式。
代码示例
package com.lagou.jdbc02;import com.lagou.utils.JDBCUtils;import org.junit.Test;import java.sql.Connection;import java.sql.SQLException;import java.sql.Statement;public class TestDML {/** 插入数据* */@Testpublic void testInsert() throws SQLException {//1.通过JDBCUtils工具类 获取连接Connection con = JDBCUtils.getConnection();//2.获取Statement对象Statement statement = con.createStatement();//2.1 编写SQLString sql = "insert into jdbc_user values(null,'张百万','123','2020/11/11')";//2.2 执行SQLint i = statement.executeUpdate(sql);System.out.println(i);//3.关闭流JDBCUtils.close(con,statement);}/** 更新操作 根据id修改用户名* */@Testpublic void testUpdate() throws SQLException {Connection connection = JDBCUtils.getConnection();Statement statement = connection.createStatement();String sql = "update jdbc_user set username = '刘能' where id = 1";statement.executeUpdate(sql);JDBCUtils.close(connection,statement);}/** 删除操作* 删除 id为 1 和 2 的数据** */@Testpublic void testDelete() throws SQLException {Connection connection = JDBCUtils.getConnection();Statement statement = connection.createStatement();String sql = "delete from jdbc_user where id in(1,2)";statement.executeUpdate(sql);JDBCUtils.close(connection,statement);}}
/** 插入数据* */@Testpublic void testInsert() throws SQLException {//1.通过JDBCUtils工具类 获取连接Connection con = JDBCUtils.getConnection();//2.获取Statement对象Statement statement = con.createStatement();//2.1 编写SQLString sql = "insert into jdbc_user values(null,'张百万','123','2020/11/11')";//2.2 执行SQLint i = statement.executeUpdate(sql);System.out.println(i);//3.关闭流JDBCUtils.close(con,statement);}
更新记录
根据ID 需改用户名称
/** 更新操作 根据id修改用户名* */@Testpublic void testUpdate() throws SQLException {Connection connection = JDBCUtils.getConnection();Statement statement = connection.createStatement();String sql = "update jdbc_user set username = '刘能' where id = 1";statement.executeUpdate(sql);JDBCUtils.close(connection,statement);}
删除记录
删除id为 3 和 4 的记录
/** 删除操作* 删除 id为 1 和 2 的数据** */@Testpublic void testDelete() throws SQLException {Connection connection = JDBCUtils.getConnection();Statement statement = connection.createStatement();String sql = "delete from jdbc_user where id in(1,2)";statement.executeUpdate(sql);JDBCUtils.close(connection,statement);}
DQL操作
3.3.1 查询姓名为张百万的一条记录
package com.lagou.jdbc02;import com.lagou.utils.JDBCUtils;import java.sql.*;public class TestDQL {// 查询姓名为张百万的一条记录public static void main(String[] args) throws SQLException {//1.获取连接Connection connection = JDBCUtils.getConnection();//2.创建Statement对象Statement statement = connection.createStatement();//3. 编写SQLString sql = "select * from jdbc_user where username = '张百万'";ResultSet resultSet = statement.executeQuery(sql);//4.处理结果集while(resultSet.next()){// 通过列名的方式获取int id = resultSet.getInt("id");String username = resultSet.getString("username");String password = resultSet.getString("password");Date birthday = resultSet.getDate("birthday");System.out.println(id + " : " + username + " : " + password + " : " + birthday );}//5.释放资源JDBCUtils.close(connection,statement,resultSet);}}
SQL注入问题
Sql注入演示
向jdbc_user表中 插入两条数据
# 插入2条数据INSERT INTO jdbc_user VALUES(NULL,'jack','123456','2020/2/24');INSERT INTO jdbc_user VALUES(NULL,'tom','123456','2020/2/24');
SQL注入演示
# SQL注入演示-- 填写一个错误的密码SELECT * FROM jdbc_user WHERE username = 'tom' AND PASSWORD = '123' OR '1' = '1';
如果这是一个登陆操作,那么用户就登陆成功了.显然这不是我们想要看到的结果
sql注入案例:用户登陆
需求
用户在控制台上输入用户名和密码, 然后使用 Statement 字符串拼接的方式 实现用户的登录。
步骤- 得到用户从控制台上输入的用户名和密码来查询数据库
- 写一个登录的方法
- 通过工具类得到连接
- 创建语句对象,使用拼接字符串的方式生成 SQL 语句
- 查询数据库,如果有记录则表示登录成功,否则登录失败
- 释放资源
Sql注入方式: 123’ or ‘1’=’1
代码示例
package com.lagou.jdbc03;import com.lagou.utils.JDBCUtils;import java.sql.Connection;import java.sql.ResultSet;import java.sql.SQLException;import java.sql.Statement;import java.util.Scanner;public class TestLogin01 {/*** 用户登录案例* @param args*/public static void main(String[] args) throws SQLException {//1.获取连接Connection con = JDBCUtils.getConnection();//2.获取Statement对象Statement statement = con.createStatement();//3.获取用户输入的用户名和密码Scanner sc = new Scanner(System.in);System.out.println("请输入用户名: ");String name = sc.nextLine();System.out.println("请输入密码: ");String pass = sc.nextLine();//4.拼接SQL语句String sql = "select * from jdbc_user where username = '" + name + "' and password = '" + pass +"'";System.out.println(sql);//5.执行查询 获取结果集对象ResultSet resultSet = statement.executeQuery(sql);//6.处理结果集if(resultSet.next()){System.out.println("登录成功! 欢迎您: " + name);}else{System.out.println("登录失败! ");}//7.关闭流JDBCUtils.close(con,statement,resultSet);}}
问题分析
什么是SQL注入?
我们让用户输入的密码和 SQL 语句进行字符串拼接。用户输入的内容作为了 SQL 语句语法的一部分,改变了 原有SQL 真正的意义,以上问题称为 SQL 注入 .
如何实现的注入
根据用户输入的数据,拼接处的字符串
select * from jdbc_user where username = ‘abc’ and password = ‘abc’ or ‘1’=’1’
name=’abc’ and password=’abc’ 为 假 ‘1’=’1’ 真
相当于 select * from user where true=true; 查询了所有记录
如何解决
要解决 SQL 注入就不能让用户输入的密码和我们的 SQL 语句进 行简单的字符串拼接。
预处理对象
PreparedStatement 接口介绍
PreparedStatement 是 Statement 接口的子接口,继承于父接口中所有的方法。它是一个预编译的 SQL 语句对象.
预编译: 是指SQL 语句被预编译,并存储在 PreparedStatement 对象中。然后可以使用此对象多次高效地执行该语句。
PreparedStatement 特 点
因为有预先编译的功能,提高 SQL 的执行效率。可以有效的防止 SQL 注入的问题,安全性更高
获取PreparedStatement对象
通过Connection创建PreparedStatement对象
| Connection 接口中的方法 | 说明 |
|---|---|
PreparedStatement prepareStatement(String sql) |
指定预编译的 SQL 语句, SQL 语句中使用占位符 ? 创建一个语句对象 |
PreparedStatement接口常用方法
| 常用方法 | 说明 |
|---|---|
| int executeUpdate(); | 执行insert update delete语句. |
| ResultSet executeQuery(); | 执行select语句. 返回结果集对象 Resulet |
- 使用PreparedStatement的步骤
- 编写 SQL 语句,未知内容使用?占位:
“SELECT * FROM jdbc_user WHERE username=? AND password=?”;
- 获得 PreparedStatement 对象 3) 设置实际参数:setXxx( 占位符的位置, 真实的值) 4) 执行参数化 SQL 语句 5)
关闭资源
| setXxx重载方法 | 说明 |
|---|---|
| void setDouble(int parameterIndex, double x) | 将指定参数设置为给定 Java double 值。 |
| void setInt(int parameterIndex, int x) | 将指定参数设置为给定 Java int 值。 |
| void setString(int parameterIndex, String x) | 将指定参数设置为给定 Java String 值。 |
| void setObject(int parameterIndex, Object x) | 使用给定对象设置指定参数的值。 |
使用PreparedStatement完成登录案例
使用 PreparedStatement 预处理对象,可以有效的避免SQL注入
步骤:
1.获取数据库连接对象
2.编写SQL 使用? 占位符方式
3.获取预处理对象 (预编译对象会将Sql发送给数据库 进行预编译)
4.提示用户输入用户名 & 密码
5.设置实际参数:setXxx(占位符的位置, 真实的值)
6.执行查询获取结果集
7.判断是否查询到数据
8.关闭资源
package com.lagou.jdbc03;import com.lagou.utils.JDBCUtils;import java.sql.*;import java.util.Scanner;public class TestLogin02 {/** SQL注入* 用户输入的用户名和密码 与我们编写的SQL进行了拼接,用户输入的内容成为了SQL语法的一部分,* 用户会利用这里漏洞 输入一些其他的字符串,改变SQL原有的意思** 如果解决* 要解决SQL注入 就不能让用户输入的数据和我们的SQL进行直接的拼接** 预处理对象 PrepareStatement 他是 Statement接口的子接口* 使用预处理对象 他有预编译的功能,提高SQL的执行效率* 使用预处理对象 通过占位符的方式 设置参数 可以有效的防止SQL注入*** */public static void main(String[] args) throws SQLException {//1.获取连接Connection con = JDBCUtils.getConnection();//2.获取PrepareStatement 预处理对象//使用 ? 占位符的方式来设置参数String sql = "select * from jdbc_user where username = ? and password = ?";PreparedStatement ps = con.prepareStatement(sql);//3.获取用户输入的用户名和密码Scanner sc = new Scanner(System.in);System.out.println("请输入用户名: ");String name = sc.nextLine();System.out.println("请输入密码: ");String pass = sc.nextLine();//4.设置参数 使用setXXX(占位符的位置(整数),要设置的值)的方法设置占位符的参数ps.setString(1,name); // 设置第一个问号值 为 nameps.setString(2,pass);//5.执行查询ResultSet resultSet = ps.executeQuery();//6.处理结果集//6.处理结果集if(resultSet.next()){System.out.println("登录成功! 欢迎您: " + name);}else{System.out.println("登录失败! ");}//7.关闭流JDBCUtils.close(con,ps,resultSet);}}
PreparedStatement的执行原理
分别使用 Statement对象 和 PreparedStatement对象进行插入操作代码示例
package com.lagou.jdbc03;import com.lagou.utils.JDBCUtils;import java.sql.Connection;import java.sql.PreparedStatement;import java.sql.SQLException;import java.sql.Statement;public class TestPS {public static void main(String[] args) throws SQLException {Connection connection = JDBCUtils.getConnection();//获取StatementStatement statement = connection.createStatement();//向数据库插入两条数据statement.executeUpdate("insert into jdbc_user values(null,'张三','123456','2000/12/26')");statement.executeUpdate("insert into jdbc_user values(null,'李四','654321','1900/12/26')");//获取预处理对象PreparedStatement ps = connection.prepareStatement("insert into jdbc_user values(?,?,?,?)");//先插入第一条数据ps.setObject(1,null);ps.setString(2,"小斌");ps.setString(3,"qwer");ps.setString(4,"1999/11/11");//执行插入ps.executeUpdate();//插入第二条数据ps.setObject(1,null);ps.setString(2,"长海");ps.setString(3,"asdf");ps.setString(4,"2000/11/11");//执行插入ps.executeUpdate();//释放资源statement.close();ps.close();connection.close();}}

Statement 与 PreparedStatement的区别?
- Statement用于执行静态SQL语句,在执行时,必须指定一个事先准备好的SQL语句。
- PrepareStatement是预编译的SQL语句对象,语句中可以包含动态参数“?”,在执行时可以为“?”动态设置参数值。
- PrepareStatement可以减少编译次数提高数据库性能。
JDBC 控制事务
之前我们是使用 MySQL 的命令来操作事务。接下来我们使用 JDBC 来操作银行转账的事务。数据准备
-- 创建账户表CREATE TABLE account(-- 主 键id INT PRIMARY KEY AUTO_INCREMENT,-- 姓 名NAME VARCHAR(10),-- 转账金额money DOUBLE);-- 添加两个用户INSERT INTO account (NAME, money) VALUES ('tom', 1000), ('jack', 1000);
事务相关API
我们使用 Connection中的方法实现事务管理
| 方法 | 说明 |
|---|---|
| void setAutoCommit(boolean autoCommit) | 参数是 true 或 false 如果设置为 false,表示关闭自动提交,相当于开启事务 |
| void commit() | 提交事务 |
| void rollback() | 回滚事务 |
开发步骤
- 获取连接
- 开启事务
- 获取到 PreparedStatement , 执行两次更新操作
- 正常情况下提交事务
- 出现异常回滚事务
- 最后关闭资源
代码示例
```java package com.lagou.jdbc04;
import com.lagou.utils.JDBCUtils;
import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.SQLException;
public class TestJDBCTransaction {
//使用JDBC操作事务public static void main(String[] args) {Connection con = null;PreparedStatement ps = null;try {//1.获取连接con = JDBCUtils.getConnection();//2.开启事务con.setAutoCommit(false); //手动提交事务//3.获取预处理对象 执行SQL (两次修改操作)//3.1 tom账户 - 500ps = con.prepareStatement("update account set money = money - ? where name = ?");ps.setDouble(1,500.0);ps.setString(2,"tom");ps.executeUpdate();//模拟 tom转账之后出现异常System.out.println(1 / 0);//3.2 jack账户 + 500ps = con.prepareStatement("update account set money = money + ? where name = ?");ps.setDouble(1,500.0);ps.setString(2,"jack");ps.executeUpdate();//4.提交事务 (正常情况)con.commit();System.out.println("转账成功! !");} catch (SQLException e) {e.printStackTrace();//5.出现异常就回滚事务try {con.rollback();} catch (SQLException ex) {ex.printStackTrace();}} finally {//6.释放资源JDBCUtils.close(con,ps);}}
} ```
