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]如何创建一个View的分割线
如何创建一个View的分割线,如上图 我们见介绍三种可以创建看起来很不错的view的分割线,如在button之间添加分割线. 这个例子是将为LinearLayout内的三个Button间添加分割线. ...
- picasso_强大的Android图片下载缓存库
tag: android pic skill date: 2016/07/09 title: picasso-强大的Android图片下载缓存库 [本文转载自:泡在网上的日子 参考:http://bl ...
- Chain of Responsibility 责任链模式
简介 责任链模式是一种对象的行为模式.在责任链模式里,很多对象由每一个对象对其[下家]的引用而连接起来形成一条链,请求在这个链上[传递],直到链上的某一个对象决定处理此请求.发出这个请求的客户端并不知 ...
- 使用ef code first模式,在部署后服务器端把数据库同步到最新版本的方法
共有两种方法: 1.使用migrate.exe 具体使用方法请参考 msdn migrate使用方法,这里只做介绍 复制migrate.exe 在使用 NuGet 安装实体框架时,migrate.ex ...
- HttpClient 发送图片
var httpClient = new HttpClient(); using (FileStream fs = new FileStream("C:\\1.jpg", File ...
- call, apply, bind作用
call, apply作用就是(改变方法中的this指向)借用别人的方法来调用,就像调用自己的一样 function Person(name) { this.name = name; } Person ...
- ThinkPHP 的CURD 基本操作
说起CURD,懂点SQL的人都知道,就是增删改查,做业务系统的时候,往往离不开这CURD,最近也是刚刚接触ThinkPHP,ThinkPHP的灵活性是比原生PHP好用的多,下面我就简单的介绍一下我的学 ...
- UNIX基础知识
一.线程 线程是进程某程序段的一次运行. 1.线程共享资源,利用共享的资源,线程很容易能够互相通信 (1)进程代码段:每个线程有各自的寄存器组,在运行时期拷贝给cpu寄存器,来确定运行的是哪段代码段. ...
- 利用php的ob缓存机制实现页面静态化
利用php的ob缓存机制实现页面静态化 首先介绍一下php中ob缓存常用到的几个常用函数ob_start():开启缓存机制ob_get_contents():获取ob缓存中的内容ob_clean()清 ...
- python3数据类型--数字
数字 Python数字数据类型用于存储数值.数字数据类型是不允许改变的,所以如果改变数字数据类型的值,将重新分配内存空间. 以下实例在变量赋值时Number对象被创建: #!/usr/bin/env ...