【JDBC】Java连接MySQL数据库

news2024/12/23 11:21:10

文章目录

  • 一、数据库编程
  • 二、Java数据库编程JDBC
    • 2.1 什么是JDBC
    • 2.2 JDBC的工作原理
  • 三、JDBC基本操作
    • 3.1 JDBC API
    • 3.2 数据库连接Connection
    • 3.3 Statement对象
    • 3.4 ResultSet对象
  • 四、应用案例


一、数据库编程

数据库编程指的是通过编程语言与数据库进行交互和操作的过程,包括使用编程语言创建、连接、查询、更新和删除数据库中的数据,以及管理数据库结构和其他相关工作等。

另外,不同的数据库,对应不同的编程语言提供了不同的数据库驱动包,如:MySQL提供了Java的驱动包mysql-connector-java,如果要使用Java操作MySQL数据库就需要使用该驱动包。同样的,要基于Java操作Oracle数据库则需要Oracle的数据库对应的驱动包。

二、Java数据库编程JDBC

2.1 什么是JDBC

JDBC(Java Database Connectivity)是Java语言用于与关系型数据库进行交互的标志 API。这个 API 由java.sql以及javax.sql包中的一些类和接口组成,使Java应用程序能够通过标准化的方式连接和操作各种不同的数据库

2.2 JDBC的工作原理

JDBC 为多种关系数据库提供了统一访问方式,作为特定厂商数据库访问API的一种高级抽象,它主要包含一些通用的接口类。

JDBC访问数据库层次结构:

JDBC访问数据库的层次结构主要就是上图所示的三个层次:应用层、JDBC接口层、JDBC驱动层。

  1. 应用层就是指使用JDBC的Java应用程序的代码层。它包含了与数据库进行交互的业务逻辑和应用程序的其他组件。在应用层中,开发人员使用JDBC接口层提供的API来编写数据库连接、查询和更新等一系列操作的代码。

  2. JDBC接口层是JDBC的核心组成部分,它提供了一组标志的接口和类,用于与不同数据库进行通信。JDBC的接口层定义了一套规范,定义了各种数据库操作的 API,使得开发人员能够统一的方式和不同的数据库进行交互

  3. JDBC驱动层是实现JDBC接口层的具体数据库驱动程序。每个数据库供应商都提供了特定数据库的JDBC驱动程序,用于实现与该数据库的通信和相关操作。

    • JDBC驱动层的主要功能是将JDBC接口层的方法调用转换为数据库特定的协议和命令,以及处理与数据库的底层通信。
    • 驱动程序通过实现JDBC接口层定义的接口和类,提供了与特定数据库的交互功能。

三、JDBC基本操作

3.1 JDBC API

在Java JDBC编程中对数据库的操作均使用JDK自带的API统一处理,通常与特定数据库的驱动类是完全解耦的。所以掌握Java JDBC API (位于 java.sql 包下) 即可掌握Java数据库编程。

JDBC API的常见接口和类如下表所示:
以下是JDBC API的常见接口和类,以Markdown表格形式展示:

接口/类描述
DriverManager驱动程序管理类,用于加载数据库驱动程序并建立数据库连接
DataSource数据源接口,用于获取数据库连接
Connection表示与数据库的连接
Statement用于执行静态SQL语句
PreparedStatement用于执行预编译的SQL语句,支持参数化查询
CallableStatement用于执行数据库存储过程或函数
ResultSet表示从数据库返回的结果集
ResultSetMetaData提供有关结果集中列的信息,如列名、数据类型等
DatabaseMetaData提供有关数据库的信息,如数据库版本、支持的特性等
Savepoint用于事务中的部分回滚操作
SQLException表示与数据库操作相关的异常
Driver数据库驱动程序的接口,用于注册和创建驱动程序
Blob/Clob用于处理二进制/大文本数据
ConnectionPoolDataSource连接池数据源接口,用于连接池的管理

这些接口和类是JDBC API的核心组成部分,通过它们可以实现数据库的连接、查询、更新和事务管理等功能。根据实际需求,还可以使用这些API的具体实现类或其他辅助类来完成更复杂的数据库操作。

3.2 数据库连接Connection

在Java JDBC编程中获取数据库连接 Connection 的方式通常有两种:

  1. 通过DriverManager(驱动管理类)的静态方法获取;
  2. 通过DataSource(数据源)对象获取,实际应用中会使用DataSource对象。

1. 通过DriverManager获取连接

DriverManager是Java提供的一个用于管理不同数据库驱动程序的类。通过DriverManager,可以使用数据库驱动程序的URL、用户名和密码来获取数据库连接。这种方式是传统的连接方式,适用于简单的应用场景

下面是一个通过DriverManager获取数据库连接的示例代码:

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.SQLException;

public class Main {
    public static void main(String[] args) {
        String url = "jdbc:mysql://localhost:3306/mydatabase?CharacterEncoding=utf8&useSSL=false";
        String username = "root";
        String password = "mypassword";

        try {
            Connection conn = DriverManager.getConnection(url, username, password);
            // 使用连接执行数据库操作
            // ...
            conn.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
}

在上述代码中,使用DriverManager.getConnection()方法传入数据库URL、用户名和密码来获取数据库连接。

2. 通过DataSource对象获取连接

DataSource是Java提供的一个接口,用于获取数据库连接。DataSource提供了更多的功能和配置选项,可以管理连接池、实现连接的缓存和复用等。在实际应用中,使用DataSource对象获取连接是更常见的方式,特别是在大型应用或需要高并发访问数据库的场景中

使用DataSource获取连接的代码示例:

import com.mysql.cj.jdbc.MysqlDataSource;

import javax.sql.DataSource;
import java.sql.Connection;
import java.sql.SQLException;
public class Main {
    public static void main(String[] args) throws SQLException {
        // 使用 DataSource 体现了高内聚,有利于后面更改数据库
        DataSource dataSource = new MysqlDataSource();
        // 设置数据库所在位置,当前固定写法
        ((MysqlDataSource)dataSource).setUrl("jdbc:mysql://127.0.0.1:3306/mydatabase?characterEncoding=utf8&useSSl=false");
        ((MysqlDataSource)dataSource).setUser("root");
        ((MysqlDataSource)dataSource).setPassword("mypassword");

        /*  // 也可以这样写,但不推荐
        MysqlDataSource mysqlDataSource = new MysqlDataSource();
        mysqlDataSource.setUrl();*/

        // 和数据库建立网络连接
        Connection connection = dataSource.getConnection();

		// 后续操作
		// ...
}

上面这段代码使用了MysqlDataSource作为具体的DataSource实现类,通过该实现类获取数据库连接。

需要注意的细节:
使用DataSource对象获取数据库连接的好处是可以通过更改具体的DataSource实现类来切换不同的数据库或者调整连接配置,而无需修改大量的代码。这样做提高了代码的灵活性和可维护性,特别是在多数据库支持或者切换数据库的场景中。因此这是更推荐的获取数据库连接的方法。

使用DataSource对象获取连接相比于DriverManager的方式,有以下一些优势:

  • 连接池管理:DataSource可以实现连接池来管理数据库连接,提高连接的复用性和性能。
  • 配置灵活:DataSource提供了更多的配置选项,可以根据需求设置连接超时、最大连接数、连接验证等参数。
  • 适应多种数据源:DataSource可以适配多种数据库类型,即可以使用同一套代码连接不同的数据库,提高了代码的可移植性和扩展性。

因此,在实际应用中,使用DataSource对象获取数据库连接是更为常见和推荐的方式,特别是在复杂的应用场景中,可以更好地管理和优化数据库连接。

3.3 Statement对象

Statement对象的主要作用发送SQL语句到数据库并执行,Java API 中主要提供了三种Statement对象。

三种Statement对象的类型,它们分别是:StatementPreparedStatementCallableStatement。这些对象都是java.sql.Statement接口的实现类,用于发送SQL语句到数据库并执行。

以下是对这三种Statement对象的简要介绍:

  1. Statement
    Statement是最基本的Statement对象,用于执行静态的SQL语句。它通过执行executeQuery()方法执行查询语句,返回一个ResultSet对象,通过执行executeUpdate()方法执行更新语句,返回受影响的行数。

  2. PreparedStatement
    PreparedStatementStatement的子接口,它是预编译的Statement对象。预编译是指在执行SQL语句之前,将SQL语句发送到数据库进行预处理,以提高执行效率和安全性。使用PreparedStatement可以通过占位符(如?)来代替具体的参数值,然后使用setXxx()方法设置参数的值。这样可以避免SQL注入攻击,并且在多次执行相同的SQL语句时可以提高性能。

  3. CallableStatement
    CallableStatement也是Statement的子接口,用于执行存储过程(Stored Procedure)的SQL语句。存储过程是在数据库中预先定义好的一段可重用的代码,可以接受参数并返回结果。CallableStatement允许调用存储过程,并传递参数,然后通过执行execute()方法执行存储过程,并获取返回的结果。

这些Statement对象在执行SQL语句时有各自的优势和适用场景。PreparedStatement的预编译特性使其在重复执行相同的SQL语句时效率更高,而CallableStatement适用于执行存储过程。需要根据具体的需求选择合适的Statement对象类型来进行操作。

例如向 Student 数据库表中新增数据:使用Statement对象

// 使用 jdbc 往 数据库 中插入数据
// 提前准备数据库(test)和数据表(student)

public class JDBCInsertDemo {
    public static void main(String[] args) throws SQLException {
        // 1. 创建数据源,描述了数据库服务器在哪

        // 使用 DataSource 体现了该内聚,有利于后面更改数据库
        DataSource dataSource = new MysqlDataSource();
        // 设置数据库所在位置,当前固定写法
        ((MysqlDataSource)dataSource).setUrl("jdbc:mysql://127.0.0.1:3306/test?characterEncoding=utf8&useSSl=false");
        ((MysqlDataSource)dataSource).setUser("root");
        ((MysqlDataSource)dataSource).setPassword("passwd");

        // 2. 和数据库建立网络连接
        Connection connection = dataSource.getConnection();

        // 通过控制台,输入用户信息
        Scanner scanner = new Scanner(System.in);
        System.out.println("请输入学号:");
        int id = scanner.nextInt();
        System.out.println("请输入姓名:");
        String name = scanner.next();

        // 3. 构建一个SQL语句,完成插入操作

        // 描述sql
        String sql = "insert into student values(" + id + ", '" + name +"');"; // 存在SQL注入

        Statement statement = connection.createStatement();


        // 4. 执行SQL语句,针对增、删、改 使用executeUpdate
        //               针对查,使用executeQuery
        int ret = statement.executeUpdate(sql); // 返回值表示影响了几行

        System.out.println("影响了:" + ret + "行!");

        // 5. 断开和数据库的链接,释放必要的资源
        statement.close();
        connection.close();
    }
}

这段代码是一个使用Statement对象进行插入操作的示例。但是,需要注意的是,代码中存在SQL注入的安全风险

这里直接将用户输入的学号和姓名拼接到SQL语句中。然而,这种方式存在SQL注入的安全风险,因为用户的输入没有经过任何过滤或转义。恶意用户可能会利用这种情况来执行恶意的SQL语句。为了防止SQL注入,应该使用参数化查询或预编译语句

例如向 Student 数据库表中新增数据:使用PreparedStatement对象

为了防止SQL注入攻击,改用参数化查询或预编译语句。使用参数化查询时,可以使用占位符(如?)来代替具体的参数值,然后使用setXxx()方法设置参数的值。这样,数据库驱动程序会正确处理参数,并避免了SQL注入的风险。

下面是一个改用参数化查询的示例:

String sql = "insert into student values(?, ?)";
PreparedStatement preparedStatement = connection.prepareStatement(sql);
preparedStatement.setInt(1, id);
preparedStatement.setString(2, name);
int ret = preparedStatement.executeUpdate();

在这个示例中,使用了参数化查询,通过占位符(?)来代替具体的参数值。然后,使用setXxx()方法设置占位符的值,再执行插入操作。这样,即使用户输入包含特殊字符,也会被正确处理,避免了SQL注入的风险。

3.4 ResultSet对象

ResultSet对象是Java中用于表示SQL查询结果集的接口,它是在java.sql包中定义的。

当使用StatementPreparedStatement对象执行查询语句时,会返回一个ResultSet对象,它包含了查询结果的数据。通过ResultSet对象,可以遍历结果集的行,并获取每一行中的列数据。

以下是一些ResultSet对象的常用方法:

  1. next():将游标移动到结果集的下一行,并返回一个布尔值,表示是否还有更多的行。可以使用while循环来遍历整个结果集。

  2. getXxx(int columnIndex)getXxx(String columnLabel):这些方法用于获取当前行中指定列的值。getXxx()中的Xxx表示具体的数据类型,如getInt()getString()等。可以根据列的索引或列名来获取对应的值。

  3. getMetaData():返回一个ResultSetMetaData对象,用于获取结果集的元数据信息,如列名、列类型等。

  4. close():关闭ResultSet对象。

下面是一个简单的示例,展示了如何使用ResultSet对象遍历查询结果集:

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

public class Main {
       public static void main(String[] args) throws SQLException {
        DataSource dataSource = new MysqlDataSource();
        ((MysqlDataSource)dataSource).setUrl("jdbc:mysql://127.0.0.1:3306/test?characterEncoding=utf8&useSSL=false");
        ((MysqlDataSource)dataSource).setUser("root");
        ((MysqlDataSource)dataSource).setPassword("passwd");

        Connection connection = dataSource.getConnection();

        String sql = "select * from student;";

        PreparedStatement statement = connection.prepareStatement(sql);

        // 结果集合
        ResultSet set = statement.executeQuery();

        // next相当于移动一下光标,光标指向下一行,移动到结尾时返回false
        // 初始位置:光标指向第一行的前一行
        while (set.next()){
            // 使用 getXX 方法获取每一列,参数是数据库表的列名
            int id = set.getInt("id");
            String name = set.getString("name");

            System.out.println("id = " + id + ", name = " + name + ".");
        }

        // 释放资源
        set.close();
        statement.close();
        connection.close();
    }
}

在上述示例中,使用Statement对象执行查询语句,返回一个ResultSet对象。通过next()方法将游标移动到下一行,并使用getXxx()方法获取每一行中指定列的值。这样可以遍历整个结果集并获取其中的数据。

需要注意的是,在使用ResultSet对象遍历结果集后,应当及时关闭它以释放资源,可以使用close()方法或通过try-with-resources语句块来确保资源的正确关闭。

四、应用案例

下面是一个综合案例,展示了使用JDBC进行数据库表student的增删改查操作的示例代码:


// 下面是一个综合案例,展示了使用JDBC进行数据库增删改查操作的示例代码:

import com.mysql.cj.jdbc.MysqlDataSource;

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

public class JDBCCrudExample {
    public static void main(String[] args) throws SQLException {
        DataSource dataSource = new MysqlDataSource();
        ((MysqlDataSource) dataSource).setUrl("jdbc:mysql://127.0.0.1:3306/test?characterEncoding=utf8&useSSL=false");
        ((MysqlDataSource) dataSource).setUser("root");
        ((MysqlDataSource) dataSource).setPassword("passwd");

        Connection connection = dataSource.getConnection();

        // 插入数据
        insertData(connection, 1, "张三");
        insertData(connection, 2, "李四");
        insertData(connection, 3, "王五");
        insertData(connection, 4, "赵六");

        // 查询数据
        selectData(connection);

        // 更新数据
        updateData(connection, 1, "张老三");

        // 查询更新后的数据
        selectData(connection);

        // 删除数据
        deleteData(connection, 1);

        // 查询删除后的数据
        selectData(connection);
    }

    private static void insertData(Connection connection, int id, String name) throws SQLException {
        String sql = "INSERT INTO student (id, name) VALUES (?, ?)";
        PreparedStatement statement = connection.prepareStatement(sql);
        statement.setInt(1, id);
        statement.setString(2, name);
        int rowsInserted = statement.executeUpdate();
        System.out.println(rowsInserted + " row(s) inserted.");
        statement.close();
    }

    private static void selectData(Connection connection) throws SQLException {
        String sql = "SELECT * FROM student";
        PreparedStatement statement = connection.prepareStatement(sql);
        ResultSet resultSet = statement.executeQuery();
        while (resultSet.next()) {
            int id = resultSet.getInt("id");
            String name = resultSet.getString("name");
            System.out.println("ID: " + id + ", Name: " + name);
        }
        resultSet.close();
        statement.close();
    }

    private static void updateData(Connection connection, int id, String name) throws SQLException {
        String sql = "UPDATE student SET name = ? WHERE id = ?";
        PreparedStatement statement = connection.prepareStatement(sql);
        statement.setString(1, name);
        statement.setInt(2, id);
        int rowsUpdated = statement.executeUpdate();
        System.out.println(rowsUpdated + " row(s) updated.");
        statement.close();
    }

    private static void deleteData(Connection connection, int id) throws SQLException {
        String sql = "DELETE FROM student WHERE id = ?";
        PreparedStatement statement = connection.prepareStatement(sql);
        statement.setInt(1, id);
        int rowsDeleted = statement.executeUpdate();
        System.out.println(rowsDeleted + " row(s) deleted.");
        statement.close();
    }
}

这个示例展示了数据库的增删改查操作:

  1. insertData() 方法用于插入数据,通过预编译的 PreparedStatement 对象执行插入操作。

  2. selectData() 方法用于查询数据,通过预编译的 PreparedStatement 对象执行查询操作,并遍历结果集打印每一行的数据。

  3. updateData() 方法用于更新数据,通过预编译的 PreparedStatement 对象执行更新操作。

  4. deleteData() 方法用于删除数据,通过预编译的 PreparedStatement 对象执行删除操作。

main() 方法中,先连接到数据库,然后按顺序执行插入、查询、更新和删除操作,并输出操作结果。

运行结果:

查看数据库表中的数据:

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

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

相关文章

uniapp 一些常用的公共方法

封装代码可以看这篇文章: uniapp 封装公共方法(无需每个页面引用,直接可以调用)_前端小胡兔的博客-CSDN博客uniapp 封装公共方法https://blog.csdn.net/weixin_44805839/article/details/131684296?spm1001.2014.3001.5501 常用方法: 自定义返回页面 (直接使用uni.naviga…

DuiLib中的list控件以及ListContainerElement控件

文章目录 前言1、创建list控件2、创建 ListContainerElement 元素,并添加到 List 控件中,这里的ListContainerElement用xml来表示3、在 ListContainerElement 元素中添加子控件 1、List控件2、ListContainerElement控件 前言 在 Duilib 中,List 控件用于…

【微服务】集成其他已有的模块

目录 下载新的模块信息删除git信息将已有模块复制到当前项目里面在父pom文件中,加上复制进的模块重新解析结果 下载新的模块信息 删除git信息 将已有模块复制到当前项目里面 在父pom文件中,加上复制进的模块 重新解析 结果 集成完成

第二十四章:索引的数据结构

第二十四章:索引的数据结构 24.1:为什么使用索引 ​ 索引是存储引擎用于快速找到数据记录的一种数据结构,就好比一本教课书的目录部分,通过目录中找到对应文章的页码,便可快速定位到需要的文章。MySQL中也是一样的道…

Java实现图片与Base64编码互转

Java实现图片与Base64编码互转 淘宝里面的html用base64转换图片,不知道为什么,不过看起来好像很美好,话不多说,直接上代码: import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOE…

Effective Java笔记(6)避免创建不必要的对象

一般来说,最好能重用单个对象,而不是在每次需要 的时候就创建一个相同功能的新对象 。 重用方式既快速,又流行 。 如果对象是不可变的( immutable ) (详见第 17 条),它就始终可以被重用 。 作为…

Golang gui walk入门教程(一)安装walk环境

一、golang环境 Go 1.11.x or later 二、安装walk go get github.com/lxn/walk 三、安装rsrc 运行walk程序需要manifest,rsrc提供了这个功能 go install github.com/akavel/rsrc 安装完成后在GOPATH的bin下面会有一个rsrc.exe的可执行文件 在idea的termial输入r…

Tauri 提供界面 + 使用 Rust 实现连接远程 Linux 服务器、发送文件、执行命令

Tauri 提供界面 使用 Rust 实现连接远程 Linux 服务器、发送文件、执行命令 文章目录 Tauri 提供界面 使用 Rust 实现连接远程 Linux 服务器、发送文件、执行命令一、Tauri 概述二、界面预览三、代码参考1、main.rs2、App.vue3、Greet.vue4、依赖 一、Tauri 概述 Tauri 是一…

C语言动态获取设备的网络接口名称和状态以及对应的IP地址

一、目的 在实际项目中需要获取设备的IP地址然后通过广播的形式通知局域网内的其他设备。 二、介绍 方法一 通过ioctl方式获取SIOCGIFADDR信息 /** C Program to Get IP Address*/ #include <stdio.h> #include <string.h> #include <sys/types.h> #includ…

mfc120u.dll丢失修复,mfc120u.dll缺失的解决方法

MFC120u.dll缺失的原因 当系统中缺少或损坏了MFC120u.dll文件时&#xff0c;就会出现"MFC120u.dll缺失"的错误提示。造成MFC120u.dll缺失的原因可能有以下几种情况&#xff1a; 1.文件删除或损坏&#xff1a;MFC120u.dll文件可能因为误删除、病毒感染、硬盘故障等原…

pearsonr报错:计算结果为nan、warning .warn (stats.constantinputwarning (msg))

【1】两个都是ndarry 最后结果为nan&#xff1a; &#xff08;1&#xff09;数据类型转换&#xff1a;都转为一样的float32&#xff1a;依旧报错nan &#xff08;2&#xff09;进入函数内部debug&#xff1a; if (xx[0]).all() or (yy[0]).all() warning .warn (stats.consta…

13-接口、代码、命令如何测试?

可以通过工具&#xff0c;也可以通过代码来进行测试。 这里使用工具Postman。 1.针对接口进行测试 粘贴导入到Postman中&#xff1a; 也可直接复制URL&#xff1a; 粘贴到Postman中&#xff1a; 没有对应的参数&#xff0c;那么可以直接请求&#xff1a; 这个接口的GET、POST…

无法将“pip“识别为cmdlet、函数、脚本文件或可运行程序的名称。

出现问题如下&#xff1a; 出现问题原因&#xff1a; 没有添加pip对应的安装目录进入环境变量里面的系统变量。 解决方案&#xff1a; 1.确定python的安装路径 将python的路径添加到系统变量中 2.输入pip所在的安装路径&#xff1a; python路径\Lib\site-packages 3.添加…

PyTorch训练RNN, GRU, LSTM:手写数字识别

文章目录 pytorch 神经网络训练demoResult参考来源 pytorch 神经网络训练demo 数据集&#xff1a;MNIST 该数据集的内容是手写数字识别&#xff0c;其分为两部分&#xff0c;分别含有60000张训练图片和10000张测试图片 图片来源&#xff1a;https://tensornews.cn/mnist_intr…

eureka使用错误

错误 java.net.UnknownHostException: INVENTORYSERVICE 分析&解决&#xff1a; 这里的请求执行错误 但eureka可以找到服务 手动创建RestTemlate到容器中&#xff0c;未加LoadBalanced注解 加上注解后重试&#xff0c;成功

java详细显示try/catch块的异常类方法等信息

示例&#xff1a; Testpublic void testException(){try {double theorynumDouble Double.parseDouble(null);} catch (Exception e) {StackTraceElement[] stackTraceElements e.getStackTrace();StackTraceElement stackTraceElementFirst stackTraceElements[0];String c…

一文教你学会Linux数组

目录 &#x1f380;什么是数组&#xff1f; &#x1f380;数组优点 &#x1f380;数组缺点 &#x1f380;定义数组 &#x1f380;数组的取值 &#x1f380;一次取出数组所有的值 &#x1f380;数组长度&#xff1a; 即数组元素个数 &#x1f380;数组的截取&#xff…

【ASP.NET】医学实验室管理(LIS)系统源码

一、医学实验室LIS系统概况 LIS&#xff08;全称Laboratory Information Management System&#xff09;&#xff0c;是专为医院检验科设计的一套实验室信息管理系统&#xff0c;能将实验仪器与计算机组成网络&#xff0c;使病人样品登录、实验数据存取、报告审核、打印分发&am…

MySQL数据库的索引原理与慢SQL优化的5大原则

这篇文章主要介绍了MySQL数据库的索引原理与慢SQL优化的5大原则,包括&#xff1a;建立索引的原则&#xff0c;慢查询优化基本步骤&#xff0c;慢查询优化案例&#xff0c;explain使用&#xff0c;需要的朋友可以参考下 我们知道一般的应用系统&#xff0c;读写比例在10:1左右&…

计算机系统结构与操作系统实验三(2)-进入保护模式

&#x1f4cd;实验要求 从实模式到保护模式的转变&#xff1a; 在刚进入loader尚在实模式下时&#xff0c;在第2行显示&#xff1a;real-Zhangsan 在变为保护模式后&#xff0c;在第3行显示&#xff1a;protect-Zhangsan 加载gdt、将cr0的PE位置为1 &#x1f4cd;实验过程 老…