C++ Typedef

原文传送
在阅读代码经常会遇到一些复杂的声明和定义,例如:

    (1)  void * (* (*fp1) (int)) [10];

    (2)  float (* (*fp2) (int, int, float)) (int);

    (3)  typedef double (* (* (*fp3) ()) [10]) ();

           fp3 a;

    (4)  int (* (*fp4()) [10]) ();

刚看到这些声明或者定义时,一些初学者甚至有一定经验的工程师都有可能头皮发毛,基于大惑不解。如果缺乏经验和方法来对这些内容进行理解,势必会让我们浪费大量的时间。
*下面先看一些简单的定义:

  1. 定义一个整型数

    int a;

  2. 定义一个指向整型数的指针

    int *p;

  3. 定义一个指向指针的指针,它指向的指针指向一个整型数

    int **pp;

到这一步我想大多数人都还好理解,我们可以用一些简单的代码把这三条给串起来:

int a;
int *p;
int **pp;
 
p = &a;   // p指向整数a所在的地址
pp = &p;  // pp指向指针p
  1. 定义一个包含10个整型数的数组
    int arr[10];
  2. 定义一个指向包含10个整型数数组的指针
    int (*pArr) [10];

用几行代码将4、5两个定义串起来:

 int arr[10];
int (*pArr) [10];
 
pArr = &arr;
  1. 定义一个指向函数的指针,被指向的函数有一个整型参数并返回整型值
    int (*pfunc) (int);

  2. 定义一个包含10个指针的数组,其中包含的指针指向函数,这些函数有一个整型参数并返回整型值
    int (*arr[10]) (int);

用几行代码将6、7两个定义串起来:

int (*pfunc) (int);
int (*arr[10]) (int);
 
arr[0] = pfunc;

到这一步,似乎就不是那么好理解了。现在需要请出用于理解复杂定义的“ 右左法则 ”:
从变量名看起,先往右,再往左,碰到圆括号就调转阅读的方向;括号内分析完就跳出括号,还是先右后左的顺序。如此循环,直到分析完整个定义。

    让我们用这个方法来分析上面的第6条定义:int (*pfunc) (int);

    找到变量名pfunc,先往右是圆括号,调转方向,左边是一个*号,这说明pfunc是一个指针;然后跳出这个圆括号,先看右边,又遇到圆括号,这说明(*pfunc)是一个函数,所以pfunc是一个指向这类函数的指针,即函数指针,这类函数具有一个int类型的参数,返回值类型是int。

    接着分析第7条定义:int (*arr[10]) (int);

    找到变量名arr,先往右是[]运算符,说明arr是一个数组;再往左是一个*号,说明arr数组的元素是指针(注意:这里的*修饰的不是arr,而是arr[10]。原因是[]运算符的优先级比*要高,arr先与[]结合。);跳出圆括号,先往右又遇到圆括号,说明arr数组的元素是指向函数的指针,它指向的函数有一个int类型的参数,返回值类型是int。

    分析完这两个定义,相信多数人心里面应该有点谱了。可应该还有人会问:怎么判断定义的是函数指针(定义6),还是数组指针(定义5),或是数组(定义7)?可以抽象出几个模式:
type (*var)(...); // 变量名var与*结合,被圆括号括起来,右边是参数列表。表明这是函数指针
type (*var)[];    //变量名var与*结合,被圆括号括起来,右边是[]运算符。表示这是数组指针
type (*var[])...;     // 变量名var先与[]结合,说明这是一个数组(至于数组包含的是什么,由旁边的修饰决定)  

至此,我们应该有能力分析文章开始列出来了几条声明和定义:

    (1)  void * (* (*fp1) (int)) [10];

    找到变量名fp1,往右看是圆括号,调转方向往左看到*号,说明fp1是一个指针;跳出内层圆括号,往右看是参数列表,说明fp1是一个函数指针,接着往左看是*号,说明指向的函数返回值是指针;再跳出外层圆括号,往右看是[]运算符,说明函数返回的是一个数组指针,往左看是void *,说明数组包含的类型是void *。 简言之 ,fp1是一个指向函数的指针,该函数接受一个整型参数并返回一个指向含有10个void指针数组的指针。
    (2) float (* (*fp2) (int, int, float)) (int);

    找到变量名fp2,往右看是圆括号,调转方向往左看到*号,说明fp2是一个指针;跳出内层圆括号,往右看是参数列表,说明fp2是一个函数指针,接着往左看是*号,说明指向的函数返回值是指针;再跳出外层圆括号,往右看还是参数列表,说明返回的指针是一个函数指针,该函数有一个int类型的参数,返回值类型是float。简言之,fp2是一个指向函数的指针,该函数接受三个参数(int, int和float),且返回一个指向函数的指针,该函数接受一个整型参数并返回一个float。


    (3)  typedef double (* (* (*fp3) ()) [10]) ();

           fp3 a;

    如果创建许多复杂的定义,可以使用typedef。这一条显示typedef是如何缩短复杂的定义的。
    跟前面一样,先找到变量名fp3(这里fp3其实是新类型名),往右看是圆括号,调转方向往左是*,说明fp3是一个指针;跳出圆括号,往右看是空参数列表,说明fp3是一个函数指针,接着往左是*号,说明该函数的返回值是一个指针;跳出第二层圆括号,往右是[]运算符,说明函数的返回值是一个数组指针,接着往左是*号,说明数组中包含的是指针;跳出第三层圆括号,往右是参数列表,说明数组中包含的是函数指针,这些函数没有参数,返回值类型是double。简言之,fp3是一个指向函数的指针,该函数无参数,且返回一个含有10个指向函数指针的数组的指针,这些函数不接受参数且返回double值。

    这二行接着说明:a是fp3类型中的一个。

    (4)  int (* (*fp4()) [10]) ();

    这里fp4不是变量定义,而是一个函数声明。

    找到变量名fp4,往右是一个无参参数列表,说明fp4是一个函数,接着往左是*号,说明函数返回值是一个指针;跳出里层圆括号,往右是[]运算符,说明fp4的函数返回值是一个指向数组的指针,往左是*号,说明数组中包含的元素是指针;跳出外层圆括号,往右是一个无参参数列表,说明数组中包含的元素是函数指针,这些函数没有参数,返回值的类型是int。简言之,fp4是一个返回指针的函数,该指针指向含有10个函数指针的数组,这些函数不接受参数且返回整型值。

用typedef简化复杂的声明和定义
以上我们已经看到了不少复杂的声明和定义,这里再举一个例子:

    int *(*a[10]) (int, char*);

    用前面的“右左法则”,我们可以很快弄清楚:a是一个包含10个函数指针的数组,这些函数的参数列表是(int, char*),返回值类型是int*。理解已经不成问题,这里的关键是如果要定义相同类型的变量b,都得重复书写:

    int *(*b[10]) (int, char*);

这里有没有方便的办法避免这样没有价值的重复?答案就是用typedef来简化复杂的声明和定义。

typedef可以给现有的类型起个别名。这里用typedef给以上a、b的类型起个别名:

typedef int *(*A[10]) (int, char*); // 在之前定义的前面加入typedef,然后将变量名a替换成类型名A

现在要再定义相同类型的变量c,只需要:

A c;
typedef是C语言中的一个关键字,用于定义类型别名。通过typedef,可以为已有的类型赋予新的名称,方便程序的编写和阅读。 在C语言中,typedef的语法格式为:typedef 已有类型 新类型名称; 例如,我们可以使用typedef为char*类型定义一个新的名称pStr,即typedef char* pStr; 在之后的代码中,我们可以使用pStr来代替char*类型。 typedef还可以用于定义函数指针类型的别名。例如,我们可以使用typedef定义一个名为A的别名,表示指向具有两个char类型参数并返回int类型值的函数指针。使用A作为类型名,我们可以声明A类型的对象并将其指向符合上述规则的函数。这样,我们可以通过A类型的对象调用相应的函数。 总结来说,typedef是C语言中用于定义类型别名的关键字。它可以为已有的类型赋予新的名称,包括基本类型和指针类型等。同时,typedef也可以用于定义函数指针类型的别名,方便函数指针的声明和使用。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [C++ typedef详解](https://blog.csdn.net/weixin_41001497/article/details/108076377)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [C++ typedef 用法详解](https://blog.csdn.net/cuizhiyi2008/article/details/102804702)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值