Qt案例 通过调用Setupapi.h库实现对设备管理器中设备默认驱动的备份

news2024/12/27 10:46:37

参考腾讯电脑管家-软件市场中的驱动备份专家写的一个驱动备份软件案例,学习Setupapi.h库中的函数使用.通过Setupapi.h库读取设备管理器中安装的设备获取安装的驱动列表,通过bit7z库备份驱动目录下的所有文件.

目录导读

    • 实现效果
    • 相关内容示例
      • 获取SP_DRVINFO_DETAIL_DATA结构数据
      • 相关函数
        • SetupDiBuildDriverInfoList 函数
        • SetupDiEnumDriverInfo函数
        • SetupDiGetDriverInfoDetail 函数
      • 相关实现
        • 获取所有驱动列表
        • 获取驱动SP_DRVINFO_DETAIL_DATA结构数据
        • qDebug打印输出 SP_DRVINFO_DETAIL_DATA 结构数据

实现效果

参考腾讯电脑管家-软件市场中的驱动备份专家软件
在这里插入图片描述
实现对设备管理器中的驱动信息的读取和备份:
在这里插入图片描述
其中的驱动名称,版本信息和驱动目录都是通过设备管理器的属性获取,
详情参考以下文章内容:
Qt案例 调用WINDOWS API中的SETUPAPI.H库获取设备管理器中设备的详细信息中的属性值(一)
Qt案例 调用WINDOWS API中的SETUPAPI.H库获取设备管理器中设备的详细信息中的属性值(二)
驱动备份使用bit7z调用7z.exe实现相关功能,参考:
Qt 编译使用Bit7z库接口调用7z.dll、7-Zip.dll解压压缩常用Zip、ISO9660、Wim、Esd、7z等格式文件(一)

相关内容示例

遍历设备管理器中驱动程序的相关信息:

  • 获取SP_DRVINFO_DETAIL_DATA结构数据

SP_DRVINFO_DETAIL_DATA结构包含有关特定驱动程序信息结构的详细信息。
语法

typedef struct _SP_DRVINFO_DETAIL_DATA_W {
  DWORD     cbSize;  //SP_DRVINFO_DETAIL_DATA结构的大小(以字节为单位)。
  FILETIME  InfDate; //此驱动程序的 INF 文件的日期。
  DWORD     CompatIDsOffset; //此值还可用于确定 CompatID 列表之前是否有 硬件 ID 。 如果此值大于 1,则 HardwareID 缓冲区中的第一个字符串是硬件 ID。 如果此值小于或等于 1,则没有硬件 ID。
  DWORD     CompatIDsLength; //CompatID 列表的长度(以字符为单位),从 HardwareID 缓冲区的开头偏移 CompatIDsOffset 开始。
  ULONG_PTR Reserved; //保留。 仅限内部使用。
  WCHAR     SectionName[LINE_LEN]; //一个以 NULL 结尾的字符串,包含此驱动程序的 INF DDInstall 节 的名称。 这必须是基本 DDInstall 节名称(如 InstallSec),没有任何特定于 OS/体系结构的扩展。
  WCHAR     InfFileName[MAX_PATH]; //一个以 NULL 结尾的字符串,其中包含此驱动程序的 INF 文件的完整限定名称。
  WCHAR     DrvDescription[LINE_LEN]; //一个以 NULL 结尾的字符串,用于描述驱动程序。
  WCHAR     HardwareID[ANYSIZE_ARRAY]; //包含 ID 列表的缓冲区 (单个 硬件 ID ,后跟 兼容 ID 列表) 。 这些 ID 对应于 INF 模型部分中的硬件 ID 和兼容 ID。

列表中的每个 ID 都是以 NULL 结尾的字符串。
} SP_DRVINFO_DETAIL_DATA_W, *PSP_DRVINFO_DETAIL_DATA_W;

相关函数

  • SetupDiBuildDriverInfoList 函数

SetupDiBuildDriverInfoList 函数生成与特定设备关联的驱动程序列表,或与设备信息集的全局类驱动程序列表相关联。
语法

//! 如果成功,该函数将返回 TRUE 。 否则,它将返回 FALSE ,并且可以通过调用 GetLastError 来检索记录的错误。
WINSETUPAPI BOOL SetupDiBuildDriverInfoList(
  [in]      HDEVINFO         DeviceInfoSet,  //设备信息设置为包含驱动程序列表的设备信息的句柄,可以是全局所有设备信息元素,也可以是专用于单个设备信息元素。 设备信息集不得包含远程设备信息元素
  [in, out] PSP_DEVINFO_DATA DeviceInfoData,//指向 DeviceInfoSet 中设备信息元素的SP_DEVINFO_DATA结构的指针,该元素表示要为其生成驱动程序列表的设备。 此参数是可选的,可以为 NULL。 如果指定此参数,则列表与指定的设备相关联。 如果此参数为 NULL,则列表与 DeviceInfoSet 的全局类驱动程序列表相关联。
  [in]      DWORD            DriverType //要生成的驱动程序列表的类型
);
  • SetupDiEnumDriverInfo函数

SetupDiEnumDriverInfo 函数枚举驱动程序列表的成员。
语法

//! 如果成功,该函数将返回 TRUE 。 否则,它将返回 FALSE ,并且可以通过调用 GetLastError 来检索记录的错误。
WINSETUPAPI BOOL SetupDiEnumDriverInfoA(
  [in]           HDEVINFO           DeviceInfoSet, //包含要枚举的驱动程序列表 的设备信息集 的句柄。
  [in, optional] PSP_DEVINFO_DATA   DeviceInfoData, //指向 SP_DEVINFO_DATA 结构的指针,该结构指定 DeviceInfoSet 中的设备信息元素。 此参数是可选的,可以为 NULL。 如果指定此参数, SetupDiEnumDriverInfo 将枚举指定设备的驱动程序列表。 如果此参数为 NULL, 则 SetupDiEnumDriverInfo 枚举与 DeviceInfoSet 关联的全局类驱动程序列表, (此列表的类型为 SPDIT_CLASSDRIVER) 。
  [in]           DWORD              DriverType, //要枚举的驱动程序列表的类型
  [in]           DWORD              MemberIndex, //要检索的驱动程序信息成员的从零开始的索引。
  [out]          PSP_DRVINFO_DATA_A DriverInfoData //指向调用方初始化 的SP_DRVINFO_DATA 结构的指针,该结构接收有关枚举驱动程序的信息。 调用方必须设置 DriverInfoData。在调用 SetupDiEnumDriverInfo 之前,cbSize 为 sizeof ( SP_DRVINFO_DATA) 。 如果未正确设置 cbSize 成员, 则 SetupDiEnumDriverInfo 将返回 FALSE。
);
  • SetupDiGetDriverInfoDetail 函数

SetupDiGetDriverInfoDetail 函数检索设备信息集或设备信息集中特定设备信息元素的驱动程序信息详细信息。
语法

//! 如果成功,该函数将返回 TRUE 。 否则,它将返回 FALSE 
WINSETUPAPI BOOL SetupDiGetDriverInfoDetailW(
  [in]            HDEVINFO                  DeviceInfoSet, //包含要为其检索驱动程序信息的驱动程序信息元素 的设备信息集 的句柄。
  [in, optional]  PSP_DEVINFO_DATA          DeviceInfoData, //指向 SP_DEVINFO_DATA 结构的指针,该结构指定表示要为其检索驱动程序信息的设备的设备信息元素。 此参数是可选的,可以为 NULL。 如果指定此参数, SetupDiGetDriverInfoDetail 将检索指定设备的驱动程序列表中有关驱动程序的信息。 如果此参数为 NULL, 则 SetupDiGetDriverInfoDetail 将检索有关作为 DeviceInfoSet 全局类驱动程序列表成员的驱动程序的信息。
  [in]            PSP_DRVINFO_DATA_W        DriverInfoData, //指向 SP_DRVINFO_DATA 结构的指针,该结构指定要为其检索详细信息的驱动程序信息元素。 如果指定 了 DeviceInfoData ,则驱动程序必须是 DeviceInfoData 指定的设备的驱动程序列表的成员。 否则,驱动程序必须是 DeviceInfoSet 的全局类驱动程序列表的成员。
  [in, out]       PSP_DRVINFO_DETAIL_DATA_W DriverInfoDetailData, //指向 SP_DRVINFO_DETAIL_DATA 结构的指针,该结构接收有关指定驱动程序的详细信息。 如果未指定此参数, DriverInfoDetailDataSize 必须为零。 如果指定此参数,则 DriverInfoDetailData。在调用 SetupDiGetDriverInfoDetail 之前,cbSize 必须设置为 size of ( SP_DRVINFO_DETAIL_DATA) 的值。
  [in]            DWORD                     DriverInfoDetailDataSize, //DriverInfoDetailData 缓冲区的大小(以字节为单位)。
  [out, optional] PDWORD                    RequiredSize //向变量的指针,该变量接收存储详细驱动程序信息所需的字节数。 此值包括结构的大小,以及包含硬件 ID 列表和兼容 ID 列表的末尾的可变长度字符缓冲区所需的其他字节。
);

相关实现

获取所有驱动列表
//! QList<CONST GUID*> GUID_DEVCLASS_LIST 设备标识
for(int t=0;t<GUID_DEVCLASS_LIST.count();t++)
    {
        QString IconPath=Lib_ExtrationDrives::getInstance().GeticonPath(GUID_DEVCLASS_LIST[t]);
        HDEVINFO deviceInfoSet = SetupDiGetClassDevs(GUID_DEVCLASS_LIST[t], NULL, NULL, DIGCF_PRESENT);
        if (deviceInfoSet == INVALID_HANDLE_VALUE) {
            printf("Failed to get device information set. Error code: %d\n", GetLastError());
            return ;
        }

        SP_DEVINFO_DATA deviceInfoData;
        deviceInfoData.cbSize = sizeof(SP_DEVINFO_DATA);
        for (DWORD i = 0; SetupDiEnumDeviceInfo(deviceInfoSet, i, &deviceInfoData); i++) {

            //SetupDiBuildDriverInfoList 函数生成与特定设备关联的驱动程序列表,或与设备信息集的全局类驱动程序列表相关联。
            bool ret = SetupDiBuildDriverInfoList(deviceInfoSet, &deviceInfoData, SPDIT_CLASSDRIVER);
            if(!ret)
            {
                qDebug("Error enumerating driver info. Error code: %d\n", GetLastError());
                continue ;
            }
            SP_DRVINFO_DATA  rDriverInfoData;
            rDriverInfoData.cbSize = sizeof(SP_DRVINFO_DATA);
            //SetupDiEnumDriverInfo 函数枚举驱动程序列表的成员
            for(int j=0;SetupDiEnumDriverInfo(deviceInfoSet, &deviceInfoData, SPDIT_CLASSDRIVER, j, &rDriverInfoData);j++)
            {
                PSP_DRVINFO_DETAIL_DATA rDriverInfoDetail=new SP_DRVINFO_DETAIL_DATA() ;
                rDriverInfoDetail->cbSize=sizeof(SP_DRVINFO_DETAIL_DATA);

                if(Lib_ExtrationDrives::getInstance().GetSetupDiGetDriverInfoDetail(deviceInfoSet,&deviceInfoData, &rDriverInfoData,rDriverInfoDetail))
                {
                    qDebug()<<"-----"<<i<<" - "<<j;
//                    Print_DriverInfoDetail(rDriverInfoDetail);
                    qDebug()<<"[cbSize] :"<<rDriverInfoDetail->cbSize;
                    qDebug()<<"[SectionName] :"<<QString::fromWCharArray(rDriverInfoDetail->SectionName);
                    //! SetupOpenInfFile 函数打开一个 INF 文件并返回它的句柄。
                    qDebug()<<"[InfFileName] :"<<QString::fromWCharArray(rDriverInfoDetail->InfFileName);
                    qDebug()<<"[DrvDescription] :"<<QString::fromWCharArray(rDriverInfoDetail->DrvDescription);
                    qDebug()<<"[CompatIDsOffset] :"<<rDriverInfoDetail->CompatIDsOffset;
                    qDebug()<<"[CompatIDsLength] :"<<rDriverInfoDetail->CompatIDsLength;

                }
                free(rDriverInfoDetail);
            }
        }
        SetupDiDestroyDeviceInfoList(deviceInfoSet);
    }
获取驱动SP_DRVINFO_DETAIL_DATA结构数据

通过SetupDiGetDriverInfoDetail函数检索SP_DRVINFO_DETAIL_DATA包含有关特定驱动程序信息结构的详细信息,预防ERROR_INSUFFICIENT_BUFFER异常

bool Lib_ExtrationDrives::GetSetupDiGetDriverInfoDetail(HDEVINFO infoset,
                                                        PSP_DEVINFO_DATA device,
                                                        PSP_DRVINFO_DATA  rDriver,
                                                        PSP_DRVINFO_DETAIL_DATA& rDriverInfoDetail) const
{
    bool isSuccess=true;
    DWORD needSize=0;
    if(!SetupDiGetDriverInfoDetail(infoset,
                                   device,
                                   rDriver,
                                   rDriverInfoDetail,
                                   sizeof(SP_DRVINFO_DETAIL_DATA),
                                   &needSize))
    {
        if((GetLastError()== ERROR_INSUFFICIENT_BUFFER))
        {
            //                qDebug()<<"cbSize : "<<rDriverInfoDetail->cbSize<<" needSize : "<<needSize;
            rDriverInfoDetail=(SP_DRVINFO_DETAIL_DATA*)malloc(needSize);
            rDriverInfoDetail->cbSize = sizeof(SP_DRVINFO_DETAIL_DATA)  ;
            if (!SetupDiGetDriverInfoDetail(infoset, device, rDriver, rDriverInfoDetail, needSize, NULL))
            {
                isSuccess=false;
                qDebug()<<"Error getting detailed driver info . because needSize:"<<needSize;
            }
        }
        else
        {
            isSuccess=false;
            qDebug()<<"Error getting detailed driver info !";
        }
    }
    return isSuccess;
}

qDebug打印输出 SP_DRVINFO_DETAIL_DATA 结构数据

注:根据硬件 ID 和 兼容 ID 列表在 INF 模型部分中的指定方式, HardwareID 缓冲区可以类似于以下任何一种:

  • \0
  • <HWID>\0
  • <HWID>\0<COMPATID_1>\0…<COMPATID_N>\0\0
  • \0<COMPATID_1>\0…<>COMPATID_N\0\0
void Print_DriverInfoDetail(PSP_DRVINFO_DETAIL_DATA rDriverInfoDetail) 
{
    qDebug()<<"[cbSize] :"<<rDriverInfoDetail->cbSize;
    qDebug()<<"[SectionName] :"<<QString::fromWCharArray(rDriverInfoDetail->SectionName);
    //! SetupOpenInfFile 函数打开一个 INF 文件并返回它的句柄。
    qDebug()<<"[InfFileName] :"<<QString::fromWCharArray(rDriverInfoDetail->InfFileName);
    qDebug()<<"[DrvDescription] :"<<QString::fromWCharArray(rDriverInfoDetail->DrvDescription);
    qDebug()<<"[CompatIDsOffset] :"<<rDriverInfoDetail->CompatIDsOffset;
    qDebug()<<"[CompatIDsLength] :"<<rDriverInfoDetail->CompatIDsLength;
    // parse the hardware ID, if it exists
    if (rDriverInfoDetail->CompatIDsOffset > 1)
    {
        // Parse for hardware ID from index 0.
        // This is a single NULL-terminated string
        qDebug()<<"设备的硬件 id:";
        //            qDebug()<<QString::fromWCharArray(rDriverInfoDetail->HardwareID);
        wchar_t* hardwareid=new wchar_t[MAX_PATH];
        int col=0;
        for(int i=0;i<rDriverInfoDetail->CompatIDsOffset;i++)
        {
            if(rDriverInfoDetail->HardwareID[i]!='\0')
            {
                hardwareid[col]=rDriverInfoDetail->HardwareID[i];
                col++;
            }
            else
            {
                hardwareid[col]='\0';
                qDebug()<<QString::fromWCharArray(hardwareid);
                free(hardwareid);
                hardwareid=new wchar_t[MAX_PATH];
                col=0;
            }
        }
    }
    // Parse the compatible IDs, if they exist
    if (rDriverInfoDetail->CompatIDsLength > 0)
    {
        qDebug()<<"设备的兼容 id:";
        // Parse for list of compatible IDs from CompatIDsOffset.
        // This is a double NULL-terminated list of strings (i.e. MULTI-SZ)
        wchar_t* compatibleid=new wchar_t[MAX_PATH];
        int col=0;
        int lengths=rDriverInfoDetail->CompatIDsOffset+rDriverInfoDetail->CompatIDsLength;
        for(int i=rDriverInfoDetail->CompatIDsOffset;i<lengths;i++)
        {
            if(rDriverInfoDetail->HardwareID[i]!='\0')
            {
                compatibleid[col]=rDriverInfoDetail->HardwareID[i];
                col++;
            }
            else
            {
                compatibleid[col]='\0';
                qDebug()<<QString::fromWCharArray(compatibleid);
                free(compatibleid);
                if(i+1<lengths &&(rDriverInfoDetail->HardwareID[i+1]=='\0')) //结束
                    break;
                compatibleid=new wchar_t[MAX_PATH];
                col=0;
            }
        }
    }
    qDebug()<<endl;

}

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

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

相关文章

DevOps已死?2024年的DevOps将如何发展

随着我们进入2024年&#xff0c;DevOps也发生了变化。新兴的技术、变化的需求和发展的方法正在重新定义有效实施DevOps实践。 IDC预测显示&#xff0c;未来五年&#xff0c;支持DevOps实践的产品市场继续保持健康且快速增长&#xff0c;2022年-2027年的复合年增长率&#xff0…

数字人解决方案——Champ单个视频单张图像生成可控且一致的人体视频生成

概述 Champ是阿里巴巴集团、南京大学和复旦大学的研究团队共同提出了一种创新的人体动画生成技术&#xff0c;Champ能够在仅有一段原始视频和一张静态图片的情况下&#xff0c;激活图片中的人物&#xff0c;使其按照视频中的动作进行动态表现&#xff0c;极大地促进了虚拟主播…

Git-LFS 远程命令执行漏洞 CVE-2020-27955 漏洞复现

今天遇到了一个比较有意思的洞&#xff0c;复现一下下.......... 漏洞描述 Git LFS 是 Github 开发的一个 Git 的扩展&#xff0c;用于实现 Git 对大文件的支持 一些受影响的产品包括Git&#xff0c;GitHub CLI&#xff0c;GitHub Desktop&#xff0c;Visual Studio&#xff0…

C#开发常用的库使用-Ioc库Autofac的使用

介绍 Autofac 是一个适用于 .NET 的强大的 IoC&#xff08;控制反转&#xff09;容器。它管理类之间的依赖关系&#xff0c;使得随着应用程序规模和复杂性的增长&#xff0c;应用程序仍然易于修改。它允许您管理 .NET 应用程序中的组件依赖关系&#xff0c;并自动将依赖项解析…

数据库(mysql)-基本查询语句(DQL)

查询语句 这边查询是给予一定表格,这边先做个解释 教师表包括(name(姓名),gender(性别),salary(工资),title(职位),subject_id(课程的编号),comm(奖金)) 学生表包括(姓名(name),gender(性别),job(职位),生日(birth)) 模版 SELECT 字段名 FROM 查询表 WHERE 查询语句 或与非…

Linux|从 STDIN 读取 Awk 输入

简介 在之前关于 Awk 工具的系列文章中&#xff0c;主要探讨了如何从文件中读取数据。但如果你希望从标准输入&#xff08;STDIN&#xff09;中读取数据&#xff0c;又该如何操作呢&#xff1f; 在本文中&#xff0c;将介绍几个示例&#xff0c;展示如何使用 Awk 来过滤其他命令…

Go语言实战:掌握html/template包的高效开发技巧

Go语言实战&#xff1a;掌握html/template包的高效开发技巧 引言Go标准库html包概览html包的基本用途html/template包的核心功能开始使用html/template应用场景 安全的HTML内容处理防范XSS攻击使用示例内容安全策略&#xff08;CSP&#xff09;小结 操作HTML元素创建和解析HTML…

java数据结构与算法刷题-----LeetCode628. 三个数的最大乘积

java数据结构与算法刷题目录&#xff08;剑指Offer、LeetCode、ACM&#xff09;-----主目录-----持续更新(进不去说明我没写完)&#xff1a;https://blog.csdn.net/grd_java/article/details/123063846 文章目录 排序选择线性搜索最值 排序 解题思路&#xff1a;时间复杂度O( …

简介有向无环图DAG

Sui创纪录的每秒交易量部分归功于数学构造&#xff0c;即有向无环图&#xff08;Directed Acyclic Graph&#xff0c;DAG&#xff09;&#xff0c;该构造通过以最高效的方式处理交易来加速网络交易&#xff0c;而不是按照先来先服务的线性进展。 区块链是设计用于确保数据完整…

APx500音频分析仪硬件简介

两通道模拟输出&#xff0c;两通道或以上的模拟输入接口 线性编码数字音频接口&#xff08;AES/EBU,TOSLINK,SPDIF&#xff09;Linear PCM 脉冲密度调制码流&#xff08;需要APx-PDM选件支持&#xff09; Bluetooth蓝牙音频码流&#xff08;需APx-BT选件支持&#xff09; 最…

基于Android的记单词App系统的设计与实现

博主介绍&#xff1a;✌IT徐师兄、7年大厂程序员经历。全网粉丝15W、csdn博客专家、掘金/华为云//InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;&#x1f3…

js 求出对应年月的开始日期和结束日期

实现效果&#xff1a; 背景&#xff1a; element日期时间选择组件&#xff0c;通过时间选择年-月&#xff0c;求出对应年月的开始日期和结束日期。 核心代码&#xff1a; /*** 返回给定年份和月份的起始和结束日期。* 202403 -> 2024-03-01 2024-03-31* param {number} y…

11-新热文章-实时计算

热点文章-实时计算 1 今日内容 1.1 定时计算与实时计算 1.2 今日内容 kafkaStream 什么是流式计算 kafkaStream概述 kafkaStream入门案例 Springboot集成kafkaStream 实时计算 用户行为发送消息 kafkaStream聚合处理消息 更新文章行为数量 替换热点文章数据 2 实时…

Linux下解决Java输出文件,中文变问号和中文乱码问题

# 启用 zh_CN.UTF-8 字体 echo "zh_CN.UTF-8 UTF-8" >> /etc/locale.gen # 生成并安装 locale 数据 locale-gen # 更新当前 shell 环境 source /etc/default/locale 相当于以下操作&#xff1a; # Debian sudo apt install locales sudo dpkg-reconfigur…

基于SSM+Jsp+Mysql的超市管理系统

开发语言&#xff1a;Java框架&#xff1a;ssm技术&#xff1a;JSPJDK版本&#xff1a;JDK1.8服务器&#xff1a;tomcat7数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09;数据库工具&#xff1a;Navicat11开发软件&#xff1a;eclipse/myeclipse/ideaMaven包…

【第二十四篇】使用Burpsuite实现反射、储存、DOM型XSS(靶场实战案例)

目录 反射性XSS储存型XSSDOM XSS反射性XSS 搜索1后,审查元素: 猜测<font>标签中没有进行XSS特殊字符转义,而在<font>标签内,可使用<script>标签: <script>alert(1)</script>储存型XSS 该模块对姓名、电子邮件、网站做过滤处理,但评论处…

Few-Shot目标检测数据集 | Few-Shot目标检测数据集_已经整理成MS-COCO数据格式_含60000+张图_可直接用于目标检测算法训练

项目应用场景 面向 Few-Shot 目标检测场景&#xff0c;项目提供 6000 张图&#xff0c;已经整理成 MS-COCO 数据格式&#xff0c;可用于 Few-Shot 目标检测的训练数据集&#xff0c;或作为 Few-Shot 目标检测数据集的补充。 数据集展示 数据集下载 > 具体参见项目 README.m…

【算法详解】二分查找

1. 二分查找算法介绍 「二分查找算法&#xff08;Binary Search Algorithm&#xff09;」&#xff1a;也叫做 「折半查找算法」、「对数查找算法」。是一种在有序数组中查找某一特定元素的搜索算法。 基本算法思想&#xff1a;先确定待查找元素所在的区间范围&#xff0c;在逐步…

SSM党员管理系统

一、系统介绍 党员管理系统: 可以方便管理人员对党员管理系统的管理&#xff0c;提高信息管理工作效率及查询效率&#xff0c;有利于更好的为用户提供服务。 主要的模块包括&#xff1a; 1、后台功能&#xff1a; 管理员角色&#xff1a;首页、个人中心&#xff0c;党员管理…

MySQL高级(索引结构B + Tree)

以一颗最大度数&#xff08;max - degree&#xff09;为 4 &#xff08;4阶&#xff09;的 b tree 为例&#xff1a; 绿色框框起来的部分&#xff0c;是索引部分&#xff0c;仅仅起到索引数据的作用&#xff0c;不存储数据。红色框框起来的部分&#xff0c;是数据存储部分&…