当前位置: 首页 > article >正文

c语言笔记 函数参数的等价(下)

为什么这三种写法是等价的?

    这三种写法是等价的,数组在作为函数参数的时候会变成指针,数组的大小会被系统编译器自动忽略
所以char *(argv[argc]) 等价于 char *(argv[]) 
*和[]是可以相互转换的所以char *(argv[])等价 char *(*argv)

作为命令行:这里我们需要的是一个指针数组,其实上面的理解都可以理解成一个指针数组,因为指针数组
与char型的二维数组好搭配
int *p[5] = {指针,指针,指针...} 变量的类型 int *[5]  数组的元素类型:int * --->指针
char *(argv[argc])    char *(argv[])   char *(*argv)
上面三种写法只有在作为函数参数的时候是等价的,如果是在函数内定义的那么它们就是不一样的

  1. char *(argv[argc])

    • 这种写法声明argv是一个数组,其大小为argc,每个元素是一个指向char的指针。

    • 在函数参数中,数组的大小argc实际上会被编译器忽略,因为数组作为参数传递时会退化为指针。

  2. char *(argv[])

    • 这种写法声明argv是一个数组,但没有指定数组的大小。

    • 在函数参数中,未指定大小的数组会被编译器视为一个指针。

  3. char **argv

    • 这种写法直接声明argv是一个指向char *的指针,即二级指针。

        在C语言中,当数组作为函数参数传递时,它会退化为指向其首元素的指针。这意味着,无论你如何声明数组参数(无论是指定大小还是不指定大小),编译器最终都会将其视为一个指针。因此,char *(argv[argc])char *(argv[])在语义上等价于char **argv

代码示例

以下是一个简单的示例,展示这三种声明方式的等价性:

#include <stdio.h>

// 使用 char *(argv[argc])
void printArguments(int argc, char *(argv[argc])) {
    for (int i = 0; i < argc; i++) {
        printf("Argument %d: %s\n", i, argv[i]);
    }
}

// 使用 char *(argv[])
void printArguments2(int argc, char *(argv[])) {
    for (int i = 0; i < argc; i++) {
        printf("Argument %d: %s\n", i, argv[i]);
    }
}

// 使用 char **argv
void printArguments3(int argc, char **argv) {
    for (int i = 0; i < argc; i++) {
        printf("Argument %d: %s\n", i, argv[i]);
    }
}

int main() {
    char *arguments[] = {"arg0", "arg1", "arg2"};
    int argCount = sizeof(arguments) / sizeof(arguments[0]);

    printArguments(argCount, arguments);
    printArguments2(argCount, arguments);
    printArguments3(argCount, arguments);

    return 0;
}

        在这段代码中,printArgumentsprintArguments2printArguments3三个函数的参数声明方式不同,但它们的功能和行为完全相同。

总结

  • 数组参数的退化:数组作为函数参数时会退化为指向其首元素的指针。
  • 解引用和下标操作的等价性:在C语言中,*(ptr + i)ptr[i]是等价的。
  • 数组大小在参数声明中被忽略:在函数参数中声明数组时,数组的大小会被编译器忽略。

http://www.kler.cn/a/580426.html

相关文章:

  • 【SVN内网穿透】远程访问SVN服务
  • Ubuntu下的QT无法输入中文问题
  • 2025年渗透测试面试题总结-字某跳动-安全研究实习生(三面)(题目+回答)
  • Redis- 哨兵
  • 荣耀手机如何编辑图片?编辑图片技巧、软件分享
  • 笔记:代码随想录算法训练营day39:LeetCode 198.打家劫舍,213.打家劫舍II,337.打家劫舍III
  • 动态ip和静态ip适用于哪个场景?有何区别
  • 算法精讲 | 树(二):BFS层序遍历の魔法——像水波纹一样扫描整棵树
  • 逐梦DBA:解决Linux下MySQL远程登录报错
  • 4.桥接模式
  • 机器学习 Day01人工智能概述
  • Tomcat下载安装及日志乱码问题解决
  • 上位机知识篇---Linux特殊功能文件
  • 极致的灵活度满足工程美学:用Vue Flow绘制一个完美流程图
  • 27. Harmonyos Next仿uv-ui 组件NumberBox 步进器组件禁用状态
  • Docker save命令怎么用
  • Flutter开发避坑指南:高频问题排查与性能调优实战
  • 如何在需求分析阶段考虑未来扩展性
  • ⭐LeetCode周赛 Q1. 找出最大的几近缺失整数——模拟⭐
  • [Python爬虫系列]bilibili