泛型编程让你编写完全一般化并可重复使用的算法,其效率与针对某特定数据类型而设计的算法相同。在C语言中,可以通过一些手段实现这样的泛型编程。这里介绍一种方法——通过无类型指针void*
看下面的一个实现交换两个元素内容的函数swap,以整型int为例:
void swap(int* i1,int* i2){ int temp; temp = *i1; *i1 = *i2; *i2 = temp; }
当你想交换两个char类型时,你还得重写一个参数类型为char的函数,是不是能用无类型的指针来作为参数呢?看如下改动:
void swap(void *vp1,void *vp2){ void temp = *vp1; *vp1 = *vp2; *vp2 = temp; }
这段代码是错误的,是通不过编译的。首先,变量是不能声明为void无类型的。而你不知道调用此函数传进的参数是什么类型的,无法确定一种类型的声明。同时,不能将*用在无类型指针上,因为系统没有此地址指向对象大小的信息。在编译阶段,编译器无法得知传入此函数参数的类型的。这里要想实现泛型的函数,需要在调用的地方传入相关要交换的对象的地址空间大小size,同时利用在头文件string.h中定义的memcpy()函数来实现。改动如下:
void swap(void *vp1,void *vp2,int size){ char buffer[size];//注意此处gcc编译器是允许这样声明的 memcpy(buffer,vp1,size); memcpy(vp1,vp2,size); memcpy(vp2,buffer,size); }
在调用这个函数时,可以像如下这样调用(同样适用于其它类型的x、y):
int x = 27,y = 2; swap(&x,&y,sizeof(int));
下面看另一种功能的函数:
int lsearch(int key,int array[],int size){ for(int i = 0;i < size; ++i) if(array[i] == key) return i; return -1; }
此函数在数组array中查找key元素,找到后返回它的索引,找不到返回-1.如上,也可以实现泛型的函数:
void* lsearch(void* key, void *base, int n, int elemSize){ for(int i = 0;i < n; ++i){ void *elemAddr = (char *)base+i*elemSize; if(memcmp(key, elemAddr, elemSize) == 0) return elemAddr; } return NULL; }
代码第三行:将数组的首地址强制转换为指向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。
如果使用函数指针,则可以实现其行为的泛型:
void *lsearch(void *key,void *base,int n,int elemSize,int(*cmpfn)(void*,void*,int)){ for(int i = 0;i < n; ++i){ void *elemAddr = (char *)base+i*elemSize; if(cmpfn(key,elemAddr,elemSize) == 0) return elemAddr; } return NULL; }
再定义一个要调用的函数:
int intCmp(void* elem1,void* elem2){ int* ip1 = elem1; int* ip2 = elem2; return *ip1-*ip2; }
看如下调用:
int array[] = {1,2,3,4,5,6}; int size = 6; int number = 3; int *found = lsearch(&number,array,size,sizeof(int),intCmp); if(found == NULL) printf("NO\n"); else printf("YES\n");
C语言也可以实现一定的泛型编程,但这样是不安全的,系统对其只有有限的检查。在编程时一定要多加细心。
相关推荐
在C语言中,可以通过一些手段实现这样的泛型编程。这里介绍一种方法——通过无类型指针void* 看下面的一个实现交换两个元素内容的函数swap,以整型int为例: void swap(int* i1,int* i2){ int temp; temp = *i1; ...
【C语言泛型指针技术】C语言是一种面向过程的编程语言,虽然不具备内置的泛型编程机制,但可以通过泛型指针(void*)来实现泛型编程。泛型指针可以指向任何类型的数据,它不绑定特定的数据类型,而是需要在使用时...
在C语言中,泛型编程允许我们编写不依赖特定数据类型的代码。在交换两个变量的值这一常见任务中,泛型编程提供了两种策略:p99_swap1 和 p99_swap2。这两种方法都是为了确保在不同数据类型下正确地交换变量值,同时...
在C++中,实现泛型编程的核心机制是模板。 #### 泛型编程的历史背景 早期的面向对象语言,如Java,通过使用单一基类(如`Object`)的方法来实现泛型容器,这种方式在C++中并不适用,原因在于: 1. **单根继承的...
泛型纯C语法实现的底层容器 能想到的接口 都已实现 并单独提供一个正向迭代器与反向迭代器 也提供了3种遍历方式 前中后序 参见测试用例 其维护红黑树性质的相关代码 几乎是句句含带注释 使用面向对象编程思想构建...
标题中的“cpp-微小的C库利用未定义的行为来实现泛型编程”指的是一个C语言编写的库,它尝试通过利用C语言中未定义的行为(Undefined Behavior, UB)来模拟实现类似C++中的泛型编程特性。在C++中,泛型编程允许...
根据提供的文档信息,本文将重点解析“泛型编程与C...通过上述分析,我们可以看出,即使是在没有直接泛型支持的语言中(如C语言),也可以通过一些技巧实现泛型编程的效果。这对于理解和应用泛型编程原则是非常有益的。
然而,C语言本身并不直接支持泛型编程,这使得开发复用性强、可扩展的数据结构变得较为复杂。"cpp-这是一个关于数据结构的C语言静态链接库支持泛型"项目正解决了这一问题,它提供了一个静态链接库,允许C程序员在...
虽然C语言的实现不如C++ STL那样简洁和强大,但它可以帮助初学者理解数据结构和算法的基础,以及如何在没有模板支持的情况下实现泛型编程。 在提供的`List`压缩包文件中,可能包含了实现这些功能的源代码。对于初学...
在STL中,泛型编程通过模板的使用来实现。模板允许开发者编写能够接受不同类型参数的函数或类,这样就可以创建出能够在不同数据类型上工作的通用算法和数据结构。 ##### 4.2 概念和模型 在泛型编程中,“概念”是...
这个是我写的一个c语言实现的泛型和迭代器,只是简单的实现了一个链表,用long和double作为链表的元素类型. 没有实现内存释放,累了不想写了. 欢迎指正.
C++是一种静态类型的、编译式的、通用的、大小写敏感的、不仅支持过程化编程、数据抽象、面向对象编程、泛型编程等多种编程范式的编程语言。C++由Bjarne Stroustrup在贝尔实验室工作期间开发,作为C语言的扩展而诞生...
总之,本项目通过C语言实现了顺序存储和链式存储的泛型队列,提供了一套完整的队列操作接口,适用于多种场景。理解和掌握这些知识点对于提升C语言编程能力和数据结构应用能力非常有帮助。通过阅读和分析项目中的代码...
这样的泛型实现虽然比现代语言中的泛型功能略显繁琐,但它展示了C语言的灵活性和底层控制能力。 文件"stack"可能包含了实现这些功能的源代码,包括函数定义和测试用例。深入分析和理解这个文件,可以帮助你更好地...
#### 面向对象编程在C语言中的实现 《C语言面向对象编程》一书由Laurent Deniau编写,首次发布于2001年3月10日,并在同年5月17日进行了修订。该书主要探讨了如何在基于ISO C89标准的C语言项目中运用面向对象设计...
CTL,可能是"Controlled Template Library"的缩写,是针对C语言设计的一种库,旨在模拟C++ STL的部分功能,让开发者能够用C语言实现类似泛型的数据结构。在这个压缩包中,我们可以看到实现了四种基本的抽象数据类型...
4. 模板:泛型编程,创建可应用于多种数据类型的函数或类。 5. 异常处理:通过try-catch语句捕获和处理运行时错误。 6. 标准库:STL(Standard Template Library),包括容器(如vector, list, set)、算法和迭代器...
在libcstl中,迭代器的实现也是一个挑战,因为C语言没有内置的模板机制,所以可能需要使用函数指针或者其他方式来模拟泛型编程。 此外,STL还包含了算法库,如排序、查找、交换等。在C语言中实现这些算法需要更底层...
在C++中,`std::vector`是一种非常重要...C语言实现的向量库通常不会有C++中的`std::vector`那么便利,但通过巧妙的设计和编程,可以达到类似的效果。同时,由于C语言的特性,这个库可能更注重性能和内存管理的灵活性。