题外话
鄙人最近在家里听斯坦福大学的开放课程——《编程范式Programming Paradigms》。
附上veryCD的资源下载地址:
http://www.verycd.com/topics/2838268/
这个是英文授课,现在还是没有中文或英文字幕的。就课程来说非常不错。其中Lecture 4中讲到了一种在C语言中不用C++中的template实现泛型编程的方法。我在这里总结了它的笔记后,加入了我自己的一些思考和试验写下了这篇文章分享一些经验给大家。
正题
泛型编程让你编写完全一般化并可重复使用的算法,其效率与针对某特定数据类型而设计的算法相同。所谓泛型(Genericity),是指具有在多种数据类型上皆可操作的含意。C++通过参数化类型来实现通用的容器。如Java则引入了单根继承的概念。比泛型更加让你熟悉的可能就是STL,Standard template library,标准模板库。STL是一种高效、泛型、可交互操作的软件组件。STL以迭代器 (Iterators)和容器(Containers)为基础,是一种泛型算法(Generic Algorithms)库,容器的存在使这些算法有东西可以操作。STL包含各种泛型算法(algorithms)、泛型指针(iterators)、泛型容器(containers)以及函数对象(function objects)。STL并非只是一些有用组件的集合,它是描述软件组件抽象需求条件的一个正规而有条理的架构。
而在C语言中,同样也可以通过一些手段实现这样的泛型编程。如通过宏。这里讲的是另一种方法——通过无类型指针void*。
看下面的一个实现交换两个元素内容的函数swap,以整型int为例。
当你想交换两个char类型时,你还得重写一个参数类型为char的函数,你会多么想念C++中使用模板。等一下,是不是能用无类型的指针来作为参数呢?看如下改动:
这段代码是错误的,是通不过编译的。首先,变量是不能声明为void无类型的。而你不知道调用此函数传进的参数是什么类型的,无法确定一种类型的声明。同时,不能将*用在无类型指针上,因为系统没有此地址指向对象大小的信息。在编译阶段,编译器无法得知传入此函数参数的类型的。这里要想实现泛型的函数,需要在调用的地方传入相关要交换的对象的地址空间大小size,同时利用在头文件string.h中定义的memcpy()函数来实现。改动如下:
函数memcpy()
原型:void* memcpy(void *dest,void *src,unsigned int n)
作用:由src所指内存区域复制n个字节到dest所指内存区域。
说明:src和dest所指内存区域不能重叠,函数返回指向dest的指针。
之所以将buffer声明为char类型,只是因为char类型的大小是1字节,这样声明可以为buffer指向的空间分配size字节大小的空间。
在调用这个函数时,可以像如下这样调用(同样适用于其它类型的x、y):
其实,你会发现,这样的调用也是没有报错的:
两个不同类型的变量同样也可以实现交换。而根据系统的不同,即是小端法还是大端法保存数据,显示的结果是不同的。以Windows或linux下更常见的小端法为例,函数swap将整型i的低16位和短整型的16位进行了交换。
这样的调用是可行的,也提醒我们C语言下这样实现泛型编程是不安全的,编译器只进行有限的检查,更多的需要程序员的细心。如下的调用是可以的,但却毫无意义:
调用此函数,将交换i的地址向后8字节的内容与s的地址后8字节的内容的交换,而对于大小为2字节的short类型的s,大小为4字节的int类型的i这样的操作一般情况下是没有意义的。在编程时也要注意这个。C语言为我们提供了许多接近机器底层的操作,但也是一把双刃剑。
下面看这样的调用:
函数strdup():原型:char* strdup(char* s)。作用是复制字符串。
husband与wife是两个字符串,在交换这样的类型时,不能直接将两个字符串的地址传入swap函数,因为两个字符串的大小是不一定的,进行复制肯定会出现错误。这里是将保存两个字符串首地址的指针的地址进行交换,使两个指针指向对方的字符串,实现字符串的交换。sizeof()中写成char**也是可以的,写成double*也是可以的,因为所有指针的大小都是4字节。但写成char*,更加明确,在别人读代码会引起更少的误解,因为husband与wife是char类型的一级指针。
下面看另一种功能的函数:
此函数在数组array中查找key元素,找到后返回它的索引,找不到返回-1.如上,也可以实现泛型的函数:
代码第三行:将数组的首地址强制转换为指向char类型的指针,是利用char类型大小为1字节的特性,使elemAddr指向此”泛型“数组的第i-1个元素的首地址。因为之前已经说过,此时你并不知道你传入的是什么类型的数据,系统无法确定此数组一个元素有多长,跳向下个元素需要多少字节,所以强制转换为指向char的指针,再加上参数传入的元素大小信息和累加数i的乘积,即偏移地址,即可得此数组第i-1个元素的首地址。这样使无论传入的参数是指向什么类型的指针,都可以得到指向正确元素的指针,实现泛型编程。
函数memcmp(),原型:int memcmp(void *dest,const void *src,int n),比较两段长度为n首地址分别为dest、src的地址空间中的内容。
此函数在数组base中查找key元素,找到则返回它的地址信息,找不到则返回NULL。
如果使用函数指针,则可以实现其行为的泛型:
在调用时,可以将指向不同行为的形式相同的函数的指针传入此函数,以实现lsearch函数行为的不同。
函数指针:
返回类型 (*函数指针名)(形参表)
综上,C语言也可以实现一定的泛型编程,但这样是不安全的,系统对其只有有限的检查。在编程时一定要多加细心。
分享到:
相关推荐
这个是我写的一个c语言实现的泛型和迭代器,只是简单的实现了一个链表,用long和double作为链表的元素类型. 没有实现内存释放,累了不想写了. 欢迎指正.
在C语言中,可以通过一些手段实现这样的泛型编程。这里介绍一种方法——通过无类型指针void* 看下面的一个实现交换两个元素内容的函数swap,以整型int为例: void swap(int* i1,int* i2){ int temp; temp = *i1; ...
在C++中,实现泛型编程的核心机制是模板。 #### 泛型编程的历史背景 早期的面向对象语言,如Java,通过使用单一基类(如`Object`)的方法来实现泛型容器,这种方式在C++中并不适用,原因在于: 1. **单根继承的...
【C语言泛型指针技术】C语言是一种面向过程的编程语言,虽然不具备内置的泛型编程机制,但可以通过泛型指针(void*)来实现泛型编程。泛型指针可以指向任何类型的数据,它不绑定特定的数据类型,而是需要在使用时...
标题中的“cpp-微小的C库利用未定义的行为来实现泛型编程”指的是一个C语言编写的库,它尝试通过利用C语言中未定义的行为(Undefined Behavior, UB)来模拟实现类似C++中的泛型编程特性。在C++中,泛型编程允许...
虽然C语言的实现不如C++ STL那样简洁和强大,但它可以帮助初学者理解数据结构和算法的基础,以及如何在没有模板支持的情况下实现泛型编程。 在提供的`List`压缩包文件中,可能包含了实现这些功能的源代码。对于初学...
在IT行业中,C语言是一种基础且强大的编程语言,尤其在系统级编程和嵌入式领域中广泛应用。然而,C语言本身并不直接支持泛型编程,这使得开发复用性强、可扩展的数据结构变得较为复杂。"cpp-这是一个关于数据结构的...
根据提供的文档信息,本文将重点解析“泛型编程与C...通过上述分析,我们可以看出,即使是在没有直接泛型支持的语言中(如C语言),也可以通过一些技巧实现泛型编程的效果。这对于理解和应用泛型编程原则是非常有益的。
总之,本项目通过C语言实现了顺序存储和链式存储的泛型队列,提供了一套完整的队列操作接口,适用于多种场景。理解和掌握这些知识点对于提升C语言编程能力和数据结构应用能力非常有帮助。通过阅读和分析项目中的代码...
在STL中,泛型编程通过模板的使用来实现。模板允许开发者编写能够接受不同类型参数的函数或类,这样就可以创建出能够在不同数据类型上工作的通用算法和数据结构。 ##### 4.2 概念和模型 在泛型编程中,“概念”是...
泛型编程是一种编程范式,它允许程序员编写不依赖于特定数据类型的代码,从而提高了代码的复用性和灵活性。...通过深入理解和熟练运用C语言的类函数宏,可以有效地实现泛型编程,提升代码的可读性和效率。
在libcstl中,迭代器的实现也是一个挑战,因为C语言没有内置的模板机制,所以可能需要使用函数指针或者其他方式来模拟泛型编程。 此外,STL还包含了算法库,如排序、查找、交换等。在C语言中实现这些算法需要更底层...
泛型编程则是利用模板实现的一种编程范式,它专注于设计在多种数据类型上都能工作的算法和数据结构。 C++模板的主要形式有函数模板和类模板两种。函数模板允许开发者定义函数的参数类型,编译器在编译时根据模板...
“泛型”一词通常在高级编程语言中出现,如Java或C#,用来表示可以处理多种数据类型的代码。然而,在C语言中,我们没有内置的泛型机制。但是,通过巧妙地使用void指针和类型转换,我们可以模拟泛型栈的实现,使得同...
最后,书中可能会包含大量实例,展示如何将这些理论应用到实际问题中,比如构建数据结构(如链表、树、图)、设计模式(如工厂模式、单例模式)的C语言实现,以及如何设计和实现复杂的系统接口。 总之,《C语言接口...
在IT领域,泛型编程是一种强大的编程范式,它允许我们编写可以处理多种数据类型的代码。C++中的标准模板库(Standard Template Library, STL)就是泛型编程的一个典范,提供了高效且灵活的数据结构和算法。然而,...
在`vector.c`和`vector.h`这两个文件中,我们可以预期看到一个C语言实现的向量(vector)库。`vector.h`通常包含接口定义,即函数声明和可能的数据结构定义,而`vector.c`则包含实际的函数实现。 下面我们将详细...
为了实现泛型,我们可以编写一个通用的函数或宏,但C语言中参数的类型是在编译时确定的。为了解决这个问题,我们可以利用C99引入的复合文字(compound literals)特性。例如: ```c #define P00_SWAP2(A, B) p00_...
在C语言中,`sort`通常指的是排序算法的实现,特别是在C++标准库中,`std::sort`是一个常用的排序函数。然而,在纯C环境中,没有内置的`sort`函数,程序员需要自己实现排序算法。这里我们将探讨C语言中实现排序的...