c/c++排坑(5) -- c语言中的申明
C语言的申明总是令人头大,对于这块内容也一直让我头疼。希望通过这篇博客能够稍微梳理一下。材料和例子来源于《C专家编程》
一、C语言的申明的优先级规则
先来个例子,看看下面这行C代码到底是个啥玩意儿:
char * const *(*next)();
1.1 优先级规则
- A 申明从它的名字开始读取,然后按照优先级顺序依次读取。
- B 优先级从高到低依次是:
- B.1 声明中被括号括起来的那部分
- B.2 后缀操作符:
括号()表示这是一个函数,而方括号表示这是一个数组。 - B.3 前缀操作符:星号*表示 指向...的指针。
- C 如果const和(或)volatile关键字的后面紧跟类型说明符(如int,long等),那么它作用于类型说明符。在其它情况下,const和(或)volatile关键字作用于它紧邻的指针星号。
1.2 分析 char * const (next)();
- 首先,看变量名是“next”,并注意到它直接被括号括住(A)
- 把括号括住的东西看作一个整体(*next),得出 “next是一个指向...的指针”(B.1)
- 然后考虑括号外面的东西,在星号前缀和括号后缀之间做出选择:
B.2规则告诉我们优先级较高的是右边的函数括号,所以得出“next是一个函数指针,指向一个返回...的函数”(B.2) - 然后,处理前缀*号,得出指针所指的内容
- 最后,把“char * const”解释为指向字符的常量指针
所以最后得出:next是一个指针,它指向一个函数,该函数返回另一个指针,该指针指向一个类型为char的常量指针。
虽然能够搞明白,并且有规则可循。但还是不得不吐槽,C的申明也太复杂了。哎~~~
二、typedef
2.1 使用typedef为你的申明做简化
看系统调用signal函数的申明:
void (*signal(int sig, void(*func)(int)))(int);
先不看signal括号里面的东西:
void (*signal( ))(int);
可以看出来,signal是一个函数,然后它返回一个指针;该指针所指向的函数接受一个int参数,并返回void。我们再来看signal括号里面的东西,第一个参数不用解释,来看第二个参数:
void(*func)(int)
不难看出func是一个函数指针,指向的函数接受一个int参数,返回void。接下来让我们用typedef来简化它:
/* ptr_ti_func是一个函数指针,该函数接受一个int参数,返回void。 */
typedef void(*ptr_to_func) (int)
//接着再申明之前的那个signal
ptr_to_func signal(int, ptr_to_func);
2.2 typedef VS #define
#define只是做替换而已;而typedef是真正的封装--申明它之后,不能再往里面增加东西。举两个例子:
例子1:
#define myInt int
usigned myInt i; //可以没问题,就是文字替换而已:unsigned int i;
typedef int myInt;
usigned myInt i; //非法,不可以在myInt里面加usigned了。
例子2:
#define int_ptr int *
int_ptr a,b; //a是int *类型,b是一个int;
typedef int * int_ptr;
int_ptr a,b; //a和b都是int *类型。
最后想说一下作者本身对这个也只是初步了解,如有理解错误欢迎反馈!!
See you next time. Happy Coding!!!
我的github
c/c++排坑(5) -- c语言中的申明的更多相关文章
- c/c++排坑(2) -- c语言中的符号重载
所谓的符号重载就是在不同的上下文环境里有不同的意义.甚至有些关键字也被重载而具有好几种意义,这也是C语言的作用域规则对程序员不那么清晰的主要原因. 本章内容摘自<c专家编程>P37. 大家 ...
- 排坑·IPhone&IOS中不兼容正则中的断言匹配
阅文时长 | 1.14分钟 字数统计 | 1834.4字符 主要内容 | 1.问题切入 2.什么是断言匹配 3.断言匹配的替换方案 4.声明与参考资料 『排坑·IPhone&IOS中不兼容正则 ...
- c/c++排坑(4) -- c/c++中返回局部变量
返回c语言中的局部变量 先看一段代码猜猜,打印值: #include <iostream> using namespace std; char * func(); int main() { ...
- c/c++排坑(3) -- c/c++中的switch语句
switch语句的简单介绍 一个 switch 语句允许测试一个变量等于多个值时的情况.每个值称为一个 case,且被测试的变量会对每个 switch case 进行检查. switch(expres ...
- Java语言中的这些知识点有没有用过,工作中有没有入过这些坑?
在Java语言中,有一些相对生僻的知识,平时用的机会可能不是很多,但如果不了解不掌握这些知识点的话,也可能会掉入陷阱之中,今天我们就来初步梳理一下: 1. goto是java语言中的关键字. &quo ...
- 排坑·ASCII码为160的空格(nbsp)
阅文时长 | 2.83分钟 字数统计 | 1345.2字符 『排坑·ASCII码为160的空格(nbsp)』 编写人 | SCscHero 编写时间 | Wednesday, September 9, ...
- C语言中两位ASCII码可以表示汉字
最近偶然有人问到这个相关字符编码的问题,所以百度了下参考了这两个资料,进行了简单分析. ******************************************************** ...
- C语言中内存分配那些事儿
C程序的内存结构 C语言的之所以复杂,首先它的内存模型功不可没.不像某些那样的高级语言只需要在使用对象的时候,用new创建.所有之后的事情,你不需要操心.对于C语言,所有与内存相关的东西,都需要熟悉, ...
- go语言中的数组切片:特立独行的可变数组
go语言中的数组切片:特立独行的可变数组 初看go语言中的slice,觉得是可变数组的一种很不错的实现,直接在语言语法的层面支持,操作方面比起java中的ArrayList方便了许多.但是在使用了一段 ...
随机推荐
- javascript 数组总结
数组的创建: 数组可以使用Array构造函数来创建,或者使用[]快速创建. 1. Array构造函数创建数组: 无参数,创建空数组: var arry = new Array(); 参数为一个数字,指 ...
- C#实现马尔科夫模型例子
已知条件:三个缸N状态,每个缸中不同颜色球的个数M状态值,时间轴T,观察值序列O 参数:状态值序列,转移概率序列 求:概率 后台代码如下 , M = ;//N状态,M状态值 (0橙色,1绿色,2蓝色, ...
- LLDB 常用命令
dump: memory read --force --outfile [文件名] --binary [start_address] [end_address] 查找函数地址: 对于有调试符号的这样使 ...
- 应用程序无法正常启动 0xc0000013 vs2013
今天下午切换到Windows 优化代码,在debug 的时候一直出现这个问题,折腾了很久,发现原来是系统环境变量的问题,我之前装了双系统,讲原来win7 下的一块E盘删掉做了Linux 盘,而系统环境 ...
- timus 1018. Binary Apple Tree
1018. Binary Apple Tree Time limit: 1.0 secondMemory limit: 64 MB Let's imagine how apple tree looks ...
- kentico9开始移除的webpart
https://devnet.kentico.com/articles/fighting-featuritis https://blog.intercom.com/product-strategy-m ...
- bzoj1833
http://www.lydsy.com/JudgeOnline/problem.php?id=1833 2.5个小时就花在这上面了... 水到200题了...然并卵,天天做水题有什么前途... #i ...
- hibernate缓存机制详细分析(一级、二级、查询缓存,非常清晰明白)
本篇随笔里将会分析一下hibernate的缓存机制,包括一级缓存(session级别).二级缓存(sessionFactory级别)以及查询缓存,当然还要讨论下我们的N+1的问题. 随笔虽长,但我相信 ...
- PCB 钻孔补偿那点事
没有优秀的个人,只有优秀的团队,在团队共同的协作下,PCB CAM自动化[net处理]与[钻孔处理] 第一阶段开发项完成了,,后续工作可以转向PCB规则引擎开发了.这里说说PCB工程钻孔补偿的那点事, ...
- (function(){})();和(function(){}())每个括号的用途和区别
(function(){…})(); 这种写法是因为JS中没有块级作用域的概念,所以可以用lambda函数来模仿块级作用域,这个的作用是定义并立即调用一个lambda函数,这个函数中定义的任何变量,都 ...