在上一篇文章中,我们说到,C 语言系统应该由程序开发环境,C 语言本身和 C 语言的库组成。且同时说了程序开发环境做了“编写”,“预处理”,“编译”和“链接”这几件事情。但是细节并没有一一呈现。不知道同学们是否想过,这里都经历了一些什么呢?!
在这一篇文章中,我们就来看看“上帝说,要有光,就有了光。”是怎么实现的。假设你是上帝,想要实现“给我输出 ‘HELLO,WORLD!‘”。同学们使用的 IDE 是 C-Free 5,这里的所有操作,都以此编译器为例。假设编译器安装路径为“C:\Program Files\C-Free 5”。
1. 代码编写
同学们已经上过第一节课了,我想,编写一个“HELLO,WORLD!”一定不成问题。代码本身是这样的:
1 #include <stdio.h> 2 3 int main(void) 4 { 5 printf("HELLO,WORLD!\n"); 6 return 0; 7 }
将此代码保存为“HelloWord.c”后准备下一步的处理。
2. 预处理
在保存了“HelloWord.c”文件夹中打开一个“cmd”窗口(按住 Shift 后右击选择“在此处打开命令窗口”),输入:
gcc -E HelloWorld.c -o HelloWorld.i # 命令解释: # 在 C-Free 5 身后,默默工作的就是这个名叫 gcc 的编译器了。 # 当输入上面的命令,紧跟其后的“-E”告知编译器想要做的是动作是仅做预处理这个动作。 # 然后后面跟随的“.c”文件告知 gcc 需要处理的文件的名称 # -o HelloWorld.i,是告诉编译器,输出一个文件,文件名就是“HelloWorld.i” # (你看,计算机就是一个输入然后输出的系统吧)
你会发现,在此目录下生成了一个名为“HelloWorld.i”的文件,打开文件,观察内容你会发现是这样的:
1 # 1 "HelloWorld.c" 2 # 1 "<built-in>" 3 # 1 "<command line>" 4 # 1 "HelloWorld.c" 5 # 1 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 1 3 6 # 19 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 3 7 # 1 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/_mingw.h" 1 3 8 # 27 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/_mingw.h" 3 9 10 # 28 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/_mingw.h" 3 11 # 20 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 2 3 12 13 14 15 16 17 18 # 1 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stddef.h" 1 3 19 20 21 22 23 24 # 1 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/include/stddef.h" 1 3 4 25 # 213 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/include/stddef.h" 3 4 26 typedef unsigned int size_t; 27 # 325 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/include/stddef.h" 3 4 28 typedef short unsigned int wchar_t; 29 # 354 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/include/stddef.h" 3 4 30 typedef short unsigned int wint_t; 31 # 7 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stddef.h" 2 3 32 # 27 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 2 3 33 34 # 1 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdarg.h" 1 3 35 36 37 38 39 40 # 1 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/include/stdarg.h" 1 3 4 41 # 44 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/include/stdarg.h" 3 4 42 typedef __builtin_va_list __gnuc_va_list; 43 # 7 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdarg.h" 2 3 44 # 29 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 2 3 45 # 129 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 3 46 typedef struct _iobuf 47 { 48 char* _ptr; 49 int _cnt; 50 char* _base; 51 int _flag; 52 int _file; 53 int _charbuf; 54 int _bufsiz; 55 char* _tmpfname; 56 } FILE; 57 # 154 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 3 58 extern __attribute__ ((__dllimport__)) FILE _iob[]; 59 # 169 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 3 60 FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fopen (const char*, const char*); 61 FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) freopen (const char*, const char*, FILE*); 62 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fflush (FILE*); 63 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fclose (FILE*); 64 65 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) remove (const char*); 66 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) rename (const char*, const char*); 67 FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) tmpfile (void); 68 char* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) tmpnam (char*); 69 70 71 char* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _tempnam (const char*, const char*); 72 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _rmtmp(void); 73 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _unlink (const char*); 74 75 76 char* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) tempnam (const char*, const char*); 77 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) rmtmp(void); 78 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) unlink (const char*); 79 80 81 82 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) setvbuf (FILE*, char*, int, size_t); 83 84 void __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) setbuf (FILE*, char*); 85 86 87 88 89 90 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fprintf (FILE*, const char*, ...); 91 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) printf (const char*, ...); 92 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) sprintf (char*, const char*, ...); 93 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _snprintf (char*, size_t, const char*, ...); 94 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) vfprintf (FILE*, const char*, __gnuc_va_list); 95 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) vprintf (const char*, __gnuc_va_list); 96 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) vsprintf (char*, const char*, __gnuc_va_list); 97 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _vsnprintf (char*, size_t, const char*, __gnuc_va_list); 98 # 216 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 3 99 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) snprintf(char *, size_t, const char *, ...); 100 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) vsnprintf (char *, size_t, const char *, __gnuc_va_list); 101 102 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) vscanf (const char * __restrict__, __gnuc_va_list); 103 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) vfscanf (FILE * __restrict__, const char * __restrict__, 104 __gnuc_va_list); 105 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) vsscanf (const char * __restrict__, 106 const char * __restrict__, __gnuc_va_list); 107 108 109 110 111 112 113 114 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fscanf (FILE*, const char*, ...); 115 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) scanf (const char*, ...); 116 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) sscanf (const char*, const char*, ...); 117 118 119 120 121 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fgetc (FILE*); 122 char* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fgets (char*, int, FILE*); 123 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fputc (int, FILE*); 124 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fputs (const char*, FILE*); 125 char* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) gets (char*); 126 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) puts (const char*); 127 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) ungetc (int, FILE*); 128 129 130 131 132 133 134 135 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _filbuf (FILE*); 136 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _flsbuf (int, FILE*); 137 138 139 140 extern __inline__ int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) getc (FILE* __F) 141 { 142 return (--__F->_cnt >= 0) 143 ? (int) (unsigned char) *__F->_ptr++ 144 : _filbuf (__F); 145 } 146 147 extern __inline__ int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) putc (int __c, FILE* __F) 148 { 149 return (--__F->_cnt >= 0) 150 ? (int) (unsigned char) (*__F->_ptr++ = (char)__c) 151 : _flsbuf (__c, __F); 152 } 153 154 extern __inline__ int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) getchar (void) 155 { 156 return (--(&_iob[0])->_cnt >= 0) 157 ? (int) (unsigned char) *(&_iob[0])->_ptr++ 158 : _filbuf ((&_iob[0])); 159 } 160 161 extern __inline__ int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) putchar(int __c) 162 { 163 return (--(&_iob[1])->_cnt >= 0) 164 ? (int) (unsigned char) (*(&_iob[1])->_ptr++ = (char)__c) 165 : _flsbuf (__c, (&_iob[1]));} 166 # 297 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 3 167 size_t __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fread (void*, size_t, size_t, FILE*); 168 size_t __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fwrite (const void*, size_t, size_t, FILE*); 169 170 171 172 173 174 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fseek (FILE*, long, int); 175 long __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) ftell (FILE*); 176 void __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) rewind (FILE*); 177 # 340 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 3 178 typedef long long fpos_t; 179 180 181 182 183 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fgetpos (FILE*, fpos_t*); 184 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fsetpos (FILE*, const fpos_t*); 185 186 187 188 189 190 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) feof (FILE*); 191 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) ferror (FILE*); 192 # 365 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 3 193 void __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) clearerr (FILE*); 194 void __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) perror (const char*); 195 196 197 198 199 200 201 FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _popen (const char*, const char*); 202 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _pclose (FILE*); 203 204 205 FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) popen (const char*, const char*); 206 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) pclose (FILE*); 207 208 209 210 211 212 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _flushall (void); 213 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _fgetchar (void); 214 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _fputchar (int); 215 FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _fdopen (int, const char*); 216 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _fileno (FILE*); 217 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _fcloseall(void); 218 FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _fsopen(const char*, const char*, int); 219 220 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _getmaxstdio(void); 221 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _setmaxstdio(int); 222 # 402 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 3 223 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fgetchar (void); 224 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fputchar (int); 225 FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fdopen (int, const char*); 226 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fileno (FILE*); 227 # 414 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 3 228 # 1 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/sys/types.h" 1 3 229 # 21 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/sys/types.h" 3 230 # 1 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stddef.h" 1 3 231 232 233 234 235 236 # 1 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/include/stddef.h" 1 3 4 237 # 151 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/include/stddef.h" 3 4 238 typedef int ptrdiff_t; 239 # 7 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stddef.h" 2 3 240 # 22 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/sys/types.h" 2 3 241 242 243 244 245 246 typedef long time_t; 247 248 249 250 251 typedef long long __time64_t; 252 253 254 255 256 257 typedef long _off_t; 258 259 260 typedef _off_t off_t; 261 262 263 264 265 266 267 268 typedef unsigned int _dev_t; 269 270 271 272 273 274 typedef _dev_t dev_t; 275 276 277 278 279 280 281 typedef short _ino_t; 282 283 284 typedef _ino_t ino_t; 285 286 287 288 289 290 291 typedef int _pid_t; 292 293 294 typedef _pid_t pid_t; 295 296 297 298 299 300 301 typedef unsigned short _mode_t; 302 303 304 typedef _mode_t mode_t; 305 306 307 308 309 310 311 typedef int _sigset_t; 312 313 314 typedef _sigset_t sigset_t; 315 316 317 318 319 320 typedef long _ssize_t; 321 322 323 typedef _ssize_t ssize_t; 324 325 326 327 328 329 typedef long long fpos64_t; 330 331 332 333 334 typedef long long off64_t; 335 # 415 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 2 3 336 extern __inline__ FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fopen64 (const char* filename, const char* mode) 337 { 338 return fopen (filename, mode); 339 } 340 341 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fseeko64 (FILE*, off64_t, int); 342 343 344 345 346 347 348 extern __inline__ off64_t __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) ftello64 (FILE * stream) 349 { 350 fpos_t pos; 351 if (fgetpos(stream, &pos)) 352 return -1LL; 353 else 354 return ((off64_t) pos); 355 } 356 # 443 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 3 357 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fwprintf (FILE*, const wchar_t*, ...); 358 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) wprintf (const wchar_t*, ...); 359 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) swprintf (wchar_t*, const wchar_t*, ...); 360 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _snwprintf (wchar_t*, size_t, const wchar_t*, ...); 361 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) vfwprintf (FILE*, const wchar_t*, __gnuc_va_list); 362 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) vwprintf (const wchar_t*, __gnuc_va_list); 363 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) vswprintf (wchar_t*, const wchar_t*, __gnuc_va_list); 364 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _vsnwprintf (wchar_t*, size_t, const wchar_t*, __gnuc_va_list); 365 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fwscanf (FILE*, const wchar_t*, ...); 366 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) wscanf (const wchar_t*, ...); 367 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) swscanf (const wchar_t*, const wchar_t*, ...); 368 wint_t __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fgetwc (FILE*); 369 wint_t __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fputwc (wchar_t, FILE*); 370 wint_t __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) ungetwc (wchar_t, FILE*); 371 372 373 wchar_t* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fgetws (wchar_t*, int, FILE*); 374 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fputws (const wchar_t*, FILE*); 375 wint_t __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) getwc (FILE*); 376 wint_t __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) getwchar (void); 377 wchar_t* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _getws (wchar_t*); 378 wint_t __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) putwc (wint_t, FILE*); 379 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _putws (const wchar_t*); 380 wint_t __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) putwchar (wint_t); 381 FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _wfdopen(int, wchar_t *); 382 FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _wfopen (const wchar_t*, const wchar_t*); 383 FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _wfreopen (const wchar_t*, const wchar_t*, FILE*); 384 FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _wfsopen (const wchar_t*, const wchar_t*, int); 385 wchar_t* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _wtmpnam (wchar_t*); 386 wchar_t* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _wtempnam (const wchar_t*, const wchar_t*); 387 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _wrename (const wchar_t*, const wchar_t*); 388 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _wremove (const wchar_t*); 389 void __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _wperror (const wchar_t*); 390 FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _wpopen (const wchar_t*, const wchar_t*); 391 392 393 394 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) snwprintf (wchar_t* s, size_t n, const wchar_t* format, ...); 395 extern __inline__ int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) 396 vsnwprintf (wchar_t* s, size_t n, const wchar_t* format, __gnuc_va_list arg) 397 { return _vsnwprintf ( s, n, format, arg);} 398 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) vwscanf (const wchar_t * __restrict__, __gnuc_va_list); 399 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) vfwscanf (FILE * __restrict__, 400 const wchar_t * __restrict__, __gnuc_va_list); 401 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) vswscanf (const wchar_t * __restrict__, 402 const wchar_t * __restrict__, __gnuc_va_list); 403 # 497 "C:/Program Files (x86)/C-Free 5/mingw/bin/../lib/gcc/mingw32/3.4.5/../../../../include/stdio.h" 3 404 FILE* __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) wpopen (const wchar_t*, const wchar_t*); 405 406 407 408 409 410 411 wint_t __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _fgetwchar (void); 412 wint_t __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _fputwchar (wint_t); 413 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _getw (FILE*); 414 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) _putw (int, FILE*); 415 416 417 wint_t __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fgetwchar (void); 418 wint_t __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) fputwchar (wint_t); 419 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) getw (FILE*); 420 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) putw (int, FILE*); 421 # 2 "HelloWorld.c" 2 422 423 int main(void) 424 { 425 printf("HELLO,WORLD!\n"); 426 return 0; 427 }
拖动到最底部,才能够发现我们最开始写的那几行代码。但是代码又和我们之前写的似而不同:
1 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) getw (FILE*); 2 int __attribute__((__cdecl__)) __attribute__ ((__nothrow__)) putw (int, FILE*); 3 # 2 "HelloWorld.c" 2 4 5 int main(void) 6 { 7 printf("HELLO,WORLD!\n"); 8 return 0; 9 }
可以发现,第一行的 #include <stdio.h> 却再也找不到了。其实,这就是预处理的功能,它会将你的代码中的 #include <XXXXXXX.h> 全部替换到,为下面的编译做准备。
需要注意的是,预处理阶段,编译器 gcc 是不做语法上的检查的,比如我们写一段貌合神离的代码:
1 #include <stdio.h> 2 3 printf("HELLO,WORLD!\n");
然后重新执行上面的命令,你会发现,依旧会生成一个看似可用的“.i”文件。
3. 编译
编译阶段,编译器 gcc 将会对之前预处理生成的“.i”文件进行语法上的检查,看其是否符合 C 语言的语法规范,当检查通过以后,就进行编译的过程。这里,将其过程分为两步来看,首先,将预处理文件翻译为汇编文件:
1 gcc -S HelloWorld.i -o helloWorld.s
然后将汇编文件编译为目标文件:
1 gcc -c HelloWorld.s -o HelloWorld.obj
这里我们再来试试之前的“李鬼”代码,看看它能否通过流程,生成汇编文件:
1 gcc -S HelloWorld.i -o helloWorld.s 2 HelloWorldError.c:3: error: syntax error before string constant 3 HelloWorldError.c:3: error: conflicting types for ‘printf‘ 4 HelloWorldError.c:3: note: a parameter list with an ellipsis can‘t match an empty parameter name list declaration 5 HelloWorldError.c:3: error: conflicting types for ‘printf‘ 6 HelloWorldError.c:3: note: a parameter list with an ellipsis can‘t match an empty parameter name list declaration 7 HelloWorldError.c:3: warning: data definition has no type or storage class
哎呦,出了一堆的错误唉,有此可见,编译过程是需要执行内容的检查的,以确定其符合 C 语言的语法。
4. 链接
在上一步中,生成的目标文件“.obj”虽然本身也是二进制文件,但是其本身是不可执行的,需要对其进行链接。
1 gcc helloworld.obj -o helloworld.exe
到这里,就生成可以执行的二进制文件了,赶快在终端中输入“HelloWorld”尝试运行吧:
虽然同学们可能直接在 C-Free 中仅需要点击一下按钮,后台就已经给你做了以上的这些事情,但是这里还是希望同学们能够理解这一过程(链接这一过程涉及到 C 语言库函数的事情,大家可以再扩展阅读哦)。