在C++中,static有一个感觉被较少提及的用法:修饰非成员函数,这个用法实际是从C语言继承来的。其作用是表明这个函数只在当前编译单元中有效。这就使这个函数的所有引用在编译时就可以全部确定,无需进入链接阶段,链接器没有机会看到这个函数相关的一切符号,无论导入还是导出(理论上,实际编译器如何处理这个事情可能不尽相同,未作深入研究)。即使多个编译单元都包含相同的signature相同名字的函数,链接器也不会报错或合并符号。

比如以下代码:

// main.cpp

void foo();
void bar(); int main(int argc, const char* argv[]) {
foo();
bar();
return ;
}
// foo.cpp

#include <iostream>
static void test(){
std::cout << "test from foo.cpp" << std::endl;
} void foo(){
test();
}
// bar.cpp

#include <iostream>
static void test() {
std::cout << "test from bar.cpp" << std::endl;
} void bar() {
test();
}

最终编译成功,运行时输出

test from foo.cpp
test from bar.cpp

虽然这个static用法是从C语言继承而来,但却并没有被扩展到class上,static关键字无法用来修饰static(这也算是一个明智的决定,否则static class一定会晕倒一堆人,尤其从java/c#过来的人)。但C++也提供了代替方案以解决类名字冲突的问题,那就是匿名名字空间(anonymous namespace,我个人翻译的,有点绕口),望文生义,就是没有名字的名字空间,形如

namespace {
//....
}

放在这个名字空间中的类、函数、变量对当前编译单元有效,效果跟static修改非成员函数一样。

// foo.h

namespace test {

class foo {
public:
foo();
}; }
// bar.h

namespace test {

class bar {
public:
bar();
}; }
// foo.cpp

#include <iostream>
#include "foo.h" namespace test {
namespace { class local_class {
public:
void test() {
std::cout << "test from foo.cpp" << std::endl;
}
}; } foo::foo() {
local_class l;
l.test();
} }
// bar.cpp

#include <iostream>
#include "bar.h" namespace test {
namespace { class local_class {
public:
void test() {
std::cout << "test from bar.cpp" << std::endl;
}
}; } bar::bar() {
local_class l;
l.test();
} }
// main.cpp

#include "foo.h"
#include "bar.h" int main(int argc, const char*argv[]) {
test::foo f;
test::bar b;
return ;
}

运行时输出结果

test from foo.cpp
test from bar.cpp

PS: 看到有人总结static的用途,将各种static的用途归结为一种,但忘记在哪里看到了,可能出自标准定义, 大致意思是 “被static修饰的符号只在其所在作用域(scope)中有效,并在其所在作用域有效其间一直有效(或者说与其所在作用域生命周期相同)”

【UPDATE】

编译器依然会为这些static函数或匿名名字空间中的类名生成符号,但会标记为本地符号(local symbol),通知链接器这些符号只对当前.obj文件有效。

C++ 匿名名字空间及静态非成员函数的更多相关文章

  1. C++匿名名字空间

    转自:http://blog.csdn.net/eric_arrow/article/details/8978905 名字空间(namespace),是C++提供的一个解决符合名字冲突的特性.标准规定 ...

  2. Python虚拟机函数机制之名字空间(二)

    函数执行时的名字空间 在Python虚拟机函数机制之无参调用(一)这一章中,我们对Python中的函数调用机制有个大概的了解,在此基础上,我们再来看一些细节上的问题.在执行MAKE_FUNCTION指 ...

  3. C++名字空间/C++命名空间

    0.序言 名字空间是C++提供的一种解决符号名字冲突的方法. 一个命令空间是一个作用域,在不同名字空间中命名相同的符号代表不同的实体. 通常,利用定义名字空间的办法,可以使模块划分更加方便,减少模块间 ...

  4. python tips:作用域与名字空间

    Python具有静态作用域,变量的作用域由它定义的位置决定,而与调用的位置无关. a = 2 def f(): a = 2 第一行的a的作用域是全局作用域,作用于定义位置后面的所有位置. 第四行的a的 ...

  5. C++笔记--名字空间和异常

    名字空间 成员函数可以在名字空间的定义里去声明,然后再去采用一种定义方式例如:namespace__name::member_name的方式去定义这个成员函数 namespace parser{ do ...

  6. 将对象的所有属性名放到一个数组中 || 获得对象的所有属性名 || return;不具有原子性 || 怎样自己制作异常|| 判断对象有没有某个属性 || 当传递的参数比需要的参数少的时候,没有的值会被赋予undefined || 获得函数实际传递的参数 || 怎么用函数处理一个对象 || 用一个名字空间定义一个模块所有的函数 || 给一个对象添加方法

    获得对象的所有属性名 || 将对象o的所有属性名放到数组中 var o = {x:1,y:2,z:3}; var arr = []; var i = 0; for(arr[i++] in o){};/ ...

  7. Python 3里,reduce()函数已经被从全局名字空间里移除了,它现在被放置在fucntools模块里

    reduce函数:在Python 3里,reduce()函数已经被从全局名字空间里移除了,它现在被放置在fucntools模块里 用的话要 先引入:>>> from functool ...

  8. C和C++中的名字空间和作用域

    C和C++中的名字空间和作用域 C语言中有名字空间这个概念吗? 提到名字空间(或者可能更普遍的叫法,命名空间),很可能先想到的是C++,甚至是C#.C中没有名字空间吧?一开始我也是这样认为的,直到我看 ...

  9. Python 之作用域和名字空间

    作用域与名字空间 Python有一个核心概念是名字空间(namespace),namespace是一个name到object 的映射关系,Python有很多namespace,因此,在代码中如果碰到一 ...

随机推荐

  1. Gradle学习目录总结

    如果是你想干的事情,在别人看起来可能是很难的一件事,不过你自己很喜欢,你不会觉得很苦.我开始创业那会是28岁.对我来讲,我创业的目的不是为了自己当老板,我希望有一个平台有一个环境,我可以控制一些资源, ...

  2. android anim 动画效果(转)

    动画效果编程基础--AnimationAndroid      动画类型      Android的animation由四种类型组成      XML中    alpha    渐变透明度动画效果   ...

  3. HDU 4455(dp)

    题意:给定一个序列ai,个数为n.再给出一系列w:对于每个w,求序列中,所有长度为w的连续子串中的权值和,子串权值为子串中不同数的个数. 思路:动态规划,用dp[w]表示当前长度为w的时候的权值和.显 ...

  4. HttpClient4.0

    ****************************HttpClient4.0用法***************************** 1.初始化HttpParams,设置组件参数 //Ht ...

  5. 12、SQL Server 行列转换

    SQL Server 行转列 在SQL Server 2005中PIVOT 用于将列值转换为列名(行转列),在SQL Server 2000中是没有这个关键字的 只能用case语句实现. --创建测试 ...

  6. 关于PHP导入项目的时候导入不了的情况

    导入的时候,会发现明明是一个手动创建的一个项目, 才能导入, 有时候会发现这样导入不了的情况 那是因为,可能这个项目是手动创建的,如果通过IDE可能看不出来 不过如果你进入项目的根目录的时候就会知道 ...

  7. 阿里云OSS存储开发(一)

    Step 1. 初始化一个OSSClient OSSClient是与OSS服务交互的客户端,SDK的OSS操作都是通过OSSClient完成的. 下面代码新建了一个OSSClient: using A ...

  8. c - 递归年龄

    /* 题目:有 5 个人坐在一起,问第五个人多少岁?他说比第 4 个人大 2 岁.问第 4 个人岁数,他说比第3 个人大 2 岁.问第三个人,又说比第 2 人大两.问第 2 个人,说比第一个人大两岁. ...

  9. XML样本(格式没区别,但是一个有结果 一个没结果)

    xml样本01<orderlist>        <order>        <orderid>1</orderid>        <ord ...

  10. JetBrains公司的IDE使用技巧

    1.自定义Live Templates:   点击+添加自己的.最后记住要点击,change或default来设置在哪些文件上使用代码片段.