C 语言学习的第 03 课:你的 idea 是怎么变成能够执行的程序的

在上一篇文章中,我们说到,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 }
View Code

拖动到最底部,才能够发现我们最开始写的那几行代码。但是代码又和我们之前写的似而不同:

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 语言库函数的事情,大家可以再扩展阅读哦)。

posted @ 2016-08-28 12:57  东风唯笑  阅读(1742)  评论(0编辑  收藏  举报