extern, static, const 和 inline

在使用 Objective-C 编程的时候,偶尔也会使用到 C 语言的一些特性,extern、static、const 和 inline 这四个关键词是我对于其含义较为模糊的四个关键词,特写一篇笔记用来温习一下。

extern

要理解 extern 关键字的作用,首先要明白声明和定义的区别。定义变量时,编译器为该变量分配内存,并可能还将其内容初始化为某个值。声明变量时,编译器要求在其他地方定义变量。声明通知编译器存在由该名称和类型的变量,但编译器不需要为其分配内存,因为它是在其他地方分配的。extern 关键字表示“声明而不定义”。换句话说,它是一种显式声明变量或强制声明而无需定义的方法。

必须在程序的一个模块中精确定义一个变量。 如果没有定义或多于一个定义,则可能在链接阶段产生错误。 变量可以被声明多次,只要声明彼此一致并且与定义一致(头文件很有用)。 它可以在许多模块中声明,包括定义它的模块,甚至在同一模块中多次。 但是在模块中多次声明它通常是没有意义的。

extern 变量也可以在函数内声明。 在这种情况下,必须使用 extern 关键字,否则编译器会将其视为本地(自动)变量的定义,该变量具有不同的范围,生命周期和初始值。 此声明仅在函数内部可见,而不是在整个函数模块中可见。

应用于函数原型的 extern 关键字绝对没有任何内容(应用于函数定义的 extern 关键字当然是非语义的)。 函数原型始终是一个声明,而不是一个定义。 此外,在标准 C 中,函数始终是外部的,但某些编译器扩展允许在函数内定义函数。

Example:

File 1:

// Explicit definition, this actually allocates
// as well as describing
int Global_Variable;

// Function prototype (declaration), assumes 
// defined elsewhere, normally from include file.       
void SomeFunction(void);        

int main(void) {
    Global_Variable = 1;
    SomeFunction();
    return 0;
} 

File 2:

// Implicit declaration, this only describes and
// assumes allocated elsewhere, normally from include
extern int Global_Variable;  

// Function header (definition)
void SomeFunction(void) {       
    ++Global_Variable;
}

在此示例中,变量 Global_Variable 在 File 1 中定义。为了在 File 2 中使用相同的变量,必须声明它。 无论文件数量多少,全局变量只定义一次; 但是,它必须在包含定义的文件之外的任何文件中声明。

通常的方法是分配和实际定义进入 .c 文件,但仅仅声明和原型不分配,只是描述类型和参数,以便编译器可以正常工作,并且该信息属于 .h 头文件,其他人可以安全地 include 没有任何可能的冲突。

static

在 C 编程语言(及其紧密的后代,如 C++ 和 Objective-C)中,static 是一个保留字,用于控制生命周期(作为静态变量)和可见性(取决于链接)。

在声明变量或函数时作为前缀的 static 关键字可能具有其他效果,具体取决于声明发生的位置。

Static global variable

在源文件的顶层(在任何函数定义之外)声明为 static 的变量仅在整个文件中可见。 在此用法中,关键字 static 称为“访问说明符”。

Static function

类似地,静态函数 - 在源文件的顶层(在任何类定义之外)声明为静态的函数 - 仅在整个文件中可见。

Static local variables

在函数内声明为静态的变量是静态分配的,因此在整个程序执行期间保持其内存单元,同时具有与自动局部变量(auto 和 register)相同的可见范围,这意味着保持函数的本地性。 因此,当一次调用时函数放入其静态局部变量的任何值在再次调用函数时仍然存在。

const

在 C 语言中,const 是类型的一部分,而不是对象的一部分。例如,在 C 中,const int x = 1; 声明一个 const int 类型的对象 x - const 是该类型的一部分,就好像它被解析为“(const int)x”

这有两个微妙的结果。 首先,const 可以应用于更复杂类型的部分 - 例如,int const * const x; 声明一个指向常量整数的常量指针,而 int const * x; 声明一个指向常量整数的变量指针,并且 int * const x; 声明一个指向变量整数的常量指针。 其次,因为 const 是类型的一部分,所以它必须作为类型检查的一部分匹配。

虽然常量在程序运行时不会更改其值,但是在程序运行时,声明为 const 的对象确实可能会更改其值。 一个常见的例子是嵌入式系统中的只读寄存器,如数字输入的当前状态。 数字输入的数据寄存器通常声明为 const 和 volatile。 这些寄存器的内容可能会在程序执行任何操作(volatile)时发生变化,但你也不应该向它们写入(const)。

使用方式

在 C 中,所有数据类型(包括用户定义的数据类型)都可以声明为 const,而 const-correctness 规定所有变量或对象都应该声明为这样,除非需要修改它们。 这种对 const 的主动使用使得值“更容易理解,跟踪和推理”,从而提高了代码的可读性和可理解性,使团队工作和维护代码更简单,因为它传达了有关值的预期用途的信息。 在推理代码时,这可以帮助编译器和开发人员。 它还可以使优化编译器生成更高效的代码。

简单的数据类型

对于简单的非指针数据类型,应用 const 限定符很简单。 由于历史原因,它可以在类型的任何一侧出现(即,const char foo ='a';等同于 char const foo ='a';)。 在某些实现中,在类型的两侧使用 const(例如,const char const)会生成警告但不会生成错误。

指针和引用

对于指针和引用类型,const 的含义更复杂 - 指针本身或指向的值或两者都可以是 const。此外,语法可能令人困惑。指针可以声明为指向可写值的 const 指针,或指向 const 值的可写指针,或指向 const 值的 const 指针。 const 指针不能重新分配以指向与最初分配的对象不同的对象,但它可用于修改它指向的值(称为指针对象)。因此,引用变量是 const 指针的替代语法。另一方面,指向 const 对象的指针可以重新分配以指向另一个内存位置(应该是相同类型或可转换类型的对象),但它不能用于修改它的内存指向还可以声明指向 const 对象的 const 指针,既不能用于修改指针,也不能重新指定指向另一个对象。以下代码说明了这些细微之处:

void Foo( int * ptr,
          int const * ptrToConst,
          int * const constPtr,
          int const * const constPtrToConst )
{
    *ptr = 0; // OK: modifies the "pointee" data
    ptr  = NULL; // OK: modifies the pointer

    *ptrToConst = 0; // Error! Cannot modify the "pointee" data
    ptrToConst  = NULL; // OK: modifies the pointer

    *constPtr = 0; // OK: modifies the "pointee" data
    constPtr  = NULL; // Error! Cannot modify the pointer

    *constPtrToConst = 0; // Error! Cannot modify the "pointee" data
    constPtrToConst  = NULL; // Error! Cannot modify the pointer
}

遵循通常的 C 约定声明,声明遵循使用,并且指针中的 * 写在指针上,表示解除引用。 例如,在声明 int *ptr 中,解除引用的形式 *ptr是一个 int,而引用形式 ptr 是一个指向 int 的指针。 因此 const 将名称修改为右侧。

int *ptr; // *ptr is an int value
int const *ptrToConst; // *ptrToConst is a constant (int: integer value)
int * const constPtr; // constPtr is a constant (int *: integer pointer)
int const * const constPtrToConst; // constPtrToConst is a constant (pointer)
                                   // as is *constPtrToConst (value)

inline

在 C 和 C++ 编程语言中,内联函数是使用关键字 inline 限定的函数;这有两个目的。首先,它充当编译器指令,建议(但不要求)编译器通过执行内联扩展替换函数体,即通过在每个函数调用的地址处插入函数代码,从而节省开销一个函数调用。在这方面,它类似于寄存器存储类说明符,它类似地提供了一个优化提示。内联的第二个目的是改变链接行为;这个细节很复杂。这是必要的,因为 C/C++ 单独的编译+链接模型,特别是因为函数的定义(主体)必须在使用它的所有转换单元中重复,以允许在编译期间内联,如果函数具有外部链接,在链接过程中导致冲突(它违反了外部符号的唯一性)。

内联函数可以用 C 或 C++ 编写,如下所示:

static inline void swap(int *m, int *n)
{
  int temp = *m;
  *m = *n;
  *n = temp;
}

然后,声明如下:

swap(&x, &y);

可以被翻译成(如果编译器决定进行内联,通常需要启用优化):

int temp = x;
x = y;
y = temp;

当实现执行大量交换的排序算法时,这可以提高执行速度。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容