c语言复杂声明的解释

目标:分析一个将c语言复杂申明解释为英语的一个程序

首先介绍一种简单的读声明的方法,来源网络

1.右左法则

从标识符开始(或者最内层的结构,如果不存在标识符的话,通常出现于函数指针),首先向右看,直到遇到 ) 括号或者结束,看到什么就说出来;然后向左看,直到遇到 ( 括号或者回到行首,看到什么就说出来。跳出一层括号,重复上述过程:右看看,说出来;左看看,说出来。直到你说出变量的类型或者返回值(针对函数指针),也就表示你把声明都读完了。

2.例子

int *a[3];

从标示符a开始,向右看,a是一个包含三个元素的数组,向左看数组的每个元素是指针,向右看到结尾了,向左看,指针指向int。综合:a是一个三元数组,数组的每个元素是指针,指向int。

int (*a)[3];

从标示符a开始,向右看,遇到(,向左看,a是一个指针,向右看,指针指向一个包含三个元素的数组,向左看,数组类型为int。

int (*foo)();

从标示符foo开始,向右看,遇到),向左看,foo是一个指针,想左看,指针指向一个无参数的函数,向右看,函数返回类型位int。

int (*(*vtable)[])();

这个比较复杂,还是从标示符table开始,向右看,遇到),向左看,table是一个指针,指针指向一个数组,向左看,数组的元素是指针,向右看,指针指向一个无参数的函数,向右看,函数的返回类型为int。

实践完这些例子,基本就能明白右左法则。此法则能帮助我们快速的理解复杂声明,接下来将分析一个《c程序语言设计》中的一个复杂声明解释程序,但所用方法与此处的左右法则不同。

此程序是基于声明符语法编写,其语法简化形式:

dcl: 前面带有可选的direct-dcl

direct-dcl: name,(dcl),direct-dcl(),direct-dcl[ ]

简而言之,声明符dcl就是前面可能带有多个*的direct-dcl,direct-dcl可以是name,由一对圆括号括起来的dcl,后面跟有一对圆括号的direct-dcl,后面跟有用方括号括起来的direct-dcl。

代码

#include <stdio.h>
#include <string.h>
#include <ctype.h> #define MAXTOKEN 100 enum { NAME, PARENS, BRACKETS }; void dcl(void);
void dirdcl(void);
int gettoken(void);
int get_line(char s[], int lim);
int tokentype;
char token[MAXTOKEN];
char name[MAXTOKEN];
char datatype[MAXTOKEN];
char out[1000]; //衔接词 #define BUFSIZE 1024
char buf[BUFSIZE];
char line[BUFSIZE];
int bufp = 0;
int linep = 0; int getch(void){
if (bufp > 0 || (line[linep] !='\0'))
return (bufp > 0) ? buf[--bufp] : line[linep++];
else
return EOF;}
void ungetch(int c)
{
if (bufp >= BUFSIZE)
printf("ungetch: too many characters ");
else
buf[bufp++] = c;
} int main(int argc, char* argv[])
{
get_line(line, BUFSIZE);
while (gettoken() != EOF) {
strcpy(datatype, token);
out[0] ='\0';
dcl();
if (tokentype !='\n')
printf("syntax error ");
printf("%s: %s %s ", name, out, datatype);
}
return 0;
} int gettoken(void)
{
int c;
char *p = token;
while ((c = getch()) == ' ' || c == '\t') //去除空格和换行符
;
if (c == '(') {
if ((c = getch()) == ')') {
strcpy(token, "()");
return tokentype = PARENS;
} else {
ungetch(c);
return tokentype = '(';
}
}
else if (c == '[') {
for (*p++ = c; (*p++ = getch()) != ']';)
;
*p ='\0';
return tokentype = BRACKETS;
}
else if (isalpha(c)) {
for (*p++ = c; isalnum(c = getch());)
*p++ = c;
*p ='\0';
ungetch(c);
return tokentype = NAME;
}
else
return tokentype = c;
}
int get_line(char s[], int lim)
{
int c, i;
i = 0;
while (--lim > 0 && (c = getchar()) != EOF && c !='\n')
s[i++]=c;
if (c == '\n')
s[i++]=c;
s[i]='\0';
return i;
}
void dcl(void)
{
int ns;
for (ns = 0; gettoken() == '*';)
ns++;
dirdcl();
while (ns-- > 0)
strcat(out, " pointer to");
} void dirdcl(void)
{
int type; if (tokentype == '(') {
dcl();
if (tokentype != ')')
printf("error: missing ) ");
}
else if (tokentype == NAME)
strcpy(name, token);
else
printf("error: expected name or (dcl) ");
while ((type = gettoken()) == PARENS || type == BRACKETS)
if (type == PARENS)
strcat(out, " function returning");
else {
strcat(out, " array");
strcat(out, token);
strcat(out, " of");
}
}

c语言复杂申明解释程序的更多相关文章

  1. C语言函数申明关键字inline

    内联inline是给编译器的优化提示,如果一个函数被编译成inline的话,那么就会把函数里面的代码直接插入到调用这个函数的地方,而不是用调用函数的形式.如果函数体代码很短的话,这样会比较有效率,因为 ...

  2. c/c++排坑(5) -- c语言中的申明

    C语言的申明总是令人头大,对于这块内容也一直让我头疼.希望通过这篇博客能够稍微梳理一下.材料和例子来源于<C专家编程> 一.C语言的申明的优先级规则 先来个例子,看看下面这行C代码到底是个 ...

  3. C语言中的声明解析规则——数组,指针与函数

    摘要:C语言的申明存在的最大问题是:你无法以一种人们所习惯的自然方式和从左向右阅读一个声明,在引入voliatile和const关键字以后,情况更加糟糕了.由于这些关键字只能出现在声明中,是的声明形式 ...

  4. 彻底搞定C语言指针(精华版)

    1.语言中变量的实质 要理解C指针,我认为一定要理解C中“变量”的存储实质, 所以我就从“变量”这个东西开始讲起吧! 先来理解理解内存空间吧!请看下图: 内存地址→ 6 7 8 9 10 11 12 ...

  5. go语言调度器源代码情景分析之三:内存

    本文是<go调度器源代码情景分析>系列 第一章 预备知识的第2小节. 内存是计算机系统的存储设备,其主要作用是协助CPU在执行程序时存储数据和指令. 内存由大量内存单元组成,内存单元大小为 ...

  6. 基于windows IIS的C语言CGI WEB服务器环境搭建

    网页编程对我来说特别亲切,因为我就是从html.ASP.PHP一步步接触编程的.自己的编程爱好也是从那里一点一点被满足.不过离开大学之后很久没有碰过WEB了,最近看到嵌入式中的涉及到的web服务器,了 ...

  7. Python - Tips

    01 - input与raw_input的区别 input() #可以直接输入数字,但输入字符的要用引号''或者双引号"" raw_input() #将所有的输入都直接当作一串字符 ...

  8. Unity3D着色器Shader编程入门(一)

    自学Unity3D也有大半年了,对Shader一直不敢入坑,最近看了些资料,以及通过自己的实践,对Shader还是有一点了解了,分享下仅作入门参考. 因Shader是对图像图像渲染的,学习前可以去了解 ...

  9. 彻底搞定 C/C++ 指针

    1.语言中变量的实质 要理解C指针,我认为一定要理解C中“变量”的存储实质, 所以我就从“变量”这个东西开始讲起吧! 先来理解理解内存空间吧!请看下图: 内存地址→ 6 7 8 9 10 11 12 ...

随机推荐

  1. Java 之 OutputStreamReader类

    OutputStreamReader类 1.概述 转换流 java.io.OutputStreamReader ,是Writer的子类,是从字符流到字节流的桥梁. 它使用指定的字符集将字符编码为字节. ...

  2. Linux之virtualenv和virtualenvwrapper

    一,介绍 在使用 Python 开发的过程中,工程一多,难免会碰到不同的工程依赖不同版本的库的问题:亦或者是在开发过程中不想让物理环境里充斥各种各样的库,引发未来的依赖灾难.此时,我们需要对于不同的工 ...

  3. RestFramework之频率组件

    一.频率组件的使用 频率组件的存在对我们这web开发有着很大的影像,它的作用就是限制用户在一段时间内访问的次数. 下面让我们介绍一下频率组件怎样使用 1.首先需要导入 from rest_framew ...

  4. Bootstrap学习地址

    第一步:https://www.runoob.com/bootstrap/bootstrap-tutorial.html  //菜鸟教程 第二步:https://v3.bootcss.com/gett ...

  5. springboot 配置 中查找application.properties中对应的数据,添加对应的prefix前缀

    @ConditionalOnProperty(prefix = "spring.redis", name = "enabled", havingValue = ...

  6. 性能三 powerVR specfication

    2.Optimising Geometry Interleaving Attributes VBO Draw call size Triangle Size  32个像素/primitive    - ...

  7. linux系统常用软件

    输入法---搜狗输入法 音乐播放器---网易云音乐 邮箱---

  8. 从设计稿到实现React应用(分类数据处理)

    1. 确定设计稿和数据 设计稿: 数据: [ {category: "Sporting Goods", price: "$49.99", stocked: tr ...

  9. Linux 理解Linux的memory overcommit 与 OOM Killer

    Memory Overcommit的意思是操作系统承诺给进程的内存大小超过了实际可用的内存.一个保守的操作系统不会允许memory overcommit,有多少就分配多少,再申请就没有了,这其实有些浪 ...

  10. Oracle RAC 服务启动流程

    启动流程步骤层次梳理:第一层:OHASD 启动: cssdagent - 负责启动 CSSD 的 Agent.orarootagent - 负责启动所有 root 用户下的 ohasd 资源 的Age ...