typedef是C++中的一个十分重要的关键字,它有强大的功能和方法的用途。但是有时候,碰到一些用到typedef的地方却感到很奇怪了。

给个栗子尝尝:

  1. typedef void(*pFun)(void);

很奇怪,你不觉得奇怪吗?反正我是信了,一个字-“怪”。

好,下面就讲一下C++中的一怪“typedef”。

typedef的定义是,为现有类型创建一个新的名字,或称为类型别名。这就是一个关键的突破点,无论typedef怎么应用,都不会脱离它本身的定义。

1、typedef定义一种类型的别名

  1. typedef int* PINT;
  2. PINT pa,pb;//pa和pb都是int*类型的指针

给一种类型进行重定义别名,可以方便我们使用。

2、typedef和struct

在旧的C代码中,struct定义一个结构体时,必须加上struct

  1. typedef struct Point{
  2. int x;
  3. int y;
  4. }Point;

3、typedef与平台无关性

用typedef来定义与平台无关的类型。比如定义一个叫 REAL 的浮点类型,在目标平台一上,让它表示最高精度的类型为:typedef long double REAL;

在不支持 long double 的平台二上,改为:typedef double REAL;

在连 double 都不支持的平台三上,改为:typedef float REAL;

也就是说,当跨平台时,只要改下 typedef 本身就行,不用对其他源码做任何修改。标准库就广泛使用了这个技巧,比如size_t。另外,因为typedef是定义了一种类型的新别名,不是简单的字符串替换,所以它比宏来得稳健(虽然用宏有时也可以完成以上的用途)。

4、typedef定义函数指针

首先说明一下函数型指针,我们知道每种类型对应着属于自身类型的指针,当然函数也不例外。当一个函数被执行时,在程序空间中占据一定空间,这个空间的起始地址是用函数名来表示的,称为函数的入口地址。也可以用指针指向这个入口地址,并通过该指针变量来调用这个函数。这种指针变量称为函数型指针变量,其一般形式为:

数据类型标识符 (*指针变量名) () ; 例如:void (*f)( );

上式定义了指针f, f指向的函数返回void类型类数据。注意其中(*f)中的括弧不可缺少,标识f是先与*结合,是指针变量,然后再与后面的()结合,表示此指针指向函数。

为复杂的声明定义一个新的简单的别名。方法是:在原来的声明里逐步用别名替换一部分复杂声明,如此循环,把带变量名的部分留到最后替换,得到的就是原声明的最简化版。举例:

1. 原声明:int *(*a[5])(int, char*);
变量名为a,直接用一个新别名pFun替换a就可以了:
typedef int *(*pFun)(int, char*); 
原声明的最简化版:
pFun a[5];

2. 原声明:void (*b[10]) (void (*)());
变量名为b,先替换右边部分括号里的,pFunParam为别名一:
typedef void (*pFunParam)();
再替换左边的变量b,pFunx为别名二:
typedef void (*pFunx)(pFunParam);
原声明的最简化版:
pFunx b[10];

3. 原声明:doube(*)() (*e)[9]; 
变量名为e,先替换左边部分,pFuny为别名一:
typedef double(*pFuny)();
再替换右边的变量e,pFunParamy为别名二
typedef pFuny (*pFunParamy)[9];
原声明的最简化版:
pFunParamy e;

理解复杂声明可用的“右左法则”:
从变量名看起,先往右,再往左,碰到一个圆括号就调转阅读的方向;括号内分析完就跳出括号,还是按先右后左的顺序,如此循环,直到整个声明分析完。举例:
int (*func)(int *p);
首先找到变量名func,外面有一对圆括号,而且左边是一个*号,这说明func是一个指针;然后跳出这个圆括号,先看右边,又遇到圆括号,这说明(*func)是一个函数,所以func是一个指向这类函数的指针,即函数指针,这类函数具有int*类型的形参,返回值类型是int。
int (*func[5])(int *);
func右边是一个[]运算符,说明func是具有5个元素的数组;func的左边有一个*,说明func的元素是指针(注意这里的*不是修饰func,而是修饰func[5]的,原因是[]运算符优先级比*高,func先跟[]结合)。跳出这个括号,看右边,又遇到圆括号,说明func数组的元素是函数类型的指针,它指向的函数具有int*类型的形参,返回值类型为int。

也可以记住2个模式:
type (*)(....)函数指针 
type (*)[]数组指针

5、通过成员指针调用成员函数

可以在不必知道函数名的情况下,通过成员指针调用对象的成员函数。例如,函数dispatcher有一个变量pmf,通过它调用类成员函数,不管它调用的是strcpy()函数还是strcat()函数。指向外部原函数的指针和指向类成员函数的指针是有很大区别的。后者必须指向被调函数的宿主对象。因此,除了要有成员指针外,还要有合法对象或对象指针。

现举例做进一步说明。假设A有二个实例,成员函数指针支持多态性。这样在成员指针调用虚成员函数时是动态处理的(即所谓后联编 - 译注)。注意,不可调用构造和析构函数。示例如下:

  1. A a1, a2;
  2. A *p= &a1; //创建指向A的指针
  3. //创建指向成员的指针并初始化
  4. void (A::*pmf)(char *, const char *) = &A::strcpy;
  5. //要将成员函数绑定到pmf,必须定义呼叫的对象。
  6. //可以用*号引导:
  7. void dispatcher(A a, void (A::*pmf)(char *, const char *))
  8. {
  9.  char str[];
  10.  (a.*pmf)(str, abc”); //将成员函数绑定到pmf
  11. }
  12. //或用A的指针表达方式指向成员指针:
  13. void dispatcher(A * p, void (A::*pmf)(char *, const char *))
  14. {
  15.  char str[]; (p->*pmf)(str, abc”);
  16. }
  17. //函数的调用方法为:
  18. dispatcher(a, pmf); // .* 方式
  19. dispatcher(&a, pmf); // ->* 方式

6、高级使用技巧

以上是成员函数的基本知识。现在介绍它的高级使用技巧。 
6.1 成员指针数组

在下例,声明了一个含有二个成员指针的数组,并分配类的成员函数地址给成员指针

  1. PMA pmf[]= {&A::strcpy, &A::strcat};

等同于

  1. void (A::*PMA[])(char *, const char *)= {&A::strcpy, &A::strcat};

这样的数组在菜单驱动应用中很有用。选择菜单项后,应用将调用相应的回调函数,如下所示:

  1. enum MENU_OPTIONS { COPY, CONCAT };
  2. int main()
  3. {
  4.  MENU_OPTIONS option; char str[];
  5.  //从外部资源读取选项
  6.  switch (option)
  7.  {
  8. case COPY:
  9.   (pa->*pmf[COPY])(str, abc”);
  10.   break;
  11. case CONCAT:
  12.   (pa->*pmf[CONCAT])(str, abc”);
  13.   break;
  14.  }
  15. }

6.2 void类型的指针

void含义:

void是“无类型”,void*则为无类型指针,void*可以指向任何类型的数据。

void a;//此变量没有任何实际意义,无法编译通过“illegal use of type”

void 的作用:

1、对程序返回的限定

2、对函数参数的限定

我们知道,如何指针p1和p2的类型相同,那么我们可以直接在p1和p2间赋值,如果不同,必须使用强制类型转换。

如:float *p1; int *p2;

若:p1 = p2; 编译出错:“can not covert from int* to float*”

必须为:p1 = (float*)p2;

而void*不同,任何类型的指针都可以直接赋为它,不需要强制类型转换:

如:void *p1; int *p2;

可作:p1 =p2;

无类型可以包容有类型,有类型不能包容无类型:

必须为:p2 = (int*)p1;

如果函数的参数可以是任意类型指针,那么应声明其参数为void *

典型的如内存操作函数memcpy和memset的函数原型分别为:

  1. void* memcpy(void *dest, const void *src, size_t len);
  2. void* memset(void *buffer,int c, size_t num);

这样,任何类型的指针都可以传入memcpy和memset中,这也真实地体现了内存操作函数的意义,因为它操作的对象仅仅是一片内存,而不论内存是什类型。

7、typedef 与 #define的区别

案例一:

通常讲,typedef要比#define要好,特别是在有指针的场合。请看例子:

  1. typedef char *pStr1;
  2.  
  3. #define pStr2 char *;
  4.  
  5. pStr1 s1, s2;
  6.  
  7. pStr2 s3, s4;

在上述的变量定义中,s1、s2、s3都被定义为char *,而s4则定义成了char,不是我们所预期的指针变量,根本原因就在于#define只是简单的字符串替换而typedef则是为一个类型起新名字。

案例二:

下面的代码中编译器会报一个错误,你知道是哪个语句错了吗?

  1. typedef char * pStr;
  2. char string[] = "abc";
  3. const char *p1 = string;
  4. const pStr p2 = string;
  5. p1++;
  6. p2++;

是p2++出错了。这个问题再一次提醒我们:typedef和#define不同,它不是简单的文本替换。上述代码中const pStr p2并不等于const char * p2。const pStr p2和const long x本质上没有区别,都是对变量进行只读限制,只不过此处变量p2的数据类型是我们自己定义的而不是系统固有类型而已。因此,const pStr p2的含义是:限定数据类型为char *的变量p2为只读,因此p2++错误。

见怪不怪的typedef的更多相关文章

  1. 19-typedef

    本文目录 一.typedef作用简介 二.typedef与指针 三.typedef与结构体 三.typedef与指向结构体的指针 四.typedef与枚举类型 五.typedef与指向函数的指针 六. ...

  2. #define与typedef在重定义类型中的区别

    #define 为完全的替换 typedef 重新定一个数据类型 eg #define charp1 char* typedef char* charp2charp1 a,b; //a char* b ...

  3. typedef

    第一.四个用途 用途一: 定义一种类型的别名,而不只是简单的宏替换.可以用作同时声明指针型的多个对象.比如:char* pa, pb; // 这多数不符合我们的意图,它只声明了一个指向字符变量的指针, ...

  4. 用typedef定义函数指针的问题

    在学习windows API的时候,遇到下面这段代码   以前见过的typedef的用法都是给一个数据类型取一个别名 typedef oldTypeName newTypeName   这种给数据类型 ...

  5. C语言语法 typedef小结

    在总结typedef之前,先了解一个专业术语: 常量指针(const pointer):常量指针在定义的时候必须被初始化,而且一旦初始化完成,则它的值就不能再改变. int errNumb = 0; ...

  6. 如何理解typedef void (*pfun)(void)

    问题: 在刚接触typedef void (*pfun)(void) 这个结构的时候,存在疑惑,为什么typedef后只有一"块"东西,而不是两"块"东西呢?那 ...

  7. C及C++中typedef的简单使用指南

    又是在学数据结构的时候,发现了之前学习的知识遗忘很多,在发现对C/C++中关键字typedef的理解还是没有到位后,我翻阅了学C++用到的课本,又问了度娘,也看了不少关于typedef用法的博客.于是 ...

  8. [转]关于typedef的用法总结

    不管实在C还是C++代码中,typedef这个词都不少见,当然出现频率较高的还是在C代码中.typedef与#define有些相似,但更多的是不同,特别是在一些复杂的用法上,就完全不同了,看了网上一些 ...

  9. typedef 和 #define 的区别

    本文已迁移至: http://www.danfengcao.info/c/c++/2014/02/25/difference-between-define-and-typedef.html typed ...

随机推荐

  1. GL_GL系列 - 会计期间管理分析(案例)

    2014-07-07 Created By BaoXinjian

  2. UVA 272 TEX Quotes

    TEX Quotes 题意: 变引号. 题解: 要想进步,真的要看一本好书,紫书P45 代码: #include<stdio.h> int main() { int c,q=1; whil ...

  3. 通讯簿(apple)

    ylbtech-dbs:ylbtech-cnblogs(博客园)-2,Admin(用户后台) DatabaseName:Contacts/通讯簿(iOS) 1.A,数据库关系图(Database Di ...

  4. 如何限制textarea文本框的输入字数

    代码实例如下: <!doctype html><html><head><meta charset="UTF-8"><title ...

  5. oracle查看所有表的数据量

    源地址:http://blog.csdn.net/zhanggnol/article/details/6683697 select t.table_name,t.num_rows from user_ ...

  6. expdp和impdp的用法

    源地址:http://blog.chinaunix.net/uid-23622436-id-2394094.html

  7. HTTP gzip和deflate的几点区别

    gzip是一种数据格式,默认且目前仅使用deflate算法压缩data部分:deflate是一种压缩算法,是huffman编码的一种加强. deflate与gzip解压的代码几乎相同,可以合成一块代码 ...

  8. 有没有一行文件字过多后可以省略号显示,我说的不是用其他样式,BT本身有没有?谢谢

    .text-overflow {display: inline-block;max-width: 200px;overflow: hidden;text-overflow: ellipsis;whit ...

  9. 华硕X84L无线驱动查找

    打开官网:http://www.asus.com.cn/ 点击导航栏的服务与支持 产品型号识别http://www.asus.com.cn/support/Article/565/ 我的是:X84L  ...

  10. jQuery formValidator表单验证插件常见问题

    1.    如何实现一个控件,根据不同的情况,实现不同的控制? 2.    一个页面上我有几个tab页,如何实现每个Tab页上的控件单独校验? 3.    我采用的页面上文字问题的方式,点提交的时候, ...