【C语言基础】双指针在qsort函数中的应用

news/2025/4/22 0:49:59

在C语言中使用 qsort 对字符串数组(如 char* 数组)排序时,必须转换为双指针(char**,这是由字符串数组的内存结构和 qsort 的工作原理决定的。以下是详细解释:


一、底层原理分析

1. 字符串数组的内存结构

假设有一个字符串数组:

char* strs[] = {"apple", "banana", "cherry"};

其内存布局如下:

strs[0] → 指向 "apple" 的首地址
strs[1] → 指向 "banana" 的首地址
strs[2] → 指向 "cherry" 的首地址

每个元素 strs[i] 的类型是 char*(字符串指针),因此:

  • 数组名 strs 的类型是 char**(指向指针的指针)
  • qsort 传递给比较函数的是 strs[i] 的地址(即 &strs[i]),其类型为 char**

2. qsort 的比较函数参数

qsort 的比较函数原型为:

int compar(const void *a, const void *b);
  • 参数 ab 是数组元素的指针,即 &strs[i]&strs[j]
  • 对于字符串数组,ab 的类型是 char**(指针的指针)。

二、正确转换步骤

1. 错误写法(直接转 char*
// 错误!会导致比较的是字符串内容,而非指针地址
int compare_strings_wrong(const void *a, const void *b) {const char *str1 = (const char*)a; // ❌ 直接转换const char *str2 = (const char*)b;return strcmp(str1, str2);
}
  • 问题ab 本质是 &strs[i]char** 类型),若直接转为 char*,实际比较的是指针地址值,而非字符串内容。

2. 正确写法(转双指针后解引用)
int compare_strings(const void *a, const void *b) {const char **str1 = (const char **)a; // ✅ 转换为双指针const char **str2 = (const char **)b;return strcmp(*str1, *str2); // 解引用得到实际的字符串指针
}
  • 关键点
    • a/b 转为 char**(因为它们本质是 char** 类型)
    • 解引用(*str1)得到实际的 char*(字符串首地址)
    • 比较字符串内容而非指针地址。

三、类比其他类型

int 数组为例:

int arr[] = {5, 2, 8};
  • 比较函数的参数是 int*(数组元素地址),直接解引用即可:
int compare_ints(const void *a, const void *b) {int num1 = *(const int*)a; // ✅ 直接转 int*int num2 = *(const int*)b;return num1 - num2;
}
  • 字符串数组与之不同,因为数组元素本身是指针(char*),所以需要多一层解引用。

四、常见错误场景

1. 错误结果示例

若错误地直接转换:

const char *str1 = (const char*)a; // a 是 char**,转 char* 后值等于 &strs[i]
const char *str2 = (const char*)b;
  • str1 的值是 &strs[i](即 char** 的地址值)
  • strcmp(str1, str2) 会比较这两个地址值的 ASCII 码,而非字符串内容,导致排序混乱。

2. 内存访问崩溃

若字符串数组元素为 NULL

char* strs[] = {"apple", NULL, "cherry"};
  • 错误转换后,strcmp 会尝试访问 NULL 地址,导致段错误。

五、总结

数据类型qsort 参数类型比较函数转换方式
int[]int*(指向元素的指针)*(const int*)a
char*[]char**(指向指针的指针)*(const char**)a

核心原则qsort 始终传递数组元素的地址,需根据元素类型决定如何转换和解引用。

以下是引用

逐步解释

  1. 定义比较函数

    int compare_strings(const void *a, const void *b) {const char **str1 = (const char **)a;const char **str2 = (const char **)b;return strcmp(*str1, *str2);
    }
    
    • const void *aconst void *bqsort 传递给比较函数的参数。
    • const char **str1const char **str2 是将 const void* 转换为 const char** 后的结果。
    • *str1*str2 分别是 str1str2 指向的字符串。
  2. 调用 qsort

    qsort(strs, len, sizeof(char*), compare_strings);
    
    • strs 是要排序的数组。
    • len 是数组的长度。
    • sizeof(char*) 是每个元素的大小(即每个元素是指针)。
    • compare_strings 是比较函数。
  3. 输出排序后的结果

    for (size_t i = 0; i < len; i++) {printf("%s ", strs[i]);
    }
    printf("\n");
    

为什么需要双指针转换

  • 第一次转换:将 const void* 转换为 const char**,因为 qsort 传递的是指向数组元素的指针。
  • 第二次转换:通过 *str1*str2 获取实际的字符串指针,以便使用 strcmp 函数进行比较。

通过这种方式,我们可以正确地对字符串数组进行排序,而不会出现类型不匹配的问题。


https://dhexx.cn/news/show-5497369.html

相关文章

PyTorch逻辑回归总结

目录 PyTorch逻辑回归总结神经网络基础基本结构学习路径 线性回归简单线性回归多元线性回归 逻辑回归核心原理损失函数 梯度下降法基本思想关键公式学习率影响 PyTorch实现数据准备模型构建代码优化 核心概念对比 PyTorch逻辑回归总结 神经网络基础 基本结构 输入节点隐藏节…

宝塔面板中解锁Laravel日志查看的奥秘

目录 一、前言二、Laravel 日志基础认知2.1 日志的作用2.2 Laravel 日志的默认配置 三、查找 Laravel 日志文件位置3.1 常规存储路径3.2 自定义路径查找 四、查看 Laravel 日志内容4.1 宝塔面板文件管理器查看4.2 使用命令行查看 五、常见问题及解决方法5.1 权限不足无法查看5.…

双目视觉中矩阵等参数说明及矫正

以下是标定文件中各个参数的详细解释&#xff1a; 1. 图像尺寸 (imageSize) 参数值: [1280, 1024]含义: 相机的图像分辨率&#xff0c;宽度为1280像素&#xff0c;高度为1024像素。 2. 相机内参矩阵 (leftCameraMatrix / rightCameraMatrix) 结构: yaml data: [fx, 0, cx, 0,…

基本元器件—电阻器(2025.4.14)

1.常见的电阻 2.电阻的表示方法和测试方法 2.1直标法和色标法 2.2万用表和数字电桥 3. 4.电阻参数 允许误差指的是电阻的标称阻值和实际值差值再除以标称阻值 5.电阻的功能 &#xff08;1&#xff09;分压就是电阻串联分到的电压不一样 &#xff08;2&#xff09;限流&#…

从零开始:Python运行环境之VSCode与Anaconda安装配置全攻略 (1)

从零开始&#xff1a;Python 运行环境之 VSCode 与 Anaconda 安装配置全攻略 在当今数字化时代&#xff0c;Python 作为一种功能强大且易于学习的编程语言&#xff0c;被广泛应用于数据科学、人工智能、Web 开发等众多领域。为了顺利开启 Python 编程之旅&#xff0c;搭建一个稳…

Ubuntu 常用命令行指令

1. 文件与目录操作 命令作用示例ls列出目录内容ls -l&#xff08;详细列表&#xff09;cd切换目录cd ~/Documentspwd显示当前目录路径pwdmkdir创建目录mkdir new_folderrm删除文件rm file.txtrm -r递归删除目录rm -r old_dircp复制文件cp file.txt backup/mv移动/重命名文件mv…

通道注意力机制|Channel Attention Neural Network

一、通道注意力机制 论文&#xff1a;ECA-Net: Efficient Channel Attention for Deep Convolutional Neural Networks 近年来&#xff0c;通道注意力机制在提高深度卷积神经网络CNN的性能方面显示出了巨大潜力。然而&#xff0c;大多数现有方法致力于开发更复杂的注意力模块…

回溯算法:List 还是 ArrayList?一个深拷贝引发的思考

在学习和使用回溯算法解决问题时&#xff0c;我们经常会遇到需要维护一个结果列表&#xff0c;例如所有可能的子集、组合或排列。 这个结果列表通常是一个 List<List<Integer>>&#xff0c;其中内部的 List<Integer> 代表一个具体的解。 然而&#xff0c;在…

【FPGA】——DDS信号发生器设计

目录 一 、IP核的使用 &#xff08;1)RAM IP核 (2)FIFO IP核 二、DDS信号发生器设计 &#xff08;1&#xff09;代码 &#xff08;2&#xff09;仿真波形 一 、IP核的使用 IP核&#xff1a;ASIC或FPGA中预先设计好具有某种功能的电路模块&#xff0c;参数可修改&#xf…

【差分隐私相关概念】瑞丽差分隐私(RDP)命题2

分步解析与答案 1. c-稳定变换的定义 c-稳定变换是一种将群体数据集&#xff08;如数据库集合&#xff09;的相邻性映射到个体数据集&#xff08;如单条记录变化&#xff09;的变换。具体来说&#xff0c;若变换 g : D ′ → D g: \mathcal{D} \to \mathcal{D} g:D′→D 是 …