c语言学习笔记十四



多目标文件的链接


示例代码如下:
test18.c
char stack[512];
int top=-1;
void push(char c){
stack[++top]=c;
}
char pop(){
return stack[top--];
}
int is_empty(void){
return top==-1;
}
 


test19.c


#include <stdio.h>
int a,b=1;
int main(void){
  push('a');
push('b');
push('c');


       while(!is_empty()){
putchar(pop());
}
putchar("\n");
return 0;
}




一起编译: gcc test19.c test18.c -o test19
分开编译: gcc -c test19.c
           gcc -c test18.c
           gcc test19.o test18.o -o test19


运行程序:./test19
查看符号表:readlf -a test19


以下是合并后的符号表信息 
  [13] .text       PROGBITS   08048330 000330 00021c 00  AX  0   0 16
      52: 08048481    19 FUNC    GLOBAL DEFAULT   13 is_empty
      54: 08048466    27 FUNC    GLOBAL DEFAULT   13 pop
      68: 080483e4    88 FUNC    GLOBAL DEFAULT   13 main
      69: 0804843c    42 FUNC    GLOBAL DEFAULT   13 push


  [24] .data       PROGBITS   0804a00c 00100c 000010 00  WA  0   0  4
      49: 0804a014     4 OBJECT  GLOBAL DEFAULT   24 b
      59: 0804a018     4 OBJECT  GLOBAL DEFAULT   24 top


  [25] .bss        NOBITS     0804a020 00101c 000220 00  WA  0   0 32
      66: 0804a028     4 OBJECT  GLOBAL DEFAULT   25 a
      71: 0804a040   512 OBJECT  GLOBAL DEFAULT   25 stack




查看默认链接角本:ld --verbose
 
 示例代码如下:
   ENTRY(_start) //整个程序的入口点,可以修改为其他函数


   PROVIDE (__executable_start = SEGMENT_START("text-segment", 


0x08048000)); . = SEGMENT_START("text-segment", 0x08048000) + 


SIZEOF_HEADERS;   //text Segment的起始地址, 每个段的格式,段名:{组成}




定义和声名
 示例代码:
test19.c


#include <stdio.h>
int a,b=1;
int main(void){
  push('a');
push('b');
push('c');


       while(!is_empty()){
putchar(pop());
}
putchar("\n");
return 0;
}




以上函数用以下命令编译会出问题
  gcc -c test19.c -wall
  错误信息:  


 test19.c: 在函数‘main’中:
 test19.c:4: 警告: 隐式声明函数‘push’
 test19.c:8: 警告: 隐式声明函数‘is_empty’
 test19.c:9: 警告: 隐式声明函数‘pop’
 
  
 函数声明没有明确指出参数类型和个数,不算函数原型,所以编译器只能用作隐式声





正确方法:


 #include <stdio.h>
/*声明用到的几个函数的原型 
 *Extern 关键字表示这个标识符具有External Linkage.
 * 外部链接:external linkage 多个文件中声明多次也都代表同一标识符
 * static 作函数声明,
 * 内部链接:internal linkage (某个文件)中声明多次也都代表同一标识符
 */
extern void push(char);
extern char pop(void);
extern int is_empty(void);
/*
 *如果相直接访问test18.c上的top属性
 *有点像java中的子类的味道extends 
 *它的存储空间是在test18.c中分配
 * extern在作函数声明时可写不可写都行
 * 但在变量声明时必需要写
 */
extern int top; 


int a,b=1;
int main(void){
  push('a');
push('b');
push('c');


       while(!is_empty()){
putchar(pop());
}
putchar('\n');
return 0;
}




示例代码:
test18.c
char stack[512];
/* 
 *如果不想让外界访问可以用static来修饰
 *有点像java 中的private修饰符
 */
int top=-1;
void push(char c){
stack[++top]=c;
}
char pop(){
return stack[top--];
}
int is_empty(void){
return top==-1;
}






头文件
 为了避免重复的代码(即在test.19中用作函数声明的代码)
  可以将其抽象到一个头文件中(有点java的味道)
 
示例代码如下:


test18.h头文件


/*
 * 此处定义的STACK_H这个宏是为了防止头文件被得复包含
 */
#ifndef STACK_H


#define STACK_H
extern void push(char);
extern char pop(void);
extern int is_empty(void);
#endihile(!iq






test19.c文件修改为:
#include <stdio.h>
/*
 * <头文件> 直接从库文件中查找
 * "头文件" 先从当前目录查找,然后到库目录中查询
 * 这跟c++中有点类似
 * 如果头文件放在子录中s
 * 可以用如下格式: #include "子目录名/test18.h"
 */
#include "test18.h"
 


int a,b=1;
int main(void){
  push('a');
push('b');
push('c');


       while(!is_empty()){
putchar(pop());
}
putchar('\n');
return 0;
}


注:如果头文件放到下一子目录中,需要用-I 参数指定目录名
     命令格如下
     gcc -c test19.c -I  子目录名
    

你可能感兴趣的:(java,c,object,gcc,语言,编译器)