C语言 超详细讲解链接器

 更新时间:2022年03月25日 15:03:25   作者:鹿九丸  
在C语言中,一个重要的思想就是分别编译,即若干个源程序能够在不一样的时候单独进行编译,而后在恰当的时候整合到一块儿。可是链接器通常是与C编译器分离的,链接器如何作到把若干个C源程序合并成一个总体呢,我们一起来看看

1 什么是链接器

典型的链接器把由编译器或汇编器生成的若干个目标模块,整合成一个被称为载入模块或可执行文件的实体–该实体能够被操作系统直接执行。

截图

链接器通常把目标模块看成是由一组外部对象组成的。每个外部对象代表着机器内存中的某个部分,并通过一个外部名称来识别。因此,==程序中的每个函数和每个外部变量,如果没有被声明为static,就都是一个外部对象。==某些C编译器会对静态函数和静态变量的名称做一定改变,将它们也作为外部对象。由于经过了“名称修饰”,因此它们不会与其它源程序文件中的同名函数或同名变量发生命名冲突。

2 声明与定义

extern int a;

上面的这段代码并不是对a的定义,而是说明a是一个外部整型变量。

注意:引入之后,假如引入的位置在函数之外,就相当于在那个位置定义了全局变量,同样遵循局部变量优先原则,如果引入位置在某个函数之内,就相当于是一个局部变量,作用域与那个地方定义的局部变量相类似,此处讨论声明周期没有任何意义。

int a;
extern int a;

上面的这两条语句既可以是在同一个源文件中,也可以位于程序的不同源文件之中。

==注意:每个外部变量只能定义一次。==如果外部变量的多个定义各指定一个初始值,例如:

int a = 7;

出现在一个源文件中,而

int a = 9;

出现在另一个源文件中,大多数系统都会拒绝接收该程序。但是,如果一个外部变量在多个源文件中定义却没有指定初始值,那么**某些系统会接受这个程序,而另外一些系统则不会接受。**所以,每个外部变量必须只定义一次。

3 命名冲突

3.1 命名冲突

如果在两个不同的源文件中都包括了定义

int a;

那么它要么表示程序错误(如果链接器进制外部变量重复命名的话),要么在两个源文件中共享a的同一个实例(无论两个源文件中的外部变量是否应该被共享)。即使其中a的一个定义是出现在系统提供的库文件中,也仍然进行同样的处理。

3.2 static修饰符

static int a;

static修饰a之后,a的作用域将被限制在一个源文件中,对于其它源文件,a是不可见的,且无法再被extern所引用,当然,static也适用于函数。使用static之后,我们就可以在其它的源文件中定义和这个已经被static修饰后的同名的变量或者函数。

4 形参、实参、返回值

如果我们使用的函数并未进行声明,但是已经在后面进行了定义,此时会默认函数返回类型为int型,这会造成极其严重的后果。

使用的函数如果在使用之前并未定义或者可能在其他的文件中,那么就要进行声明,函数声明的目的就是告知编译器函数的返回值的类型。

注意:如果一个函数没有float、short、或者char类型的参数,在函数声明中完全可以省略掉参数类型的说明(注意,函数定义中不能省略参数类型的说明)。这种做法依赖于调用者能够提供数目正确且类型恰当的实参。这里,“恰当”并不意味着“等同”:float类型的参数会自动转换为double类型,short或者char类型的参数会自动转换为int类型。

在ANSI C标准发布之前,常常会有下面的这种声明和定义函数的方式:

int isvowel();//声明函数的方式
int isvowel(c)
		char c;
{
	return c =='a' ;
}

实际上,上面这种写法与下面这种写法是等价的:

int isvowel(int i)
{
	char c;
	return c=='a';
}

上述两种方式在VS2019中都是支持的。

看下面的例子:

#include<stdio.h>
int main()
{
	int i;
	char c;
	for (i = 0; i < 5; i++)
	{
		scanf("%d", &c);
		printf("%d ", i);
	}
	printf("\n");
	return 0;
}

表面上,这个程序从标准输入设备读入5个数,在标准输出设备设备上写5个数:

0 1 2 3 4

实际上,这个程序并不一定得到上面的结果。例如,在某个编译器上,它的输出是(当然,在VS2019环境下程序会崩溃,因为非法修改了内存空间)

0 0 0 0 0 1 2 3 4

为什么呢?问题的关键在于,这里的c被声明为char类型,而不是int类型。如果程序要求scanf读入一个整数,应该传递给他一个指向整数的指针。而程序中scanf函数得到的却是一个指向字符的指针,scanf函数并不能分辨这种情况,它只是将这个指向字符的指针作为指向整数的指针而接受,并且在指针指向的位置存储一个整数。因为整数所占的存储空间要大于字符所占的存储空间,所以字符c附近的内存被覆盖。

字符c附近的内存中存储的内容是由编译器决定的,在本例中它所存放的是整数i的低端部分。因此,每次读入一个数值到c时,都会将i的低端部分覆盖为0,而i的高端部分本来就是0,相当于i每次被重新设置为0,循环将一直进行。当到达文件的结束位置后,scanf函数不再试图读入新的值到c。这时,i才可以正常的运行,最后终止循环。

5 检查外部类型

注意:保证一个特定类型的所有外部定义在每个目标模块中都有相同的类型,“相同的类型”也应该是严格意义上的相同。

例如,在一个文件中包含定义:

char filename[] = "/etc/passwd";

而在另一个文件中包含声明:

extern char *filename;

在定义时,filename是一个字符数组的名称。尽管在一个语句中引用filename的值将得到指向该数组起始元素的指针,但是filename的类型是”字符数组“,而不是字符指针。在第二个声明中,filename被确定为一个指针。这两种方式使用存储空间的方式是不同的,它们无法以一种合乎情理的方式共存。第一个例子字符数组filename的内存布局如下图所示:

image-20220304221554347

第二种方式字符指针filename的内存布局如下图所示:

image-20220304221842175

修改方法如下图所示:

char filename[] = "/etc/passwd";
extern char filename[];

也可以这样进行修改:

char*filename = "/etc/passwd";
extern char *filename;

6 头文件

注意:每个外部对象只在一个地方声明,这个声明的地方一般就在头文件种,需要用到该外部对象的所有模块也应该 包括在这个头文件。特别指出的是,定义该外部对象的模块也应该包括这个头文件。

到此这篇关于C语言 超详细讲解链接器的文章就介绍到这了,更多相关C语言 链接器内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C语言实现餐饮结账管理系统

    C语言实现餐饮结账管理系统

    这篇文章主要为大家详细介绍了C语言实现餐饮结账管理系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-11-11
  • C/C++位操作实例总结

    C/C++位操作实例总结

    这篇文章主要介绍了C/C++位操作实例总结,是C/C++程序设计中很重要的概念,需要的朋友可以参考下
    2014-08-08
  • C++实现哈希散列表的示例

    C++实现哈希散列表的示例

    本文主要介绍了C++实现哈希散列表的示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07
  • C++ OpenCV实战之文档照片转换成扫描文件

    C++ OpenCV实战之文档照片转换成扫描文件

    这篇文章主要为大家介绍一个C++ OpenCV的实战——文档照片转换成扫描文件,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2022-09-09
  • 带你深度走入C语言取整以及4种函数

    带你深度走入C语言取整以及4种函数

    大家都知道取整这回事,但是对于取整只有单一的认识,下面这篇文章主要给大家介绍了关于C语言取整以及4种函数的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-08-08
  • Qt学习教程之表格控件蚂蚁线详解

    Qt学习教程之表格控件蚂蚁线详解

    如果有用过PS的选区工具应该就会知道蚂蚁线是什么东西了,就是用来表示选区的一种虚线,关键还是要动态的!下面这篇文章主要给大家介绍了关于Qt学习教程之表格控件蚂蚁线的相关资料,需要的朋友可以参考下
    2018-07-07
  • c语言左移和右移的示例详解

    c语言左移和右移的示例详解

    这篇文章主要介绍了c语言左移和右移的示例详解,移位操作符的两个操作数必须是整型的。整个移位表达式的值的类型也是整型的,而且,左移位操作符与右移位操作符的运算并不对称。
    2020-07-07
  • C语言拓展实现Lua sleep函数

    C语言拓展实现Lua sleep函数

    这篇文章主要介绍了C语言拓展实现Lua sleep函数,本文使用C语言写出sleep函数,编译后在Lua中调用,需要的朋友可以参考下
    2015-04-04
  • C++面试基础之static关键字详解

    C++面试基础之static关键字详解

    这篇文章主要给大家介绍了关于C++面试基础之static关键字的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-02-02
  • 解析C++中四种强制类型转换的区别详解

    解析C++中四种强制类型转换的区别详解

    本篇文章是对C++中四种强制类型转换的区别进行了详细的分析介绍,需要的朋友参考下
    2013-05-05

最新评论