C/C++语言误区之:void main( )

部署运行你感兴趣的模型镜像

很多人都使用了void main()其实这是错误的。 C/C++ 中从来没有定义过void main() C++ 之父 Bjarne Stroustrup 在他的主页上的 FAQ 中明确地写着:The definition void main( ) { /* ... */ } is not and never has been C++, nor has it even been C. void main( ) 从来就不存在于 C++ 或者 C )。

下面我分别说一下 C C++ 标准中对 main 函数的定义。

 1. C

C89 中,main()是可以接受的。Brian W. Kernighan Dennis M. Ritchie 的经典巨著 The C programming Language 2e(C 程序设计语言第二版》)用的就是 main()不过在最新的 C99 标准中,只有以下两种定义方式是正确的:

int main( void )

int main( int argc, char *argv[] )

(参考资料:ISO/IEC 9899:1999 (E) Programming languages — C 5.1.2.2.1 Program startup)

当然,我们也可以做一点小小的改动。例如:char *argv[] 可以写成 char **argvargv argc 可以改成别的变量名(如 intval charval),不过一定要符合变量的命名规则。

如果不需要从命令行中获取参数,请用int main(void);否则请用int main( int argc, char *argv[] )

main 函数的返回值类型必须是 int ,这样返回值才能传递给程序的激活者(如操作系统)。

如果 main 函数的最后没有写 return 语句的话,C99 规定编译器要自动在生成的目标文件中(如 exe 文件)加入return 0; ,表示程序正常退出。不过,建议最好在main函数的最后加上return 语句,虽然没有这个必要,但这是一个好的习惯。

注意,vc6不会在目标文件中加入return 0;,大概因为 vc6 98 年的产品,所以才不支持这个特性。现在明白为什么建议你最好加上 return 语句了吧!不过,gcc3.2Linux 下的 C 编译器)会在生成的目标文件中加入 return 0;

2. C++

C++98 中定义了如下两种 main 函数的定义方式:

int main( )

int main( int argc, char *argv[] )

(参考资料:ISO/IEC 14882(1998-9-01)Programming languages — C++ 3.6 Start and termination)

int main( ) 等同于 C99 中的 int main( void ) ;

int main( int argc, char *argv[] ) 的用法也和 C99 中定义的一样.

同样,main 函数的返回值类型也必须是int。如果main函数的末尾没写return语句,C++98 规定编译器要自动在生成的目标文件中加入 return 0; 。同样,vc6 也不支持这个特性,但是 g++3.2Linux 下的 C++ 编译器)支持。

 

3. 关于 void main

     C++ 中,不接收任何参数也不返回任何信息的函数原型为“void foo(void);”。可能正是因为这个,所以很多人都误认为如果不需要程序返回值时可以把main函数定义成void main(void) 。然而这是错误的!main 函数的返回值应该定义为 int 类型,C C++ 标准中都是这样规定的。虽然在一些编译器中,void main 可以通过编译(如 vc6),但并非所有编译器都支持 void main ,因为标准中从来没有定义过 void main g++3.2 中如果 main 函数的返回值不是 int 类型,就根本通不过编译。而 gcc3.2 则会发出警告。所以,如果你想你的程序拥有很好的可移植性,请一定要用 int main

 

4. 返回值的作用

main 函数的返回值用于说明程序的退出状态。如果返回 0,则代表程序正常退出,否则代表程序异常退出。

下面我们在 winxp 环境下做一个小实验。首先编译下面的程序:

int main( void )

{ return 0; }

然后打开附件里的命令提示符,在命令行里运行刚才编译好的可执行文件,然后输入“echo %ERRORLEVEL%”,回车,就可以看到程序的返回值为 0 。假设刚才编译好的文件是 a.exe ,如果输入“a && dir”,则会列出当前目录下的文件夹和文件。但是如果改成“return -1”,或者别的非 0 值,重新编译后输入“a && dir”,则 dir 不会执行。因为 && 的含义是:如果 && 前面的程序正常退出,则继续执行 && 后面的程序,否则不执行。也就是说,利用程序的返回值,我们可以控制要不要执行下一个程序。这就是 int main 的好处。如果你有兴趣,也可以把 main 函数的返回值类型改成非 int 类型(如 float),重新编译后执行“a && dir”,看看会出现什么情况,想想为什么会出现那样的情况。顺便提一下,如果输入 a || dir 的话,则表示如果 a 异常退出,则执行 dir

 

5. 那么 int main( int argc, char *argv[], char *envp[] ) 呢?

这当然也不是标准 C 里面定义的东西!char *envp[] 是某些编译器提供的扩展功能,用于获取系统的环境变量。因为不是标准,所以并非所有编译器都支持,故而移植性差,不推荐使用。

您可能感兴趣的与本文相关的镜像

Stable-Diffusion-3.5

Stable-Diffusion-3.5

图片生成
Stable-Diffusion

Stable Diffusion 3.5 (SD 3.5) 是由 Stability AI 推出的新一代文本到图像生成模型,相比 3.0 版本,它提升了图像质量、运行速度和硬件效率

你提到的问题: > 代码有问题,无法访问 `b` 这是一个非常常见的误解,尤其是在涉及**友元函数**和**类定义顺序**时。 --- ## ❌ 错误原因分析: 你的代码如下: ```cpp class prt; // 前向声明 class rum { friend void prt::mun(); // ❌ 问题就出在这里 public: rum() { a = 10; b = 20; } int a; private: int b; }; class prt { public: void mun(); private: rum* p; }; ``` --- ## 🧠 核心问题:友元函数声明时机不对 你写的是: ```cpp friend void prt::mun(); ``` 但此时 `prt` 类虽然前向声明了,但它的成员函数 `mun()` **尚未声明**,所以编译器根本不知道 `prt::mun()` 是什么。 > **C++ 规则**:你不能将一个尚未声明的成员函数声明为友元。 --- ## ✅ 正确做法(关键): 要让 `rum` 类能够将 `prt::mun()` 声明为友元函数,你必须: 1. **先完整声明 `prt` 类**,至少要声明 `void mun();` 2. 然后再定义 `rum` 类,并在其中声明 `friend void prt::mun();` --- ## ✅ 正确代码如下: ```cpp #include <iostream> using namespace std; // 第一步:前向声明 + 声明 mun 函数原型 class prt { public: void mun(); // 提前声明 mun 函数 }; // 第二步:定义 rum 类,并将 prt::mun() 声明为友元 class rum { // 此时 prt::mun() 已经声明,可以作为友元 friend void prt::mun(); public: rum() { a = 10; b = 20; } int a; private: int b; }; // 第三步:完整定义 prt 类 class prt { public: prt() { p = new rum(); } ~prt() { delete p; } void mun(); // 这是之前声明的 mun() private: rum* p; }; // 第四步:实现 mun 函数 void prt::mun() { cout << p->a << endl; // 公有成员 cout << p->b << endl; // 私有成员,因为是友元函数可以访问 ✅ } // 第五步:主函数 int main() { prt p; p.mun(); return 0; } ``` --- ## ✅ 代码解释: | 部分 | 说明 | |------|------| | `class prt { void mun(); };` | 提前声明 `mun()`,确保友元声明可用 | | `friend void prt::mun();` | 在 `rum` 类中合法声明友元 | | `void prt::mun()` | 实现函数,可以访问 `rum` 的私有成员 `b` | | `cout << p->b << endl;` | ✅ 正确,因为 `mun()` 是 `rum` 的友元函数 | --- ## ❗常见误区总结: | 错误写法 | 原因 | 正确做法 | |----------|------|----------| | `friend void prt::mun();` 但 `prt` 未完整定义 | 编译器不知道 `prt::mun` 是否存在 | 提前在 `prt` 类中声明 `void mun();` | | 使用 `new rum()` 但 `rum` 未定义 | 不完整类型错误 | 调整类定义顺序,确保 `rum` 已定义 | | 尝试访问私有成员 `b` | 不是友元或成员函数 | 确保函数被正确声明为友元 | --- ## ✅ 最终输出: 运行结果应为: ``` 10 20 ``` --- ##
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值