C++ 匿名名字空间及静态非成员函数
在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++ 匿名名字空间及静态非成员函数的更多相关文章
- C++匿名名字空间
转自:http://blog.csdn.net/eric_arrow/article/details/8978905 名字空间(namespace),是C++提供的一个解决符合名字冲突的特性.标准规定 ...
- Python虚拟机函数机制之名字空间(二)
函数执行时的名字空间 在Python虚拟机函数机制之无参调用(一)这一章中,我们对Python中的函数调用机制有个大概的了解,在此基础上,我们再来看一些细节上的问题.在执行MAKE_FUNCTION指 ...
- C++名字空间/C++命名空间
0.序言 名字空间是C++提供的一种解决符号名字冲突的方法. 一个命令空间是一个作用域,在不同名字空间中命名相同的符号代表不同的实体. 通常,利用定义名字空间的办法,可以使模块划分更加方便,减少模块间 ...
- python tips:作用域与名字空间
Python具有静态作用域,变量的作用域由它定义的位置决定,而与调用的位置无关. a = 2 def f(): a = 2 第一行的a的作用域是全局作用域,作用于定义位置后面的所有位置. 第四行的a的 ...
- C++笔记--名字空间和异常
名字空间 成员函数可以在名字空间的定义里去声明,然后再去采用一种定义方式例如:namespace__name::member_name的方式去定义这个成员函数 namespace parser{ do ...
- 将对象的所有属性名放到一个数组中 || 获得对象的所有属性名 || return;不具有原子性 || 怎样自己制作异常|| 判断对象有没有某个属性 || 当传递的参数比需要的参数少的时候,没有的值会被赋予undefined || 获得函数实际传递的参数 || 怎么用函数处理一个对象 || 用一个名字空间定义一个模块所有的函数 || 给一个对象添加方法
获得对象的所有属性名 || 将对象o的所有属性名放到数组中 var o = {x:1,y:2,z:3}; var arr = []; var i = 0; for(arr[i++] in o){};/ ...
- Python 3里,reduce()函数已经被从全局名字空间里移除了,它现在被放置在fucntools模块里
reduce函数:在Python 3里,reduce()函数已经被从全局名字空间里移除了,它现在被放置在fucntools模块里 用的话要 先引入:>>> from functool ...
- C和C++中的名字空间和作用域
C和C++中的名字空间和作用域 C语言中有名字空间这个概念吗? 提到名字空间(或者可能更普遍的叫法,命名空间),很可能先想到的是C++,甚至是C#.C中没有名字空间吧?一开始我也是这样认为的,直到我看 ...
- Python 之作用域和名字空间
作用域与名字空间 Python有一个核心概念是名字空间(namespace),namespace是一个name到object 的映射关系,Python有很多namespace,因此,在代码中如果碰到一 ...
随机推荐
- Android(java)学习笔记244:多媒体之SurfaceView
1. SurfaceView: 完成单位时间内界面的快速切换(游戏界面流畅感). 我们之前知道一般的View,只能在主线程里面显示,主线程中更新UI.但是SurfaceView可以在子线程中里 ...
- nginx url 重写 [转]
本文转自 http://www.jbxue.com/article/2187.html 本文介绍nginx URL重写的相关知识,包括301重定向的内容等,希望对大家有所帮助. nginx rewri ...
- HTML5之部分显示
- MSSQLSERVER服务不能启动
自从用上mysql,好久没打开sqlserver了,今天本想打开调试下MFC连接sqlserver,然后意外发现不能登录,之后我以为是sql服务没启动,然后去启动,还是没用,并且MSSQLSERVER ...
- carousel
<!DOCTYPE html> <html lang="en" ng-app="mainApp"> <head> <m ...
- Linq101-Grouping Operators
using System; using System.Collections.Generic; using System.Linq; namespace Linq101 { class Groupin ...
- OUTPUT 在insnert delete update 的神奇功效
Inserted deleted 个人理解 应该是两个 临时表 分别存储 变动后的数据集 和 变动前的数据集 使用例子: 1.对于INSERT,可以引用inserted表以查询新行的属性 ...
- HTML5 File 对象
实例说明1: <div class="container"> <input type="file" id="file" m ...
- (转)PHP函数spl_autoload_register()用法和__autoload()介绍
转--http://www.jb51.net/article/29624.htm 又是框架冲突导致__autoload()失效,用spl_autoload_register()重构一下,问题解决 ...
- 016_openxml_forxml
016_openxml_forxml --openxml*********************************************************************** ...