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 子目录名

你可能感兴趣的:(学习笔记)