`
iwebcode
  • 浏览: 2004139 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
文章分类
社区版块
存档分类
最新评论

如何在C语言中实现简单的泛型编程(一)

 
阅读更多

原文地址


题外话

鄙人最近在家里听斯坦福大学的开放课程——《编程范式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为例。

[cpp:nogutter]view plaincopyprint?
  1. voidswap(int*i1,int*i2){
  2. inttemp;
  3. temp=*i1;
  4. *i1=*i2;
  5. *i2=temp;
  6. }

当你想交换两个char类型时,你还得重写一个参数类型为char的函数,你会多么想念C++中使用模板。等一下,是不是能用无类型的指针来作为参数呢?看如下改动:

[cpp:nogutter]view plaincopyprint?
  1. voidswap(void*vp1,void*vp2){
  2. voidtemp=*vp1;
  3. *vp1=*vp2;
  4. *vp2=temp;
  5. }

这段代码是错误的,是通不过编译的。首先,变量是不能声明为void无类型的。而你不知道调用此函数传进的参数是什么类型的,无法确定一种类型的声明。同时,不能将*用在无类型指针上,因为系统没有此地址指向对象大小的信息。在编译阶段,编译器无法得知传入此函数参数的类型的。这里要想实现泛型的函数,需要在调用的地方传入相关要交换的对象的地址空间大小size,同时利用在头文件string.h中定义的memcpy()函数来实现。改动如下:

[cpp:nogutter]view plaincopyprint?
  1. voidswap(void*vp1,void*vp2,intsize){
  2. charbuffer[size];/*注意此处gcc编译器是允许这样声明的,而我印象中VC编译器是不可以的。*/
  3. memcpy(buffer,vp1,size);
  4. memcpy(vp1,vp2,size);
  5. memcpy(vp2,buffer,size);
  6. }

from:


函数memcpy()

原型:void* memcpy(void *dest,void *src,unsigned int n)

作用:由src所指内存区域复制n个字节到dest所指内存区域。

说明:src和dest所指内存区域不能重叠,函数返回指向dest的指针。

之所以将buffer声明为char类型,只是因为char类型的大小是1字节,这样声明可以为buffer指向的空间分配size字节大小的空间。

在调用这个函数时,可以像如下这样调用(同样适用于其它类型的x、y):

[cpp:nogutter]view plaincopyprint?
  1. intx=27,y=2;
  2. swap(&x,&y,sizeof(int));

其实,你会发现,这样的调用也是没有报错的:

[cpp:nogutter]view plaincopyprint?
  1. inti=44;
  2. shorts=5;
  3. swap(&i,&s,sizeof(short));

两个不同类型的变量同样也可以实现交换。而根据系统的不同,即是小端法还是大端法保存数据,显示的结果是不同的。以Windows或linux下更常见的小端法为例,函数swap将整型i的低16位和短整型的16位进行了交换。

这样的调用是可行的,也提醒我们C语言下这样实现泛型编程是不安全的,编译器只进行有限的检查,更多的需要程序员的细心。如下的调用是可以的,但却毫无意义:

[cpp:nogutter]view plaincopyprint?
  1. inti=22;
  2. shorts=11;
  3. swap(&i,&s,8);

调用此函数,将交换i的地址向后8字节的内容与s的地址后8字节的内容的交换,而对于大小为2字节的short类型的s,大小为4字节的int类型的i这样的操作一般情况下是没有意义的。在编程时也要注意这个。C语言为我们提供了许多接近机器底层的操作,但也是一把双刃剑

下面看这样的调用:

[cpp:nogutter]view plaincopyprint?
  1. char*husband=strdup("Fred");
  2. char*wife=strdup("Linda");
  3. swap(&husband,&wife,sizeof(char*));

函数strdup():原型:char* strdup(char* s)。作用是复制字符串。

husband与wife是两个字符串,在交换这样的类型时,不能直接将两个字符串的地址传入swap函数,因为两个字符串的大小是不一定的,进行复制肯定会出现错误。这里是将保存两个字符串首地址的指针的地址进行交换,使两个指针指向对方的字符串,实现字符串的交换。sizeof()中写成char**也是可以的,写成double*也是可以的,因为所有指针的大小都是4字节。但写成char*,更加明确,在别人读代码会引起更少的误解,因为husband与wife是char类型的一级指针。

下面看另一种功能的函数:

[cpp:nogutter]view plaincopyprint?
  1. intlsearch(intkey,intarray[],intsize){
  2. for(inti=0;i<size;++i)
  3. if(array[i]==key)
  4. returni;
  5. return-1;
  6. }

此函数在数组array中查找key元素,找到后返回它的索引,找不到返回-1.如上,也可以实现泛型的函数:

[cpp:nogutter]view plaincopyprint?
  1. void*lsearch(void*key,void*base,intn,intelemSize){
  2. for(inti=0;i<n;++i){
  3. void*elemAddr=(char*)base+i*elemSize;/*重点,认真看这行代码*/
  4. if(memcmp(key,elemAddr,elemSize)==0)
  5. returnelemAddr;
  6. returnNULL;
  7. }
  8. }

代码第三行:将数组的首地址强制转换为指向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。

如果使用函数指针,则可以实现其行为的泛型:

[cpp:nogutter]view plaincopyprint?
  1. void*lsearch(void*key,void*base,intn,intelemSize,int(*cmpfn)(void*,void*,int)){
  2. for(inti=0;i<n;++i){
  3. void*elemAddr=(char*)base+i*elemSize;
  4. if(cmpfn(key,elemAddr,elemSize)==0)
  5. returnelemAddr;
  6. returnNULL;
  7. }
  8. }
  9. }

在调用时,可以将指向不同行为的形式相同的函数的指针传入此函数,以实现lsearch函数行为的不同。

函数指针:

返回类型 (*函数指针名)(形参表)

综上,C语言也可以实现一定的泛型编程,但这样是不安全的,系统对其只有有限的检查。在编程时一定要多加细心。

分享到:
评论

相关推荐

    c语言实现c++的泛型和迭代器

    这个是我写的一个c语言实现的泛型和迭代器,只是简单的实现了一个链表,用long和double作为链表的元素类型. 没有实现内存释放,累了不想写了. 欢迎指正.

    C语言泛型编程实例教程

    在C语言中,可以通过一些手段实现这样的泛型编程。这里介绍一种方法——通过无类型指针void* 看下面的一个实现交换两个元素内容的函数swap,以整型int为例: void swap(int* i1,int* i2){ int temp; temp = *i1; ...

    C语言中基于类函数宏技术的泛型顺序栈的设计与实现.pdf

    泛型编程是多数现代程序设计语言支持的,能够简化代码、提高代码...依据c语言的基础语法规则,深度挖掘c语言本身的语言特性,基于类函数宏技术实现了泛型顺序栈的编程实践,为使用c语言实践泛型编程提供了一种解决思路。

    用C语言的泛型实现交换两个变量值

    在日常编程里面经常会遇到交换两个变量的内容的任务,对于泛型类型而言有两种泛型策略来实现,下面跟着小编一起来学习学习。

    c语言实现泛型红黑树(含迭代器) 见说明

    泛型纯C语法实现的底层容器 能想到的接口 都已实现 并单独提供一个正向迭代器与反向迭代器 也提供了3种遍历方式 前中后序 参见测试用例 其维护红黑树性质的相关代码 几乎是句句含带注释 使用面向对象编程思想构建...

    C语言函数大全速查手册

    C++这个词在中国大陆的程序员圈子中通常被读做“C加加”,而西方的程序员通常读做“C plus plus”,“CPP”或者“dev.cpp"。 它是一种使用非常广泛的计算机编程语言。C++是一种静态数据类型检查的,支持多重编程范式...

    学会C语言之后还有必要学习C++吗?具体运用C++编写的代码解析.docx

    模板是C++中一种泛型编程技术,它允许程序员编写通用的代码,从而可以在不同的数据类型上重用代码。STL是C++中一个非常重要的库,它提供了许多基本数据结构和算法,如向量、列表、堆、排序等等。使用STL可以让程序员...

    基于c++实现课程设计-图书管理系统项目源码设计与实现

    它是C语言的扩展,支持面向对象编程和泛型编程。C++在计算机科学和工程领域得到了广泛应用,用于开发操作系统、编译器、游戏、图形用户界面和其他应用程序。 C++具有许多强大的特性,其中最重要的是面向对象编程。它...

    基于c++实现课程设计-学生成绩管理系统项目源码设计与实现

    它是C语言的扩展,支持面向对象编程和泛型编程。C++在计算机科学和工程领域得到了广泛应用,用于开发操作系统、编译器、游戏、图形用户界面和其他应用程序。 C++具有许多强大的特性,其中最重要的是面向对象编程。它...

    基于c++实现课程设计宾馆客房管理系统附项目源码

    它是C语言的扩展,支持面向对象编程和泛型编程。C++在计算机科学和工程领域得到了广泛应用,用于开发操作系统、编译器、游戏、图形用户界面和其他应用程序。 C++具有许多强大的特性,其中最重要的是面向对象编程。它...

    基于c++实现课程设计-产品入库管理操作系统附项目源码

    它是C语言的扩展,支持面向对象编程和泛型编程。C++在计算机科学和工程领域得到了广泛应用,用于开发操作系统、编译器、游戏、图形用户界面和其他应用程序。 C++具有许多强大的特性,其中最重要的是面向对象编程。它...

    基于c++实现课程设计-职工工作量统计系统附项目源码

    它是C语言的扩展,支持面向对象编程和泛型编程。C++在计算机科学和工程领域得到了广泛应用,用于开发操作系统、编译器、游戏、图形用户界面和其他应用程序。 C++具有许多强大的特性,其中最重要的是面向对象编程。它...

    线性表的顺序存储C++实现(类模板实现)

    大致实现了STL中的线性表基本功能,通过对比C语言实现线性表的顺序存储可以更好的理解面向对象编程和面向过程的区别,更加深入的理解C语言和C++语言的区别。相关的文章可以在我的主页算法与数据结构专栏查看。

    c++ 面试题库.pdf

    3)C语言实现了对硬件的编程操作,也适合于应用软件的开发; 4) C语言还具有效率高,可移植性强等特点。 (2) C++语言特点: 1) 在C语言的基础上进行扩充和完善,使C+t兼容了C语言的面向过程特点,又成为了一种面向...

    C++从初识到模板类-大一,面向对象编程

    这是我大一上学期刚学完C语言后,开始学习C++,开始接触...在学习C++过程中,加深了我对编程的理解,比如python为什么很多方法可以对不同的数据类型进行操作,同时模板类,泛型编程帮助我可以自己完成一个库的实现。

Global site tag (gtag.js) - Google Analytics