小黑子—JavaWeb:第一章 - JDBC

news2024/11/17 10:34:05

JavaWeb入门1.0

  • 1. javaweb介绍
  • 2. 数据库设计
    • 2.1 约束
    • 2.2 表关系
    • 2.3 多表查询
      • 2.3.1 内连接(连接查询)
      • 2.3.2 外连接(连接查询)
      • 2.3.3 子查询
    • 2.4 事务
  • 3. JDBC
    • 3.1 JDBC 快速入门
  • 4 JDBC API详解
    • 4.1 DriverManager
    • 4.2 Conncetion
    • 4.3 Statement
    • 4.4 ResultSet
    • 4.5 PreparedStatement
      • 4.5.1 SQL注入演示
      • 4.5.2 解决注入问题用法介绍
      • 4.5.3 原理
  • 5. 数据库连接池
    • 5.1 数据库连接池实现
    • 5.2 Driud使用步骤
  • 6. JDBC 练习
    • 6.1 准备环境
    • 6.2 查询所有
    • 6.3 增删改

1. javaweb介绍

什么是JavaWeb ?

  • Web:全球广域网,也称为万维网(www),能够通过浏览器访问的网站
  • JavaWeb:是用Java技术来解决相关web互联网领域的技术栈

在这里插入图片描述

  1. 网页:展现数据
  2. 数据库:存储和管理数据
  3. JavaWeb程序:逻辑处理

在这里插入图片描述

2. 数据库设计

  1. 软件的研发步骤
    在这里插入图片描述

  2. 数据库设计概念

  • 数据库设计就是根据业务系统的具体需求,结合我们所选用的DBMS,为这个业务系统构造出最优的数据存储模型
  • 建立数据库中的表结构以及表与表之间的关联关系的过程
  • 有哪些表?表里有哪些字段?表和表之间有什么关系?
  1. 数据库设计的步骤
    ① 需求分析(数据是什么?数据具有哪些属性?数据与属性的特点是什么)
    ② 逻辑分析(通过ER图对数据库进行逻辑建模,不需要考虑我们所选用的数据库管理系统)
    ③ 物理设计(根据数据库自身的特点把逻辑设计转换为物理设计)
    ④ 维护设计(1.对新的需求进行建表;2.表优化)
    在这里插入图片描述
    案例分析:
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述

2.1 约束

  1. 约束的概念
    约束是作用与表中列上的规则,用于限制加入表的数据
    约束的存在保证了数据库中数据的正确性、有效性和完整性
    在这里插入图片描述

  2. 约束的分类

约束名称描述关键字
非空约束保证列中所有数据不能有null值not null
唯一约束·保证列中所有数据各不相同unique
主键约束主键是一行数据的唯一标识,要求非空且唯一primary key
检查约束保证列中的值满足某一条件check
默认约束保持数据时,未指定值采用默认值default
外键约束外键用来让两个表的数据之间建立链接,保证数据的一致性和完整性foreign key

外键约束:
在这里插入图片描述
语法:
(1)添加约束

-- 创建表时添加外键约束
create table 表名(
	列名 数据类型,
	...
	[constraint] [外键名称] foreign key(外键列名) references 主表(主表列名)
);

例如:
在这里插入图片描述

(2)删除约束

alter table 表名 drop foreign key 外键名称;

2.2 表关系

  • 一对一:
    如:用户和用户信息
    在这里插入图片描述
    一对一关系多用于表拆分,将一个实体中经常使用的字段放一张表,不经常使用的字段放另一张表,用于提升查询性能
    在这里插入图片描述
    分开:
    在这里插入图片描述
    实现方式:在任意一方加入外键,关联另一方主键,并且设置外键为唯一(unique)
    在这里插入图片描述

  • 一对多(多对一):
    如:部门和员工
    一个部门对应多个员工,一个员工对应多个部门
    在这里插入图片描述

实现方式:在多的一方建立外键,指向另一方的主键
在这里插入图片描述

  • 多对多:
    如:商品和订单
    一个商品对应多个订单,一个订单包含多个商品
    在这里插入图片描述

实现方式:建立第三方中间表,中间表至少包含两个外键,分别关联两方主键
在这里插入图片描述

例如:

在这里插入图片描述

alter table tb_order_goods add constraint fk_order_id foreign key(oreder_id) references tb_order(id);
alter table tb_order_goods add constraint fk_goods_id foreign key(goods_id) references tb_goods(id);

在这里插入图片描述
小结:
在这里插入图片描述

2.3 多表查询

笛卡儿积:取A,B集合所有组合情况

多表查询:从多张表查询数据

2.3.1 内连接(连接查询)

内连接:相当于A、B交集数据
在这里插入图片描述

语法:

-- 隐式内连接
select 字段列表 from1,表2... where 条件

-- 显示内连接
select 字段列表 from1 [inner] join2 on 条件;

2.3.2 外连接(连接查询)

外连接:

  • 左外连接:相当于查询A表所有数据和交集部分数据
  • 右外连接:相当于查询B表所有数据和交集部分数据

语法:

-- 左外连接
select 字段列表 from1 乐翻天 [outer] join2 on 条件;

-- 右外列表
select 字段列表 from1 right [outer] join2 on 条件;

2.3.3 子查询

子查询概念:

  • 查询中嵌套查询,称嵌套查询为子查询
    例如:
    在这里插入图片描述

子查询根据查询结果不同,作用不同:

  • 单行单列:作为条件值,使用 = != > < 等进行条件判断
select 字段列表 fromwhere 字段名 = (子查询);
  • 多行单列:作为条件值,使用 in 等关键字进行条件判断
select 字段列表 fromwhere 字段名 in (子查询);

在这里插入图片描述

  • 多行多列:作为虚拟表
select 字段列表 from (子查询) where 条件;

在这里插入图片描述

2.4 事务

事务简介:

  • 数据库的事务(transaction)是一种机制、一个操作序列,包含了一组数据库操作命令
  • 事务把所有的命令作为一个整体一起系统提交或撤销操作请求,即这一组数据库命令 要么同时成功,要么同时失败
  • 事务是一个不可分割的工具逻辑单元
    在这里插入图片描述
-- 开启事务
start transaction;
或者 begin;

提交代码

-- 提交事务
commit;

-- 回滚事务
rollback;

事务四大特征:

  • 原子性(Atomicity):事务是不可分割的最小操作单位,要么同时成功,要么同时失败
  • 一致性(Consistency):事务完成时,必须使所有的数据都保持一致状态
  • 隔离性(lsolation) :多个事务之间,操作的可见性
  • 持久性(Durability):事务一旦提交或回滚,它对数据库中的数据的改变就是永久的

MySQL事务默认自动提交的

-- 查看事务的默认提交方式
select @@autocommit;

-- 1 自动提交 0 手动提交
-- 修改事务提交方式
set @@autocommit = 0;

3. JDBC

JDBC概念:(Java DataBase Connectivity)Java数据库连接
JDBC就是使用Java语音操作关系型数据库的一套API

在这里插入图片描述
在这里插入图片描述
JDBC本质:

  • 官方(sun公司)定义的一套操作所有关系型数据库的规则,即接口
  • 各个数据库厂商去实现这套接口,提供数据库驱动jar包
  • 我们可以使用这套接口(JDBC)编程,真正执行的代码是驱动jar包
    中的实现类

JDBC好处:

  • 各数据库厂商使用相同的接口,Java代码不需要针对不同数据库分别开发
  • 可随时替换底层数据库,访问数据库的Java代码基本不变

3.1 JDBC 快速入门

在这里插入图片描述
首先创建工程,导入驱动jar包mysql-connector-java-8.0.30.jar(版本不一样)
IDEA将其导入模块库,具体查看步骤
在这里插入图片描述

  1. 注册驱动
Class.forName("com.mysql.jdbc.Driver");
  1. 获取连接
Connection conn = DriverManager.getConnection(url,username,password); 
  1. 定义SQL语句
String sql = "update...";
  1. 获取执行SQL语句
Statement stmt = conn.createStatement();
  1. 执行SQL
stmt.executeUpdate(sql);
  1. 处理返回结果
  2. 释放资源

案例:

public class JDBCDemo {
    public static void main(String[] args) throws Exception {
        // 1. 注册成功
        Class.forName("com.mysql.jdbc.Driver");
        //2.获取连接使用自己数据库的用户和密码
        String url = "jdbc:mysql://127.0.0.1:3306/itmagua";
        String username = "root";
        String password = "root";
        Connection conn = DriverManager.getConnection(url,username,password);

        //3.定义sql
        String sql = "update student set no =200105 where id = 1;";

        //4.执行sql的对象Statement
        Statement stmt =conn.createStatement();

        //5.执行sql
        int count = stmt.executeUpdate(sql);//返回受影响的行数
        //6.处理结果
        System.out.println(count);


        //7.释放资源
        stmt.close();
        conn.close();
    }
}

执行成功:
在这里插入图片描述
在这里插入图片描述

4 JDBC API详解

4.1 DriverManager

DriverManager(驱动管理类)作用:
1. 注册驱动

        Class.forName("com.mysql.jdbc.Driver");
  • 查看Driver类源码(不同包的版本可能不一样)
    在这里插入图片描述

提示:

  • MySQL5之后的驱动包,可以省略注册驱动的步骤,不写也不会报错
  • 自动加载jar包钟META-INF/services/java.sql.Driver文件中的驱动类

2. 获取数据库连接
在这里插入图片描述

  • 参数:
  1. url:连接路径

语法:jdbc:mysql://ip地址(域名):端口号/数据库名称?参数键值对1&参数键值对2...
示例:jdbc:mysql://127.0.0.1:3306/itmagua
细节:

  • 如果连接的是本机mysql服务器,并且mysql服务默认端口是3306,则uri可以简写为: jdbc:mysql://数据库名称?参数键值对
    在这里插入图片描述
    在这里插入图片描述
  • 配置useSSL=false参数,禁用安全连接方式,解决警告提示,mysql5.7之后可能默认没有报错了
  1. user:用户名
  2. password:密码

4.2 Conncetion

Connection(数据库连接对象)作用:
1. 获取执行SQL对象

  • 普通执行SQL对象
Statement createStatement()
  • 预编译SQL的执行SQL对象:防止SQL注入
PrepatedStatement prepareStatement(sql)
  • 执行储存过程的对象
CallableStatement prepareCall(sql)

2. 管理事务

  • MySQL事务管理

开启事务:beigin;/start transaction;
提交事务:commit;
回滚事务:rollback;

MySQL默认自动提交事务

  • JDBC事务管理:Connection接口中定义了3个对应的方法

开启事务:setAutoCommit(boolean autoCommit):true 为自动提交事务;false为手动提交事务,即为开启事务
提交事务:commit;
回滚事务:rollback;

案例:
成功时:

public class JDBCDemo2_DriverManager {
    public static void main(String[] args) throws Exception {
        // 1. 注册成功
//        Class.forName("com.mysql.jdbc.Driver");
        //2.获取连接:如果连接的是本机mysql并且端口是默认的3306 可以简化
        String url = "jdbc:mysql:///itmagua";
        String username = "root";
        String password = "root";
        Connection conn = DriverManager.getConnection(url,username,password);

        //3.定义sql
        String sql1 = "update student set no =200105 where id = 1;";
        String sql2 = "update student set no =200105 where id = 2;";
        //4.执行sql的对象Statement
        Statement stmt =conn.createStatement();

        try {
            //开启事务
            conn.setAutoCommit(false);
            //5.执行sql
            int count1 = stmt.executeUpdate(sql1);//返回受影响的行数
            int count2 = stmt.executeUpdate(sql2);
            //6.处理结果
            System.out.println(count1);
            System.out.println(count2);

            // 提交事务
            conn.commit();
        } catch (SQLException throwables) {
            //回滚事务
            conn.rollback();
            throwables.printStackTrace();
        }

        //7.释放资源
        stmt.close();
        conn.close();
    }
}

在这里插入图片描述

在这里插入图片描述
异常失败回滚原来状态:
在这里插入图片描述
在这里插入图片描述

4.3 Statement

Statement作用:执行SQL语句

int executeUpdate(sql):执行DML、DDL语句
返回值:(1)DML语句影响的行数(2)DDL语句执行喉,执行成功也可能返回0

ResultSet executeQuery(sql):执行DQL语句
返回值:ResultSet 结果集对象

将单元测试添加到导包当中:
在这里插入图片描述
在这里插入图片描述

案例:
1、执行DML语句

public class JDBCDemo2_DriverManager {
    @Test
    public void testDML() throws SQLException {
        // 1. 注册成功
//        Class.forName("com.mysql.jdbc.Driver");
        //2.获取连接使用自己数据库的用户和密码
        String url = "jdbc:mysql://127.0.0.1:3306/itmagua";
        String username = "root";
        String password = "root";
        Connection conn = DriverManager.getConnection(url,username,password);

        //3.定义sql
        String sql = "update student set no =200105 where id = 1;";

        //4.执行sql的对象Statement
        Statement stmt =conn.createStatement();

        //5.执行sql
        int count = stmt.executeUpdate(sql);//执行完DML语句,返回受影响的行数
        //6.处理结果
//        System.out.println(count);
        if(count>0){
            System.out.println("执行成功");
        }else {
            System.out.println("执行失败");
        }

        //7.释放资源
        stmt.close();
        conn.close();
    }
}

在这里插入图片描述
2、执行DDL语句

public class JDBCDemo2_DriverManager {
    @Test
    public void testDML() throws SQLException {
        // 1. 注册成功
//        Class.forName("com.mysql.jdbc.Driver");
        //2.获取连接使用自己数据库的用户和密码
        String url = "jdbc:mysql://127.0.0.1:3306/itmagua";
        String username = "root";
        String password = "root";
        Connection conn = DriverManager.getConnection(url,username,password);

        //3.定义sql
        String sql = "drop database db2;";

        //4.执行sql的对象Statement
        Statement stmt =conn.createStatement();

        //5.执行sql
        int count = stmt.executeUpdate(sql);//执行完DML语句,返回的可能是删除成功所以为的0
        //6.处理结果
//        System.out.println(count);
//        if(count>0){
//            System.out.println("执行成功");
//        }else {
//            System.out.println("执行失败");
//        }
        System.out.println(count);

        //7.释放资源
        stmt.close();
        conn.close();
    }
}

在这里插入图片描述

4.4 ResultSet

ResultSet(结果集对象)作用:

  1. 封装了DQL查询语句的结果

ResultSet stmt.executeQuery(sql):执行DQL语句
返回值:ResultSet 结果集对象

  1. 获取查询结果

boolean next():(1)将光标从当前位置向前移动一行(2)判断当前行是否为有效行
返回值:

  • true:有效行,当前行有数据
  • false:无效行,当前行没有数据

xxx getXxx(参数):获取数据
xxx:数据类型;如:int getInt(参数);String getString(参数)
参数:

  • int:列的编号,从1开始
  • String:列的名称

使用步骤:

  1. 游标向下移动一行,并判断该行是否有数据:next()
  2. 获取数据:getXxx(参数)
// 循环判断游标是否是最后一行末尾
while(rs.next()){
//获取数据
rs.getXxx(参数);
}

案例:

public class JDBCDemo2_DriverManager {
    @Test
    public void testDML() throws SQLException {
        // 1. 注册成功
//        Class.forName("com.mysql.jdbc.Driver");
        //2.获取连接使用自己数据库的用户和密码
        String url = "jdbc:mysql:///itmagua?useSSL=false";
        String username = "root";
        String password = "root";
        Connection conn = DriverManager.getConnection(url,username,password);

        //3.定义sql
        String sql = "select * from student";

        //4. 获取statement对象
        Statement stmt = conn.createStatement();

        //5.获取sql
        ResultSet rs = stmt.executeQuery(sql);

        //6.处理结果,遍历rs中的所有数据
        //6.1 光标向下移动一行,并且判断当前是否有数据
        while(rs.next()){
            //6.2 获取数据 getXxx()
            int id = rs.getInt(1);
            String name = rs.getString(2);
            String no = rs.getString(3);

            System.out.println(id);
            System.out.println(name);
            System.out.println(no);
            System.out.println("------------------------");
        }

        //7.释放资源
        rs.close();
        stmt.close();
        conn.close();
    }
}

在这里插入图片描述
练习:
在这里插入图片描述
创建Account类:

package com.itmagua.pojo;

public class Account {
    private int id;
    private String name;
    private String no;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getNo() {
        return no;
    }

    public void setNo(String no) {
        this.no = no;
    }

    @Override
    public String toString() {
        return "Account{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", no='" + no + '\'' +
                '}';
    }
}

实现类:

public class JDBCDemo2_DriverManager {
    /*
    查询account账户表数据,封装为Account对象中,并且存储到ArrayList集合中
    1.定义实体类Account
    2. 查询数据,封装到Account对象中
    3.将Account对象存入ArrayList集合中

     */
    @Test
    public void testDML2() throws SQLException {
        // 1. 注册成功
//        Class.forName("com.mysql.jdbc.Driver");
        //2.获取连接使用自己数据库的用户和密码
        String url = "jdbc:mysql:///itmagua?useSSL=false";
        String username = "root";
        String password = "root";
        Connection conn = DriverManager.getConnection(url,username,password);

        //3.定义sql
        String sql = "select * from student";

        //4. 获取statement对象
        Statement stmt = conn.createStatement();

        //5.获取sql
        ResultSet rs = stmt.executeQuery(sql);

        List<Account> list = new ArrayList<>();
        //6.处理结果,遍历rs中的所有数据
        //6.1 光标向下移动一行,并且判断当前是否有数据
        while(rs.next()){
            Account account = new Account();
            //6.2 获取数据 getXxx()
            int id = rs.getInt(1);
            String name = rs.getString(2);
            String no = rs.getString(3);

            account.setId(id);
            account.setName(name);
            account.setNo(no);

            list.add(account);
        }
        System.out.println(list);

        //7.释放资源
        rs.close();
        stmt.close();
        conn.close();
    }
}

在这里插入图片描述

4.5 PreparedStatement

4.5.1 SQL注入演示

PreparedStatement作用:

  • 预编译SQL语句并执行:预防SQL注入问题

SQL注入:

  • SQL注入是通过操作输入来修改事先定义号的SQL语句,用以达到执行代码对服务器进行攻击的方法

在这里插入图片描述
在这里插入图片描述
1.

public class JDBCDemo2_DriverManager {

    @Test
    public void testLogin() throws SQLException {
        // 1. 注册成功
//        Class.forName("com.mysql.jdbc.Driver");
        //2.获取连接使用自己数据库的用户和密码
        String url = "jdbc:mysql:///itmagua?useSSL=false";
        String username = "root";
        String password = "root";
        Connection conn = DriverManager.getConnection(url,username,password);

        String name = "zhangsan";
        String pwd = "123";

        //3.定义sql
        String sql = "select * from tb_user where username = '"+name+"' and password = '"+pwd+"'";
        //4. 获取statement对象
        Statement stmt = conn.createStatement();

        //5.获取sql
        ResultSet rs = stmt.executeQuery(sql);

        // 判断登录是否成功
        if(rs.next()){
            System.out.println("登录成功");
        }else{
            System.out.println("登录失败");
        }

        //7.释放资源
        rs.close();
        stmt.close();
        conn.close();
    }
}

在这里插入图片描述
2.

    // 演示sql注入
    @Test
    public void testLogin_Inject() throws SQLException {
        // 1. 注册成功
//        Class.forName("com.mysql.jdbc.Driver");
        //2.获取连接使用自己数据库的用户和密码
        String url = "jdbc:mysql:///itmagua?useSSL=false";
        String username = "root";
        String password = "root";
        Connection conn = DriverManager.getConnection(url,username,password);

        String name = "hfkjsfhskj";
        String pwd = "' or '1' = '1";

        //3.定义sql
        String sql = "select * from tb_user where username = '"+name+"' and password = '"+pwd+"'";
        System.out.println(sql);
        //4. 获取statement对象
        Statement stmt = conn.createStatement();

        //5.获取sql
        ResultSet rs = stmt.executeQuery(sql);

        // 判断登录是否成功
        if(rs.next()){
            System.out.println("登录成功");
        }else{
            System.out.println("登录失败");
        }

        //7.释放资源
        rs.close();
        stmt.close();
        conn.close();
    }

密码不一样却登录成功了!
在这里插入图片描述
因为:password = 空字符串了 or 1=1恒等式拼接完成之后,由于前面用户名不成立,后面的密码也不成立,故构成了false
or会执行后面为true的条件,所以直接就登录了

这就是SQL注入

4.5.2 解决注入问题用法介绍

PreparedStatement作用:

预编译SQL并执行SQL语句

  1. 获取PreparedStatement 对象
// SQL语句中的参数值,使用?占位符替代
String sql = "select * from user where username = ? and passsword = ?";

// 通过Connection对象获取,并转入对应的sql语句
PreparedStatement pstmt = conn.prepareStatement(sql);
  1. 设置参数值
PreparedStatement对象:setXxx(参数1,参数2):给?赋值

Xxx:数据类型;如setInt(参数1,参数2)
参数:

  • 参数1:?的位置编号,从1开始
  • 参数2:?的值
  1. 执行SQL
executeUpdate();/executQuery(); :不需要再传递sql

案例:
1.

public class JDBCDemo2_DriverManager {

    @Test
    public void testPreparedStatement() throws SQLException {
        // 1. 注册成功
//        Class.forName("com.mysql.jdbc.Driver");
        //2.获取连接使用自己数据库的用户和密码
        String url = "jdbc:mysql:///itmagua?useSSL=false";
        String username = "root";
        String password = "root";
        Connection conn = DriverManager.getConnection(url,username,password);

        String name = "zhangsan";
        String pwd = "123";

        //3.定义sql
        String sql = "select * from tb_user where username = ? and password = ?";
        //4. 获取pstmt对象
        PreparedStatement pstmt = conn.prepareStatement(sql);
        // 设置?的值
        pstmt.setString(1,name);
        pstmt.setString(2,pwd);

        //5.获取sql
        ResultSet rs = pstmt.executeQuery();//不填sql

        // 判断登录是否成功
        if(rs.next()){
            System.out.println("登录成功");
        }else{
            System.out.println("登录失败");
        }

        //7.释放资源
        rs.close();
        pstmt.close();
        conn.close();
    }

    // 演示sql注入失效
    @Test
    public void testLogin_Inject() throws SQLException {
        // 1. 注册成功
//        Class.forName("com.mysql.jdbc.Driver");
        //2.获取连接使用自己数据库的用户和密码
        String url = "jdbc:mysql:///itmagua?useSSL=false";
        String username = "root";
        String password = "root";
        Connection conn = DriverManager.getConnection(url,username,password);

        String name = "hfkjsfhskj";
        String pwd = "' or '1' = '1";

        //3.定义sql
        String sql = "select * from tb_user where username = ? and password = ?";
        //4. 获取pstmt对象
        PreparedStatement pstmt = conn.prepareStatement(sql);
        // 设置?的值
        pstmt.setString(1,name);
        pstmt.setString(2,pwd);

        //5.获取sql
        ResultSet rs = pstmt.executeQuery();

        // 判断登录是否成功
        if(rs.next()){
            System.out.println("登录成功");
        }else{
            System.out.println("登录失败");
        }

        //7.释放资源
        rs.close();
        pstmt.close();
        conn.close();
    }

}

在这里插入图片描述

4.5.3 原理

原理
PrepareStatmenet好处:

  1. 预编译SQL,性能更好
  2. 防止SQL注入:将敏感字符进行专业

① PreparedStatment预编译功能开启:useServePrepStmts=true

② 配置MySQL执行日志(重启MySQL服务后生效)

log-output=FILE
general-log=1
general_log_file="D:\mysql.log"
slow-query-log=1
slow_query_log_file="D:\mysql_slow.log"
long_query_time=2

PrepareStatment原理:

  1. 在获取PreparedStatement对象时,将sql语句发送给mysql服务器进行检查,编译(这些步骤很耗时)
  2. 执行时就不用再进行这些步骤了,速度更快
  3. 如果sql模板一样,则只需要进行一次检查、编译

5. 数据库连接池

数据库连接池是个容器,负责分配、管理数据库连接(Connection);
它允许应用程序重复使用一个现有的数据库连接,而不是再重新建立一个;
释放空闲时间最大空闲时间的数据库连接来避免因为没有释放数据库连接遗漏;

好处:

  • 资源重用
  • 提升系统响应速度
  • 避免数据库连接遗漏

在这里插入图片描述

5.1 数据库连接池实现

标准接口:DataSource

  • 官方(SUN)提供的数据库连接池标准接口,由第三方组织实现此接口
  • 功能:获取连接
Connection getConnection()

常见的数据库连接池:

  • DBCP
  • C3P0
  • Druid

Druid(德鲁伊):

  • Druid连接池是阿里巴巴开源的数据库连接池项目
  • 功能强大,性能优秀,是Java最好的数据库连接池之一

5.2 Driud使用步骤

  1. 导入jar包 druid -1.1.12.jar(版本不一样)
  2. 定义配置文件
  3. 加载配置文件
  4. 获取数据库连接池对象
  5. 获取连接

案例:在这里插入图片描述

public class DruidDemo {
    public static void main(String[] args) throws Exception {
        //1.导入jar包

        //2.定义配置文件

        //3.加载配置文件
        Properties prop = new Properties();
        prop.load(new FileInputStream("/jdbc-demo/src/druid.properties"));
        //4.获取连接池对象
        DataSource dataSource = DruidDataSourceFactory.createDataSource(prop);

        //5.获取数据库连接Connection
        Connection connection = dataSource.getConnection();

        System.out.println(connection);
    }
}

在这里插入图片描述

6. JDBC 练习

在这里插入图片描述

6.1 准备环境

在这里插入图片描述

  • 准备环境
    在这里插入图片描述
  • 包装类
public class Brand {
    // 主键
    private int id;
    // 品牌
    private String brandName;
    // 企业
    private String companyName;
    // 排序字段
    private int ordered;
    // 描述信息
    private String description;
    // 状态:0:禁用 1:启用
    private Integer status ;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getBrandName() {
        return brandName;
    }

    public void setBrandName(String brandName) {
        this.brandName = brandName;
    }

    public String getCompanyName() {
        return companyName;
    }

    public void setCompanyName(String companyName) {
        this.companyName = companyName;
    }

    public int getOrdered() {
        return ordered;
    }

    public void setOrdered(int ordered) {
        this.ordered = ordered;
    }

    public String getDescription() {
        return description;
    }

    public void setDescription(String description) {
        this.description = description;
    }

    public Integer getStatus() {
        return status;
    }

    public void setStatus(Integer status) {
        this.status = status;
    }

    @Override
    public String toString() {
        return "Brand{" +
                "id=" + id +
                ", brandName='" + brandName + '\'' +
                ", companyName='" + companyName + '\'' +
                ", ordered=" + ordered +
                ", description='" + description + '\'' +
                ", status=" + status +
                '}';
    }
}

6.2 查询所有

  1. 获取Connection
  2. 定义SQL:select * from tb_brand;
  3. 获取PreparedStatement对象
  4. 设置参数:不需要
  5. 执行SQL
  6. 处理结果:List<Brand>
  7. 释放资源
public class BrandTest {
    @Test
    public void testSelectAll() throws Exception {
        //1.获取connection
        //加载配置文件
        Properties prop = new Properties();
        prop.load(new FileInputStream("/jdbc-demo/src/druid.properties"));
        //获取连接池对象
        DataSource dataSource = DruidDataSourceFactory.createDataSource(prop);

        //获取数据库连接Connection
        Connection conn = dataSource.getConnection();

        //2.定义sql
        String sql = "select * from tb_brand;";

        //3.获取pstmt对象
        PreparedStatement pstmt = conn.prepareStatement(sql);

        //4.设置参数

        //5.执行sql
        ResultSet rs = pstmt.executeQuery();

        //6.处理结果list<Brand>封装Brand对象,转载list集合
        Brand brand = null;
        List<Brand> brands = new ArrayList<>();
        while(rs.next()){
            //获取数据
            int id = rs.getInt("id");
            String brandName = rs.getString("brand_name");
            String companyName = rs.getString("company_name");
            int ordered = rs.getInt("ordered");
            String description = rs.getString("description");
            int status = rs.getInt("status");

            //获取Brand对象
            brand = new Brand();
            brand.setId(id);
            brand.setBrandName(brandName);
            brand.setCompanyName(companyName);
            brand.setOrdered(ordered);
            brand.setDescription(description);
            brand.setStatus(status);

            //转载集合
            brands.add(brand);
        }
        System.out.println(brands);

        //7.释放资源
        rs.close();
        pstmt.close();
        conn.close();

    }
}

6.3 增删改

1、增加
在这里插入图片描述

  1. 编写SQL语句
insert into tb_brand(brand_name,company_name,ordered,description,status)values(?,?,?,?,?);
  1. 是否需要参数?需要:除了id之外的所有数据
  2. 返回结果如何封装? boolean
public class BrandTest {
    @Test
    public void testAdd() throws Exception {
        //模拟接收页面提交的参数
        String brandName = "捞的一";
        String companyName = "愣头青";
        int ordered = 1;
        String description = "皮的很不谈";
        int status = 1;

        //1.获取connection
        //加载配置文件
        Properties prop = new Properties();
        prop.load(new FileInputStream("/jdbc-demo/src/druid.properties"));
        //获取连接池对象
        DataSource dataSource = DruidDataSourceFactory.createDataSource(prop);

        //获取数据库连接Connection
        Connection conn = dataSource.getConnection();

        //2.定义sql
        String sql = "insert into tb_brand(brand_name,company_name,ordered,description,status)values(?,?,?,?,?);";

        //3.获取pstmt对象
        PreparedStatement pstmt = conn.prepareStatement(sql);

        //4.设置参数
        pstmt.setString(1,brandName);
        pstmt.setString(2,companyName);
        pstmt.setInt(3,ordered);
        pstmt.setString(4,description);
        pstmt.setInt(5,status);

        //5.执行sql
        int count = pstmt.executeUpdate();

        //6.处理结果
        System.out.println(count>0);

        //7.释放资源

        pstmt.close();
        conn.close();

    }
}

在这里插入图片描述

在这里插入图片描述

2、修改
在这里插入图片描述

  1. 编写SQL语句
update tb_brand
set brand_name = ?,
	company_name= ?,
	ordered = ?,
	description = ?,
	status = ?
where id = ?
  1. 是否需要参数?需要:Brand对象所有数据
  2. 返回结果如何封装?boolean
public class BrandTest {
    @Test
    public void testUpdat() throws Exception {
        //模拟接收页面提交的参数
        String brandName = "捞的一";
        String companyName = "愣头青";
        int ordered = 1000;
        String description = "皮的很谈";
        int status = 1;
        int id = 4;

        //1.获取connection
        //加载配置文件
        Properties prop = new Properties();
        prop.load(new FileInputStream("/jdbc-demo/src/druid.properties"));
        //获取连接池对象
        DataSource dataSource = DruidDataSourceFactory.createDataSource(prop);

        //获取数据库连接Connection
        Connection conn = dataSource.getConnection();

        //2.定义sql
        String sql = "update tb_brand\n" +
                "set brand_name = ?,\n" +
                "\tcompany_name= ?,\n" +
                "\tordered = ?,\n" +
                "\tdescription = ?,\n" +
                "\tstatus = ?\n" +
                "where id = ?";

        //3.获取pstmt对象
        PreparedStatement pstmt = conn.prepareStatement(sql);

        //4.设置参数
        pstmt.setString(1,brandName);
        pstmt.setString(2,companyName);
        pstmt.setInt(3,ordered);
        pstmt.setString(4,description);
        pstmt.setInt(5,status);
        pstmt.setInt(6,id);

        //5.执行sql
        int count = pstmt.executeUpdate();

        //6.处理结果
        System.out.println(count>0);

        //7.释放资源

        pstmt.close();
        conn.close();

    }
}

在这里插入图片描述

在这里插入图片描述

3、删除
在这里插入图片描述

  1. 编写SQL语句
delete from tb_brand where id = ?
  1. 是否需要参数?需要:id
  2. 返回结果如何封装?boolean
public class BrandTest {
    @Test
    public void testDeleteById() throws Exception {
        //模拟接收页面提交的参数
        int id = 4;

        //1.获取connection
        //加载配置文件
        Properties prop = new Properties();
        prop.load(new FileInputStream("/jdbc-demo/src/druid.properties"));
        //获取连接池对象
        DataSource dataSource = DruidDataSourceFactory.createDataSource(prop);

        //获取数据库连接Connection
        Connection conn = dataSource.getConnection();

        //2.定义sql
        String sql = "delete from tb_brand where id = ?";

        //3.获取pstmt对象
        PreparedStatement pstmt = conn.prepareStatement(sql);

        //4.设置参数
        pstmt.setInt(1,id);

        //5.执行sql
        int count = pstmt.executeUpdate();

        //6.处理结果
        System.out.println(count>0);

        //7.释放资源

        pstmt.close();
        conn.close();

    }
}

在这里插入图片描述

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

13_Linux无设备树Platform设备驱动

目录 Linux驱动的分离与分层 驱动的分隔与分离 驱动的分层 platform平台驱动模型简介 platform总线 platform驱动 platform设备 platform设备程序编写 platform驱动程序编写 测试APP编写 运行测试 Linux驱动的分离与分层 像I2C、SPI、LCD 等这些复杂外设的驱动就不…

吴恩达ML2022-用于手写数字识别的神经网络

1 用到的包 导入在这个分配过程中需要的所有包。 Numpy 是使用 Python 进行科学计算的基本软件包。Matplotlib 是在 Python 中绘制图形的流行库。tensorflow是一种流行的机器学习平台。 import numpy as np import tensorflow as tf from tensorflow.keras.models import Se…

Java对象导论

对象具有状态、行为和标识。每个对象都可以拥有内部数据&#xff08;它们给出了该对象的状态&#xff09;和方法&#xff08;它们产生的行为&#xff09;&#xff0c;并且每个对象在内存中都有一个唯一的地址&#xff08;标识&#xff09;。 抽象过程就是在问题空间元素和解空…

Macbook下提升开发效率的几个小工具

最近倒腾mac笔记本&#xff0c;记录下一些高效率的工具吧。 首先就是alfred&#xff0c;内置可以自定义各种快捷命令查找&#xff0c;配合Dash来快速查找C系统API&#xff0c;其实Dash中包含了各种编程所需API文档&#xff0c;值得下载。 以前我都是直接查看cppreference.c…

【分享】Redis的五种基本数据类型和应用场景

前言&#xff1a; Redis支持五种基本数据类型&#xff1a; String&#xff08;字符串类型&#xff09;&#xff1a;可以是普通字符串&#xff0c;也可以是整数或浮点数值。可以设置过期时间&#xff1b;可以对字符串进行append、get、set、incr、decr等操作。Hash&#xff08…

【C++】位图和布隆过滤器

文章目录 位图概念难点代码 布隆过滤器概念插入查找删除优缺点代码 位图 概念 所谓位图&#xff0c;就是用每一个比特位位来存放某种状态&#xff0c;适用于海量数据&#xff0c;数据无重复的场景。通常是用来判断某个数据存不存在的。 给40亿个不重复的无符号整数&#xff…

buu_Misc总结2

目录 百里挑一 exiftool: [SUCTF2018]followme grep工具使用&#xff1a; [安洵杯 2019]Attack mimikatz工具使用&#xff1a; 百里挑一 打开文件是流量包&#xff0c;发现里面有很多图片 导出http 另存一个文件夹&#xff0c;里面很多图片&#xff0c;啥也看不出来 &…

医用影像技术

1.X光和CT原理 X光和CT&#xff08;计算机断层扫描&#xff09;都是医学成像技术&#xff0c;用于诊断和治疗。它们的原理如下&#xff1a; X光原理&#xff1a; X光是一种电磁辐射&#xff0c;与可见光类似&#xff0c;但具有更高的能量。当X光通过人体或物体时&#xff0c;…

创作一周年纪念日【道阻且长,行则将至】

✨个人主页&#xff1a; 北 海 &#x1f389;所属专栏&#xff1a; 技术之外的往事 &#x1f383;所处时段&#xff1a; 大学生涯[1/2] 文章目录 一、起点一切皆有定数 二、成果尽心、尽力 三、相遇孤举者难起&#xff0c;众行者易趋 四、未来长风破浪会有时&#xff0c;直挂云…

[MySQL]MySQL表中数据的增删查改(CRUD)

[MySQL]MySQL表中数据的增删查改(CRUD) 文章目录 [MySQL]MySQL表中数据的增删查改(CRUD)1. 新增数据1.1 单列插入1.2 多列插入1.3 插入否则更新1.4 替换 2. 基本查询数据2.1 全列查询2.2 指定列查询2.3 查询字段为表达式2.4 为查询结果指定别名2.5 结果去重2.6 where子句2.7 or…

修复漏洞(二)离线升级Tomcat版本

前言 生产环境无法联网&#xff0c;只能通过下载离线版本更新Tomcat到小版本最新注意Tomcat10和11与jdk1.8都不兼容&#xff0c;只能更新到小版本的最新前提是按照我这种方法配置Tomcat开机自启的https://blog.csdn.net/qq_44648936/article/details/130022136 步骤 备份整个…

IAR编译报错:Error[Pe065: expected a “.“ and Error[Pe007]:unrecognized token

IAR报错 Error[Pe065: expected a “.” and Error[Pe007]:unrecognized token 使用IAR编译报如下错误&#xff1a; 找到软件报错的地方&#xff0c;从肉眼看&#xff0c;并没有错误的地方&#xff0c;如下图所示&#xff1a; 这时肯定是丈二和尚摸不着头脑&#xff0c;这里…

VMware种ubuntu22.04挂载ax88179网卡不显示的问题

网上找了很多解决办法&#xff0c;都说是驱动的问题&#xff0c;其实不是。ubuntu22自带无bug的ax88179的驱动。 其实是Vmware的问题&#xff0c;在虚拟机设置种添加一个usb控制器&#xff0c;然后这样设置就好了。

HCIP第一课实验小练习

目录 题目&#xff1a;​编辑 第一步&#xff1a;地址规划&#xff08;子网划分&#xff09; 第二步&#xff1a;设计拓扑并规划地址配置 第三步&#xff1a;VLAN规划配置 LW1 LW2 第四步&#xff1a;网关配置 第五步&#xff1a;及静态路由配置 第六步防止成环 题目&…

深入理解Linux网络——TCP连接建立过程(三次握手源码详解)

文章目录 一、相关实际问题二、深入理解listen1&#xff09;listen系统调用2&#xff09;协议栈listen3&#xff09;接收队列定义4&#xff09;接收队列申请和初始化5&#xff09;半连接队列长度计算6&#xff09;小结 三、深入理解connect1&#xff09;connect调用链展开2&…

单片机第一季:零基础6——按键

目录 1&#xff0c;独立按键 2&#xff0c;矩阵按键 &#xff08;注意&#xff1a;文章中的代码仅供参考学习&#xff0c;实际使用时要根据需要修改&#xff09; 1&#xff0c;独立按键 按键管脚两端距离长的表示默认是导通状态&#xff0c;距离短的默认是断开状态&#xf…

用PuTTY作为telnet客户端

PuTTY工具可以作为telnet客户端使用。 打开PuTTY&#xff0c;输入telnet服务器的ip地址、端口号&#xff0c;选择Other&#xff0c;然后点击Open&#xff0c;就可以了&#xff1a;

C# Modbus通信从入门到精通(2)——Modbus RTU协议原理

Modbus RTU是串行链路上的协议,也就是说Modbus RTU是通过串口通信来实现的,它可以通过RS232、RS485物理层的接口来实现,同时它也是一个主从协议,在同一时间总线上只能有一个主站和一个或多个(最多247)个从站。Modbus通信总是由主站发起,从站没有接收到主站的请求时不会发…

【Maven三】——maven生命周期和插件

系列文章目录 Maven之POM介绍 maven命令上传jar包到nexus 【Maven二】——maven仓库 maven生命周期和插件 系列文章目录前言一、什么是生命周期&why1.三套生命周期2.clean生命周期3.default生命周期4.site生命周期5.命令行与生命周期 二、插件目标三、插件绑定1.内置绑定2…

大华智慧园区综合管理平台RCE漏洞复现(0day)

0x01 产品简介 “大华智慧园区综合管理平台”是一款综合管理平台&#xff0c;具备园区运营、资源调配和智能服务等功能。平台意在协助优化园区资源分配&#xff0c;满足多元化的管理需求&#xff0c;同时通过提供智能服务&#xff0c;增强使用体验。 0x02 漏洞概述 大华智慧园…