MySQL详解

这篇博客全面讲解了MySQL,从初识到高级应用,涵盖了数据类型、数据库操作、DML/DQL语言、查询优化、事务处理、索引、权限管理、备份以及JDBC的使用。特别强调了DQL查询数据的重要性,包括子查询、分组过滤,并讨论了数据库设计的规范和性能考虑。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

初识MySQL

image-20220111153213798

操作数据库

操作数据库

--创建数据库
CREATE DATABASE IF NOT EXISTS westos
--删除数据库
DROP DATABASE IF EXISTS westos
--使用数据库。tab键上面的·如果表格或字段名是特殊字符,需要带·
USE `school`

SELECT `user` FROM student
--查看 数据库
SHOW DATABASE

数据库类型

数值:

tinyint 1字节

smallint 2个字节

mediumint 3个字节

int 4个字节

bigint 8个字节

float 4个字节

double 8个字节

decimal 字符串形式的浮点数 金融计算一般使用decimal

字符串:

char 0~255

varchar 0~65535

tinytest 2^8-1

test 2^16-1

时间日期

java.util.Date

date YYYY-MM-DD,日期格式

time HH: mm: ss 时间格式

datetime YYYY-MM-DD HH: mm: ss 最常用的时间格式

timestamp 时间戳, 1970.1.1到现在的毫秒数,最为常用

year 年份表示

null

没有值,未知

注意,不要使用null进行运算,结果为null

数据库的字段属性

image-20220111195657529 image-20220111195748588

创建数据库表

image-20220111201314126

image-20220111201641854

数据表的类型

image-20220111202202199

image-20220111202701517

image-20220111202827896

修改和删除

image-20220111211259534

image-20220111211458082

MySQL数据管理

外键

image-20220112141034152

image-20220112141527673

删除有外键关系的表的时候,必须要先删除引用别人的表(从表),再删除被引用的表

以上的操作都是物理外键,数据库级别的外键,不建议使用(避免数据库过多造成困扰)

image-20220112142051672

DML语言

添加

注意事项:

1.字段之间用英文逗号隔开

2.字段可以省略,但后面的值必须一一对应

3.可以同时插入多条数据,values后面的值需要使用“,”隔开即可

修改

image-20220112144736340

image-20220112145608911

image-20220112145423588

删除

image-20220112152010301 image-20220112152518900 image-20220112152733501

DQL查询数据(最重点)

DQL

image-20220112205600030

查询指定字段

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-CIuR6Ylg-1642168282400)(https://s2.loli.net/2022/01/12/og3IG5ranBjEdHJ.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-xRITkzjq-1642168282401)(https://s2.loli.net/2022/01/12/6d37XsMzoBqDpCb.png)]

where

image-20220112155057610

image-20220112170205829

image-20220112170231426

image-20220112170306342

联表查询

image-20220112180311415

image-20220112180326990

image-20220112180455892

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-sRlc11ct-1642168282404)(https://s2.loli.net/2022/01/12/abezUgiMYAvwSK5.png)]

image-20220112180549115

排序

image-20220112202058405

分页

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-14671FcT-1642168282404)(https://s2.loli.net/2022/01/12/MzxVlUgwuFDAyX4.png)]

image-20220112202225866 image-20220112202258069

子查询

本质:在where中嵌套子查询语句

分组和过滤

image-20220113144118489

MySQL函数

常用函数

image-20220112211454423

image-20220112211534915

image-20220112211622562

image-20220112211654100

聚合函数

image-20220113144345800

数据库级别的MD5加密

增强算法复杂度,不可逆

image-20220113145322260

事务

ACID原则:原子性、一致性、隔离性、持久性

隔离所导致的问题:脏读、不可重复读、幻读(虚读)

测试事务实现转账

image-20220113151932460

image-20220113152710468

索引

索引的分类

image-20220113154040456

测试索引

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-xXTisCQL-1642168282408)(https://s2.loli.net/2022/01/13/L2VZtgE6inqyTzc.png)]

image-20220113155447125

索引原则

不要对进程变动数据加索引

小数据量的表不需要加索引

索引一般加在常用来查询的字段上

http://blog.codinglabs.org/articles/theory-of-mysql-index.html

权限管理和备份

用户管理

image-20220113164630069

image-20220113165652569

MySQL备份

为什么要备份:

保证重要数据不丢失

数据转移

MySQL数据库备份方式:

直接拷贝物理文件

在可视化工具中手动导出

使用命令行导出 mysqldump 命令行使用

image-20220113185650406

把数据库给别人:给sql文件即可

规范数据库设计

为什么需要设计

image-20220113190059310

设计数据库的步骤(个人博客)

image-20220113194458924

三大范式

为什么需要数据规范化

信息重复,更新异常,插入异常:无法正常显示信息,删除异常:丢失有效信息

三大范式

第一范式(1NF):要求数据库表的每一列都是不可分割的原子数据项。

第二范式(2NF):在1NF的基础上,非码属性必须完全依赖于候选码(在1NF基础上消除非主属性对主码的部分函数依赖)

第二范式需要确保数据库表中的每一列都和主键相关,而不能只与主键的某一部分相关(主要针对联合主键而言)。

第三范式(3NF):在2NF基础上,任何非主属性不依赖于其它非主属性(在2NF基础上消除传递依赖)

第三范式需要确保数据表中的每一列数据都和主键直接相关,而不能间接相关

规范性和性能的问题

image-20220113201528349

JDBC

数据库驱动

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-nD69SGdL-1642168282411)(https://s2.loli.net/2022/01/14/gAzwQvV6GriyHOq.png)]

JDBC

SUN公司为了简化开发人员的(对数据库的统一)操作,提供了一个(java操作数据库的)规范,JDBC

这些规范的实现由具体厂商去做

对于开发人员只需要掌握JDBC接口的操作即可

image-20220114100758506

java.sql

javax.sql

还需导入一个包mysql-connector-java-5.1.47.jar

第一个JDBC程序

创建测试数据库程序

1.创建一个普通项目

2.导入数据库驱动

3.编写测试代码

import java.sql.*;

//我的第一个jdbc程序
public class JdbcFirstDemo {
    public static void main(String[] args) throws ClassNotFoundException, SQLException {
        //1、加载驱动
        Class.forName("com.mysql.jdbc.Driver");//固定写法

        //2、用户信息和URL
        //useUnicode=true&characterEncoding=utf8&useSSL=true
        String url = "jdbc:mysql://localhost:3306/jdbcstudy?useUnicode=true&characterEncoding=utf8&useSSL=true";
        String username = "root";
        String password = "123456";
        //3、连接成功
        Connection connection = DriverManager.getConnection(url, username, password);
        //4、执行sql对象
        Statement statement = connection.createStatement();
        //5、执行sql对象去执行sql,可能存在结果,查看结果
        String sql = "select * from users";
        ResultSet resultSet = statement.executeQuery(sql);//返回的结果集,封装了全部查询结果
        while(resultSet.next()){
            System.out.println("id="+resultSet.getObject("id"));
            System.out.println("name="+resultSet.getObject("name"));
            System.out.println("psw="+resultSet.getObject("password"));
            System.out.println("email="+resultSet.getObject("email"));
            System.out.println("birth="+resultSet.getObject("birthday"));
            System.out.println("================================");
        }
        //6、释放连接
        resultSet.close();
        statement.close();
        connection.close();
    }
}

步骤总结:

1、加载驱动

2、用户信息

3、连接数据库DriverManager

4、获取执行sql对象statement

5、获得返回结果集

6、释放连接

image-20220114111821934

image-20220114145134562

image-20220114145428530

statement对象详解

image-20220114145725392

image-20220114145740246

image-20220114145817344

image-20220114145833558

代码实现

1、提取工具类

import java.io.InputStream;
import java.sql.*;
import java.util.Properties;

public class JdbcUtils {
    private  static String driver = null;
    private  static String url =null;
    private  static String username = null;
    private  static String password = null;
    static {
        try {
            InputStream in = JdbcUtils.class.getClassLoader().getResourceAsStream("db.properties");
            Properties properties = new Properties();
            properties.load(in);
            driver = properties.getProperty("driver");
            url = properties.getProperty("url");
            username = properties.getProperty("username");
            password = properties.getProperty("password");
            //1、驱动只需要加载一次
            Class.forName(driver);

        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    //获取连接
    public  static Connection getConnection() throws SQLException {
        return DriverManager.getConnection(url, username, password);
    }
    //释放连接资源
    public static void release(Connection conn, Statement st, ResultSet rs){
        if(conn!=null){
            try {
                conn.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        if(conn!=null){
            try {
                conn.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        if(st!=null){
            try {
                st.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        if(rs!=null){
            try {
                rs.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }

    }
}

2、编写增删查改的方法

db.properties

driver=com.mysql.jdbc.Driver
url=jdbc:mysql://localhost:3306/jdbcstudy?useUnicode=true&characterEncoding=utf8&useSSL=true
username=root
password=123456

TestInsert

import com.cc.lesson02.utils.JdbcUtils;

import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;

public class TestInsert {
    public static void main(String[] args) {
        Connection conn = null;
        Statement st = null;
        ResultSet rs = null;
        try {
            conn = JdbcUtils.getConnection();
            st = conn.createStatement();
            String sql = "insert into users(id,`name`,password,email,birthday)"+
                    "values(5,'kuangshen','123456','24736743@qq.com','2020-01-01')";
            int i = st.executeUpdate(sql);
            if(i>0){
                System.out.println("插入成功");
            }
        } catch (SQLException e) {
            e.printStackTrace();
        }finally {
            JdbcUtils.release(conn,st,rs);
        }
    }
}

TestSelect

import com.cc.lesson02.utils.JdbcUtils;

import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;

public class TestSelect {
    public static void main(String[] args) {
        Connection conn = null;
        Statement st = null;
        ResultSet rs = null;
        try {
            conn = JdbcUtils.getConnection();
            st = conn.createStatement();
            String sql = "select * from users where id = 1";
            rs = st.executeQuery(sql);

            while (rs.next()){
                System.out.println(rs.getString("name"));
            }
        } catch (SQLException e) {
            e.printStackTrace();
        }finally {
            JdbcUtils.release(conn,st,rs);
        }
    }
}

sql注入问题

sql存在漏洞,会被攻击导致数据泄露,SQL会被拼接“or”

select * from users where name=" ’ or '1=1"

PreparedStatement对象

可以防止SQL注入,效率更高

TestInsert

import com.cc.lesson02.utils.JdbcUtils;

import java.sql.*;

public class TestInsert {
    public static void main(String[] args) {
        Connection conn = null;
        PreparedStatement st = null;
        ResultSet rs = null;
        try {
            conn = JdbcUtils.getConnection();

            //区别 使用?占位符
			String sql = "insert into users(id,`name`,password,email,birthday)"+
                    "values(?,?,?,?,?)";
            st = conn.prepareStatement(sql);//预编译SQL,先写SQL,然后不执行
            //手动给参数赋值
            st.setInt(1,6);
            st.setString(2,"qinjiang");
            st.setString(3,"123456");
            st.setString(4,"24734673@qq.com");
            //注意点 sql.Date 数据库 java.sql.Date()
            //       util.Date Java  java.util.Date().getTime() 获得时间戳
            st.setDate(5,new Date(new java.util.Date().getTime()));

            int i = st.executeUpdate();
            if(i>0){
                System.out.println("插入成功");
            }
        } catch (SQLException e) {
            e.printStackTrace();
        }finally {
            JdbcUtils.release(conn,st,rs);
        }
    }
}

PreparedStatement防止SQL注入的本质,把传递进来的参数当字符

假设其中存在转义字符,比如说 ’ ,会被直接转义掉

login(username:"lisi",password:"123456");
//login(username:"'' or 1=1",password:"'' or 1=1")
//登录业务
    public static void login(String username,String password){
    ...
        String sql = "select * from users where `name`=? and `password`=?";
}
    

使用IDEA连接数据库

先连接,再选数据库

image-20220114192536777

修改表提交

image-20220114192858913

写sql

image-20220114193253634

事务

要么都成功,要么都失败

import com.cc.lesson02.utils.JdbcUtils;

import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;

public class TestTransaction1 {
    public static void main(String[] args) {
        Connection conn = null;
        PreparedStatement st =null;
        ResultSet rs = null;
        try {
            conn = JdbcUtils.getConnection();
            //关闭数据库的自动提交
            conn.setAutoCommit(false);//开启事务


            String sql1 = "update account set money = money-100 where name='A'";
            String sql2 = "update account set money = money+100 where name='B'";
            st = conn.prepareStatement(sql1);
            st.executeUpdate();

            conn.prepareStatement(sql2);
            st = conn.prepareStatement(sql2);
            st.executeUpdate();

            //业务完毕,提交事务
            conn.commit();
            System.out.println("成功!");

        } catch (SQLException e) {
            try {
                conn.rollback();//如果失败则回滚(默认回滚,不写也行)
            } catch (SQLException ex) {
                ex.printStackTrace();
            }finally {
                JdbcUtils.release(conn,st,rs);
            }
            e.printStackTrace();
        }
    }
}

数据库连接池(DataSource接口)

image-20220114204538533

image-20220114204728983

public class JdbcUtils_DBCP {
    private static DataSource dataSource = null;

    static {
        try {
            InputStream in = JdbcUtils_DBCP.class.getClassLoader().getResourceAsStream("resources/dbcpconfig.properties");
            Properties properties = new Properties();
            properties.load(in);

            //创建数据源 工厂模式--->创建
            dataSource = BasicDataSourceFactory.createDataSource(properties);

        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    //获取连接
    public static Connection getConnection() throws SQLException {
        return dataSource.getConnection();//从数据源中获取连接
    }

    //释放连接资源
    public static void release(Connection conn, Statement st, ResultSet rs) {
        if (conn != null) {
            try {
                conn.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        if (conn != null) {
            try {
                conn.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        if (st != null) {
            try {
                st.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        if (rs != null) {
            try {
                rs.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
    }
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-4dyWgJf0-1642168282414)(https://s2.loli.net/2022/01/14/M3L4w1tYPWgpmJV.png)]

public class JdbcUtils_C3P0 {
    private static DataSource dataSource = null;

    static {
        try {

            //xml不需要读文件,直接加载
            //创建数据源 工厂模式--->创建
            dataSource = new ComboPooledDataSource();//有配置文件的写法,可传入参数选择自定义配置
            /*
            也可不用配置文件,代码版配置:
            private static ComboPooledDataSource dataSource = null;
            ...
            dataSource = new ComboPooledDataSource();
            dataSource.setDriverClass();
            dataSource.setUser();
            dataSource.setPassword();
            dataSource.setJdbcUrl();
            dataSource.setMaxPoolSize();
            dataSource.setMinPoolSize();
            */

        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    //获取连接
    public static Connection getConnection() throws SQLException {
        return dataSource.getConnection();//从数据源中获取连接
    }

    //释放连接资源
    public static void release(Connection conn, Statement st, ResultSet rs) {
        if (conn != null) {
            try {
                conn.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        if (conn != null) {
            try {
                conn.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        if (st != null) {
            try {
                st.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        if (rs != null) {
            try {
                rs.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
    }
}
public class TestC3P0 {
    public static void main(String[] args) {
        Connection conn = null;
        PreparedStatement st = null;

        try {
            conn = JdbcUtils_C3P0.getConnection();//原来是自己实现的,现在用别人的
            
            ......
                
        finally {
            JdbcUtils_C3P0.release(conn,st,null);
        }

结论

无论使用什么数据源,本质还是一样,DataSource接口不会变,方法就不会变(只是实现不同),调用方法就不变。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值