浅谈mian函数的参数解析

我们从一开始接触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函数的标准形式是int main(int argc, char* argv[]),其中argv是一个字符串数组,用于传递命令行参数。这些参数都是字符串类型的,所以直接传递结构体显然不可能。 用户可能想要在运行程序时通过命令行输入结构体的各个成员值,然后在程序中解析这些值并填充到结构体中。比如,用户有一个结构体struct job {char name[40]; double salary; int floor;};,他们希望执行程序时传入name、salary和floor的值,然后在程序内部将这些字符串转换为结构体的相应类型。 首先,我需要确定如何从命令行参数中提取这些值。比如,假设用户运行程序时输入:./program John 50000.50 3。这里的三个参数对应结构体的三个成员。程序需要将argv[1]作为name,argv[2]转换为double类型的salary,argv[3]转换为int类型的floor。 接下来,需要考虑参数的数量是否正确。argc表示参数的数量,包括程序名本身。所以,如果结构体有三个成员,用户需要输入三个参数,此时argc应该是4(包括程序名)。如果参数数量不足,需要处理错误情况,比如抛出异常或提示用法。 然后,处理类型转换的问题。例如,将字符串转换为double和int可能会失败,需要检查转换是否成功。可以使用std::stod和std::stoi函数,但要注意这些函数可能抛出异常,需要进行异常处理,或者使用其他方法验证输入的合法性。 另外,结构体的name成员是一个字符数组,直接使用argv[1]可能存在缓冲区溢出的风险。因此,应该使用安全的方法复制字符串,比如strncpy,并确保末尾有终止符。 还需要考虑不同的命令行参数格式,比如用户可能希望使用标志来指定各个参数,例如:./program --name John --salary 50000.50 --floor 3。这种情况下,参数解析会更复杂,需要遍历argv,识别每个标志对应的值。这种方法虽然更灵活,但解析起来也更麻烦,可能需要使用库如getopt或第三方命令行解析库。 不过,用户的问题可能更倾向于简单的按顺序传递参数,而不是使用标志。因此,先处理简单的情况,再提到复杂情况的可能性。 总结步骤应该是:1. 检查参数数量是否正确;2. 解析每个参数到对应的结构体成员;3. 处理可能的转换错误;4. 填充结构体实例并使用。 另外,需要提醒用户,main函数只能接受字符串参数,结构体必须被分解为基本类型的成员,然后在程序中重新组合。如果需要传递复杂的嵌套结构体或大量数据,可能需要使用其他方法,如配置文件或序列化库,而不是通过命令行参数。 最后,给出示例代码,演示如何实现这个过程,包括错误检查和类型转换。同时,可以建议使用更高级的命令行解析库来简化复杂情况下的处理,比如Boost.Program_options或CLI11等。</think>要将结构体参数传递到C++程序的`main`函数中,需要通过命令行参数解析实现。由于`main`函数的标准形式为`int main(int argc, char* argv[])`,所有参数均为字符串,因此需要将结构体成员拆解为基本类型参数后手动解析[^2]。 ### 实现步骤 1. **定义结构体** 例如定义表示职位的结构体: ```cpp struct job { char name[40]; double salary; int floor; }; ``` 2. **命令行参数传递** 运行程序时按顺序传递参数,例如: ```bash ./program "Engineer" 85000.50 15 ``` 参数顺序对应结构体成员`name`, `salary`, `floor`。 3. **参数解析与类型转换** 在代码中实现参数解析: ```cpp #include <cstring> #include <cstdlib> job parse_arguments(int argc, char* argv[]) { if (argc != 4) { std::cerr << "Usage: " << argv[0] << " <name> <salary> <floor>\n"; exit(EXIT_FAILURE); } job result; strncpy(result.name, argv[1], sizeof(result.name) - 1); // 安全拷贝 result.salary = std::stod(argv[2]); // 字符串转double result.floor = std::stoi(argv[3]); // 字符串转int return result; } int main(int argc, char* argv[]) { job my_job = parse_arguments(argc, argv); // 使用my_job结构体 } ``` 4. **错误处理** - 参数数量校验:需严格匹配`argc == 4`(程序名+3个参数) - 类型转换异常处理:若参数无法转换为目标类型(如非数字输入),`std::stod`/`std::stoi`会抛出`std::invalid_argument`异常 ### 高级应用 - **标志参数解析**:使用`getopt`或第三方库(如Boost.Program_options)处理带标志的参数: ```bash ./program --name Engineer --salary 85000.50 --floor 15 ``` - **配置文件**:若参数复杂,可将结构体写入JSON/XML文件,通过文件路径参数传递[^3]
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值