NeoRAGEx2002曾经有一篇文章提到这个问题,但是有很多内容并没有包括,例如const和__declspec。

最近我遇到一些这方面的问题,感觉有必要做一个系统性的总结。后来经过一些实验,得出了一些结论,在这里分享给大家。

C风格变量声明

C风格的变量声明,如

extern __declspec(dllexport) void(__stdcall * const p[10])(int a, int b);

和其他语言都不一样,规则很不直观,需要一些理解

要理解它为什么是现在这样,首先需要理解它的思路,最初C的类型的写法,模拟了变量的实际使用

例如

#include <functional>

int add(int a, int b)
{
return a + b;
} int main(int argc, char **argv)
{
int a[10] = {};
int a0 = a[0]; int *p = &a0;
int p0 = *p; int (*f)(int a, int b) = add;
int result = (*f)(1, 2);
int result2 = f(1, 2); //使用函数指针时可以将(*f)简记为f,但声明时不能,因为会和函数原型混淆 int g(int a, int b); //函数原型 auto h = static_cast<int (*)(int a, int b)>(f); //转换时,需要用到没有变量名的类型声明,此时只需要去掉变量名本身 auto i = std::function<int(int a, int b)>(h); //std::function的参数是函数本身的类型
typedef int FuncType(int a, int b);
auto j = std::function<FuncType>(i); //可以用typedef来将函数本身的类型起一个名字 return 0;
}

因此,要声明一个变量,可以先写出它的实际的用法,再转换成变量声明

比如说我想要写一个包含10个函数指针元素的数组,函数为(int, int) -> int,可以先写调用

int b = (*a[0])(1, 2);

再从调用写声明

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

注意C中的运算符结合优先级,首先是按括号,然后是右边的索引[]、函数()、成员.或者->,然后是左边的*

C风格变量声明 - const, method const

int add(int a, int b)
{
return a + b;
} class A
{
private:
int a{};
int b{};
public:
void foo() const //函数后方的const表示函数调用时this为const的
{
this->a = 0; //编译错误,this为A * const;
b = 0; //编译错误
const_cast<A *>(this)->a = 0; //使用const_cast将this转换为A *则可以编译成功
}
}; int main(int argc, char **argv)
{
//const加一个类型名和类型名加const效果相同
const int a1 = 0;
int const a2 = 0; //const不与后方的*结合
const int *a3 = 0;
int const *a4 = 0;
a3 = a4; //a3、a4为到不可改变的int的指针,但指针本身可以改变 //const与前方的*结合
int (* const a5) = 0;
int * const a6 = 0;
a5 = a6; //编译错误,a5不可改变 //函数指针加const的规则与正常指针相同
int (* const f)(int a, int b) = add; //成员函数指针加const的规则与正常指针相同
void (A::* const g)() const = &A::foo; return 0;
}

C风格变量声明 - extern, __declspec, __stdcall等

extern、__declspec(Visual C++)以及一些其他链接标记,是变量的属性,直接加在最前面

__stdcall(Visual C++)等调用规范,是函数的属性,加在函数名称前面或者函数指针的*前面

这样我们就可以明白下面这个声明的含义

extern __declspec(dllexport) void(__stdcall * const p[10])(int a, int b);
//一个extern __declspec(dllexport)的变量p
//p是一个长度为10的数组,数组的每个元素是void(__stdcall * const)(int a, int b)
//这些元素是一个const的函数指针,函数本身的调用规范是__stdcall
//函数的参数是(int a, int b),返回值是void

再议C风格变量声明的更多相关文章

  1. 再议Java中的static关键字

    再议Java中的static关键字 java中的static关键字在很久之前的一篇博文中已经讲到过了,感兴趣的朋友可以参考:<Java中的static关键字解析>. 今天我们再来谈一谈st ...

  2. TypeScript 素描-变量声明

    博文读自 TypeScript 官方文档而来,不具有学习性,仅是本人学习时记录以供日后翻阅 ,有学习TypeScript的朋友还请去看更为详细的官方文档 /* 变量声明在之前的js中一直是使用var关 ...

  3. 变量声明---let,const,解构

    let在很多方面与var是相似的,但是可以帮助大家避免在JavaScript里常见一些问题. const是对let的一个增强,它能阻止对一个变量再次赋值. 块作用域 当用let声明一个变量,它使用的是 ...

  4. JavaScript变量声明提前

    上周四吃完午饭,leader发了一道JavaScript的题目给我们做,我们Team里面有做前端的,有做后台的,也有做mobile web的,所以大家对题目的理解各自都不一样,然后在QQ讨论组里面进行 ...

  5. 浅谈C语言变量声明的解析

    C语言本身提供了一种不甚明确的变量声明方式——基于使用的声明,如int *a,本质上是声明了*a的类型为int,所以得到了a的类型为指向int的指针.对于简单类型,这样声明并不会对代码产生多大的阅读障 ...

  6. 变量声明提升 Vs. 函数声明提升

    1. 变量声明提升 先看以下代码: 1)var in_window = "a" in window; console.log(in_window); 2)var in_window ...

  7. Python的类变量和对象变量声明解析

    Python的类和C++一样,也都是存在两种类型的变量,类变量和对象变量!前者由类拥有,被所有对象共享,后者由每个对象独有.这里我主要想讨论一下他们的声明办法. 首先说的是对象变量: 只要是声明在类的 ...

  8. JavaScript中的作用域与函数和变量声明的提升

      var foo = 1; function bar() {     if (!foo) {         var foo = 10;     }     alert(foo); } bar(); ...

  9. 【C语言探索之旅】 第一部分第四课第二章:变量的世界之变量声明

    内容简介 1.课程大纲 2.第一部分第四课第二章:变量的世界之变量声明 3.第一部分第四课第三章预告:变量的世界之显示变量内容 课程大纲 我们的课程分为四大部分,每一个部分结束后都会有练习题,并会公布 ...

随机推荐

  1. 网络请求三方库——OkHttp

    我们知道在Android开发中是可以直接使用现成的API进行网络请求的,就是使用 HttpClient 和 HttpURLConnention ,而Android 4.4 之后 HttpClient  ...

  2. 【转】libevent和基于libevent的网络编程

    转自: http://www.cnblogs.com/nearmeng/p/4043548.html 1 libevent介绍和安装 介绍 libevent是一个轻量级的基于事件驱动的高性能的开源网络 ...

  3. php5.3到php7.0.x新特性介绍

    <?php /*php5.3*/ echo '<hr>'; const MYTT = 'aaa'; #print_r(get_defined_constants()); /* 5.4 ...

  4. PowerProfile.java与power_profile.xml

    转载自http://blog.csdn.net/green1900/article/details/42427871 现在诸多关于电池管理的应用做的极其绚烂,可实现如耗电应用排行.剩余时间计算.关闭耗 ...

  5. python的with...as用法

    with...as叫做上下文管理器,作用是进入一个对象的作用域和离开时,可以执行执行一定的操作.这个操作是可以自己 设定的. 写个例子学习一下: class test(): def __init__( ...

  6. 【学】React的学习之旅1

    React的学习之旅1 单标签要有斜杠代表结束 用React.createClass()方法时,赋值后的组件名称首字母一定要大写 一定要先定义组件,再用ReactDOM.render调用 组件里ren ...

  7. 多个Python环境的构建:基于virtualenv 包

    假如一台计算中安装多个Python版本,而不同版本可能会pip安装不同的包,为了避免混乱,可以使用virtualenv包隔离各个Python环境,实现一个Python版本对应一套开发环境. 本地概况: ...

  8. HDU 2795 Billboard(区间求最大值的位置update的操作在query里做了)

    Billboard 通过这题,我知道了要活用线段树的思想,而不是拘泥于形式, 就比如这题 显然更新和查询放在一起很简单 但如果分开写 那么我觉得难度会大大增加 [题目链接]Billboard [题目类 ...

  9. java获取服务器IP地址及MAC地址的方法

    这篇文章主要介绍了java编程实现获取机器IP地址及MAC地址的方法,实例分析了Java分别针对单网卡及多网卡的情况下获取服务器IP地址与MAC地址的相关技巧,需要的朋友可以参考下   本文实例讲述了 ...

  10. 使用VS2013逆向生成UML类图

    引自http://blog.csdn.net/funnyfu0101/article/details/7705173 首先.打开工程,[体系结构]->[新建关系图] 生成一个类图 然后[体系结构 ...