我在编译我写的c++程序时遇到了一些困难。

这个程序非常简单,据我所知,它符合c++标准中规定的所有规则。我已经把ISO/IEC 14882:2003通读了两遍。

程序如下:

下面是我在尝试用Visual c++ 2010编译这个程序时收到的输出:

c:\dev>cl /nologo helloworld.png
cl : Command line warning D9024 : unrecognized source file type 'helloworld.png', object file assumed
helloworld.png : fatal error LNK1107: invalid or corrupt file: cannot read at 0x5172

沮丧地,我尝试了g++ 4.5.2,但它同样没有帮助:

c:\dev>g++ helloworld.png
helloworld.png: file not recognized: File format not recognized
collect2: ld returned 1 exit status

我认为Clang(3.0版本trunk 127530)一定可以工作,因为它的标准一致性受到了高度赞扬。不幸的是,它甚至没有给我一个漂亮的,突出显示的错误消息:

c:\dev>clang++ helloworld.png
helloworld.png: file not recognized: File format not recognized
collect2: ld returned 1 exit status
clang++: error: linker (via gcc) command failed with exit code 1 (use -v to see invocation)

老实说,我真的不知道这些错误消息是什么意思。

许多其他c++程序都有扩展名为.cpp的源文件,所以我想可能需要重命名我的文件。我把它的名字改为helloworld.cpp,但这并没有帮助。我认为Clang有一个非常严重的错误,因为当我试图用它来编译重命名的程序时,它跳出来,打印出“84个警告和20个错误生成”,让我的计算机发出很多哔哔声!

我哪里做错了?我是否错过了c++标准的某些关键部分?或者这三个编译器真的坏到不能编译这个简单的程序?


当前回答

在标准中,§2.1/1规定:

物理源文件字符在必要时以实现定义的方式映射到基本源字符集(为行结束指示符引入新行字符)。

你的编译器不支持这种格式(也就是不能将它映射到基本的源字符集),所以它不能进入进一步的处理阶段,因此出现了错误。你的编译器完全有可能支持从图像到基本源字符集的映射,但这不是必须的。

由于此映射是由实现定义的,因此需要查看实现文档以了解它支持的文件格式。通常,每个主要的编译器供应商都支持(规范定义的)文本文件:由文本编辑器生成的任何文件,通常是一系列字符。


注意,c++标准是基于C标准的(§1.1/2),C(99)标准在§1.2中说:

本国际标准没有规定 C程序被转换以供数据处理使用的机制 系统; C程序被数据处理调用的机制 系统; 输入数据转换为C程序使用的机制;

同样,源文件的处理方法需要在编译器文档中找到。

其他回答

在标准中,§2.1/1规定:

物理源文件字符在必要时以实现定义的方式映射到基本源字符集(为行结束指示符引入新行字符)。

你的编译器不支持这种格式(也就是不能将它映射到基本的源字符集),所以它不能进入进一步的处理阶段,因此出现了错误。你的编译器完全有可能支持从图像到基本源字符集的映射,但这不是必须的。

由于此映射是由实现定义的,因此需要查看实现文档以了解它支持的文件格式。通常,每个主要的编译器供应商都支持(规范定义的)文本文件:由文本编辑器生成的任何文件,通常是一系列字符。


注意,c++标准是基于C标准的(§1.1/2),C(99)标准在§1.2中说:

本国际标准没有规定 C程序被转换以供数据处理使用的机制 系统; C程序被数据处理调用的机制 系统; 输入数据转换为C程序使用的机制;

同样,源文件的处理方法需要在编译器文档中找到。

不幸的是,您选择了三个支持多种语言的编译器,而不仅仅是c++。他们都必须猜测你使用的编程语言。您可能已经知道,PNG格式适用于所有编程语言,而不仅仅是c++。

通常编译器可以自己找出语言。例如,如果PNG显然是用蜡笔绘制的,编译器就会知道它包含Visual Basic。如果它看起来像是用机械铅笔画的,那么很容易认出正在写FORTRAN代码的工程师。

在本例中,第二步对编译器也没有帮助。C和c++看起来太相似了,甚至连#include都不例外。因此,你必须帮助编译器决定它到底是什么语言。现在,你可以使用非标准方法。例如,Visual Studio编译器接受/TC和/TP命令行参数,或者你可以在项目文件中使用“Compile as: c++”选项。GCC和CLang有自己的机制,我不知道。

因此,我建议使用标准方法来告诉编译器下面的代码是用c++编写的。正如您现在所发现的,c++编译器对它们所接受的内容非常挑剔。因此,识别c++的标准方法是通过恐吓程序员添加到他们的c++代码。例如,下面这行代码会告诉编译器下面的代码是c++(编译时最好不要抱怨)。

// To the compiler: I know where you are installed. No funny games, capice?

OCR说:

N lml_�e <loJ+_e__}

.lnt Mk.,n ( ln+ _rSC Lhc_yh )
h_S_
_l

s_l . co__ <, " H llo uo/_d ! '` << s l� . ena_ .
TP__rn _ |
_|

公平地说,这真是太好了。

你忘了预处理器。试试这个:

pngtopnm helloworld.png | ocrad | g++ -x 'c++' -

我把你的程序从PNG转换成ASCII,但它还没有编译。供您参考,我确实尝试了线宽为100和250个字符,但都产生了类似的结果。

   `         `  .     `.      `         ...                                                         
   +:: ..-.. --.:`:. `-` .....:`../--`.. `-                                                         
           `      `       ````                                                                      
                                                                      `                             
   ` `` .`       ``    .`    `.               `` .      -``-          ..                            
   .`--`:`   :::.-``-. : ``.-`-  `-.-`:.-`    :-`/.-..` `    `-..`...- :                            
   .`         ` `    ` .`         ````:``  -                  ` ``-.`  `                            
   `-                                ..                           ``                                
    .       ` .`.           `   `    `. ` .  . `    .  `    . . .` .`  `      ` ``        ` `       
           `:`.`:` ` -..-`.`-  .-`-.    /.-/.-`.-.  -...-..`- :```   `-`-`  :`..`-` ` :`.`:`- `     
            ``  `       ```.      ``    ````    `       `     `        `    `         `   `   .     
            : -...`.- .` .:/ `                                                                      
    -       `             `` .                                                                      
    -`                                                                                              
    `