浅谈mian函数的参数解析

本文详细介绍了C语言中main函数的参数使用方法,包括argc、argv和envp的具体含义及应用场景,并通过实例演示如何利用这些参数实现命令行计算器功能。

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

我们从一开始接触C语言就在使用mian函数,但你们有没有发main函数其实也是被调用的,而且main函数还是有参数的。
main函数在_tmainCRTStartup函数中调用的,而_tmainCRTStartup函数是被mianCRTStartup调用的。这里我们主要讲main函数的参数,那我们先看看main函数的原型:

int main(int argc,char *argv[],char *envp[])
{
}

我们可以看出argc是个整形变量,char *argv[]和char *envp[]是字符指针数组,每个元素时存放字符指针。它们三具体存的是什么,我们接下来来看:

int main(int argc, char *argv[], char *envp[])
{
    int i = 0;
    for (i = 0; i < argc; i++)
    {
        printf("%s\n", argv[i]);//argv[i]是字符指针,指向这串字符的首地址,我们把这个字符指针所指向的这串字符打印出来
    }
    return 0;
}

执行结果:
这里写图片描述
这会你肯定会觉得怎么会是这样,这会我把我这个工程改一下:
点自己的工程——>属性——>调试这里写图片描述
我在命令参数那写了三个词:你好 世界 加油
然后我们来看结果:这里写图片描述
可能大家还是不明白它是干嘛的,刚刚我们是在命令参数那写入了这几个词,在linux里你输入ls -a ,cd ,pwz等命令会执行相关操作。那我们把我们的命令参数改成这样:
这里写图片描述
通过以上两个例子可以看出我们把命令参数设成什么,最后执行结果就就出来什么。那为什么它不像linux那样,能执行相关操作呢?因为我们只是把它打印出来了,上面的代码——>printf(“%s\n”, argv[i]); 还没进行操作。
上面的代码我们只用了argc和argv[ ] ,那我先来说说这两个参数是干嘛的。
argc:argc是执行程序时的命令行参数个数。需要注意的是,程序本身的文件也算一个。比如上面这个程序叫day19.exe,你在命令参数里写了-a -s -m三个参数,那argc等于4。
argv[]:存的是命令行参数(字符串)的首地址,包括程序本身的文件。还是以上面的程序为例:这里写图片描述
那我们如何能像linux命令那样能操作呢?我们来简单的举个例子:
使用main函数的参数,实现一个整数计算器,程序可以接受三个参数,第一个参数“-a”选项执行加法,“-s”选项执行减法,“-m”选项执行乘法,“-d”选项执行除法,后面两个参数为操作数。
例如:命令行参数输入:test.exe -a 1 2
执行1+2输出3

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<stdlib.h>
int Add(int x,int y)
{
    return x + y;
}

int Sub(int x, int y)
{
    return x - y;
}
int Mul(int x, int y)
{
    return x * y;
}
int main(int argv, char* argc[], char* envp[])
{
    int ret = 0if (argv != 4)
    {
        printf("输入参数个数有误\n");//因为咱们只需要3个参数,加上程序文件本身的参数共4个
        return 0;
    }
    int x = atoi(*(argc + 2));//因为这些数字是以字符串的形式存的,我们要把字符串转换成整形
    int y = atoi(*(argc + 3));
        switch (*(*(argc+1)+1))//"-a" "-m" "-s" 我们需要判断的是这些字符串的第二个字符,因为第一个字符是'-',我们要根据第二个字符来判断需要干嘛
        {
        case 'a':ret = Add(x, y);
            break;
        case 's':ret = Sub(x, y);
            break;
        case 'm':ret = Mul(x, y);
            break;
        default:printf("输入有误");
            break;
        }
        printf("ret = %d\n", ret);
    system("pause");
    return 0;
}

然后我们在把该程序编译链接,在Windows里自带的通过命令来操作的窗口。

1.先编译链接,复制改程序产生的.exe文件(要在c盘)的目录
2.在电脑搜索栏那输入cmd,打开提示符
3.就可以根据命令操作了
这里写图片描述

前两个参数说完了,那envp装的是什么,有什么用?接下来我们来说说char *envp[]
那我们来编一个小程序,来吧envp里的东西打印出来:
int main(int argc, char *argv[], char *envp[])
{
    int i = 0;
    while (envp[i]!=NULL)
    {
        printf("%s\n", envp[i]);
        i++;
    }
    system("pause");
    return 0;
}

其结果为(节选了一部分):

这里写图片描述
envp里存放正是系统的环境变量,可以右键单击计算机->属性->高级系统设置->环境变量,打开环境变量设置窗口,如图所示
这里写图片描述
把图所示窗口中的变量名称、值与程序打印出来的结果对比,它们是一样的。也就是说envp数组里保存的正式系统的环境变量。

<think>好的,我现在需要帮用户解决关于C语言main函数如何接收输入参数的问题,特别是可能与之前提到的串口回环代码相关的情况。首先,我应该回顾用户之前的提问历史,确保这次回答符合之前的上下文。 用户之前询问的是用C语言写串口回环的代码,我给出了示例代码,其中main函数是没有参数的。现在用户想了解如何在main函数中使用命令行参数,可能与串口程序结合,比如指定串口设备路径或波特率。这应该需要修改之前的代码,让main接收参数,而不是硬编码SERIAL_PORT和波特率。 接下来,我需要回忆C语言中main函数的参数机制。标准的main函数有两种形式:int main(void)和int main(int argc, char *argv[])。argc是参数个数,argv是参数数组,其中argv[0]是程序名,argv[1]开始是用户输入的参数。例如,用户可能希望运行程序时通过命令行指定串口设备和波特率,如./program /dev/ttyS0 9600,这样程序就能动态读取这些参数,而不是在代码中写死。 然后,我需要考虑如何将命令行参数应用到之前的串口回环代码中。原来的代码里,串口设备路径是宏定义的,波特率也是固定的。用户可能需要将这些值改为从命令行参数获取。所以示例代码需要修改main函数,检查参数数量,解析参数,并应用到配置中。比如,第一个参数是设备路径,第二个是波特率,但波特率可能需要从字符串转换为对应的B9600这样的常量,这需要处理。 还要注意错误处理,比如用户输入的参数数量不足,或者波特率不支持的情况。例如,如果用户输入了不支持的波特率值,程序应该提示错误并退出。此外,在之前的代码中,波特率设置使用了cfsetispeed和cfsetospeed,需要将用户输入的字符串转换为对应的B常量,比如“9600”对应B9600,可能需要使用条件判断或者查找表来处理。 另外,用户提到的引用中有关于异步读写和TCP通信的内容,但当前问题主要涉及命令行参数和串口配置,可能不需要涉及异步部分,但需要确保新的代码示例仍然保持串口回环的核心功能,只是增加了参数处理。 现在需要构建一个示例代码,展示如何在main函数中使用argc和argv来接收命令行参数,并将这些参数应用到串口配置中。例如,程序启动时接收设备路径和波特率作为参数,然后打开对应的串口并进行配置。同时,代码中需要包含参数检查和错误处理的逻辑,比如参数不足时提示用法,无效波特率时提示错误。 可能还需要解释每个步骤,比如解析参数的过程,如何将字符串转换为波特率常量,以及如何在termios结构中使用这些参数。此外,要确保代码符合之前的格式,比如使用正确的系统调用,配置串口的步骤正确,并且回环处理部分保持不变。 最后,生成相关问题时要考虑用户可能进一步的问题,比如如何处理更多参数,如何验证参数的有效性,或者如何在其他平台(如Windows)上处理命令行参数和串口配置。这样可以帮助用户深入理解相关知识点。</think>以下是在C语言中为`main`函数添加命令行参数的实现方法,结合串口回环程序的改进示例: ```c #include <stdio.h> #include <fcntl.h> #include <termios.h> #include <unistd.h> #include <stdlib.h> // 新增atoi() int main(int argc, char *argv[]) { // 参数验证 if(argc < 3) { printf("用法: %s <串口设备> <波特率>\n示例: %s /dev/ttyUSB0 115200\n", argv[0], argv[0]); return -1; } // 解析波特率 speed_t baud_rate; switch(atoi(argv[2])) { // 字符串转整数 case 9600: baud_rate = B9600; break; case 115200: baud_rate = B115200;break; default: printf("不支持的波特率,可选值:9600/115200\n"); return -1; } // 打开串口 int fd = open(argv[1], O_RDWR | O_NOCTTY); if(fd < 0) { perror("打开串口失败"); return -1; } // 配置串口(保持核心配置不变) struct termios tty; tcgetattr(fd, &tty); cfsetispeed(&tty, baud_rate); // 使用参数配置波特率 cfsetospeed(&tty, baud_rate); tty.c_cflag &= ~PARENB; tty.c_cflag &= ~CSTOPB; tty.c_cflag |= CS8; /* 其余配置保持原有逻辑不变 */ // 回环处理逻辑(同前) char buf[255]; while(1) { ssize_t n = read(fd, buf, sizeof(buf)); if(n > 0) write(fd, buf, n); } close(fd); return 0; } ``` 关键改进点说明: 1. `main`函数参数处理: - `argc`:参数数量($≥1$) - `argv[]`:参数值数组,其中: - `argv[1]`:动态指定串口设备路径(如`/dev/ttyUSB0`) - `argv[2]`:波特率数值(如`115200`) 2. 参数有效性验证: - 检查最小参数数量`argc < 3`时提示用法 - 通过`switch`语句实现波特率映射,支持扩展更多速率[^1] 3. 与原有程序的整合: - 将硬编码的`SERIAL_PORT`替换为`argv[1]` - 波特率配置改用动态参数`baud_rate`
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值