一:背景
相信大家在分析 dump 时,经常会看到 WKS
和 SRV
这样的字眼,如下代码所示:
00007ffa`778a07b8 coreclr!WKS::gc_heap::segment_standby_list = 0x00000000`00000000 00007ffa`778a3870 coreclr!WKS::qpf = 0x989680 00007ffa`7789da30 coreclr!SVR::heap_select::numa_node_to_heap_map = unsigned short [1028] 00007ffa`7789f2d0 coreclr!SVR::gc_heap::should_expand_in_full_gc = 0n0
其实这就是命名空间,即 coreclr 在编译源码的时候,为 WKS
和 SVR
各编译了一份,不知道这么做的初衷是什么,这里就不管了,接下来看下 coreclr 中大概长啥样子。
namespace WKS { #include "gcimpl.h" #include "gc.cpp" } namespace SVR { #include "gcimpl.h" #include "gc.cpp" }
二:聊一聊 namespace
其实和 C# 的 namespace 本质差不多,都是起到隔离的作用,而且和 using 的配合使用和 C# 也是如出一辙,太有意思了。
1. 简单的隔离
在 C++ 中默认只有一个 namespace,所以相同的变量会出现冲突,解决办法就是用 namespace 隔离,参考如下代码:
namespace WKS { int a = 10; int b = 11; } namespace SRV { int a = 100; int b = 101; } int main() { printf("WKS::a= %d /n", WKS::a); printf("SRV::a= %d /n", SRV::a); }
当然还可以嵌套使用,比如改成这样。
namespace WKS { namespace V1 { int a = 10; int b = 11; } } int main() { printf("WKS::a= %d /n", WKS::V1::a); }
接下来看下汇编代码:
哈哈,看到上面的 WKS::V1::a
感觉是不是挺舒服的,也特能理解目前的 coreclr!WKS::xxx
了, 不过这里有一个麻烦的地方,就是每次用 a
的时候都要输入很长的前缀,那有没有简化的方法呢? 当然有啦。
2. 使用 using 导入
接下来我们用 using 直接在 main 函数中定义字段,后续就不需要再写长长的前缀引用了,参考代码如下:
namespace WKS { namespace V1 { int a = 10; int b = 11; } } int main() { using WKS::V1::a; printf("WKS::V1::a1= %d /n", a); }
3. 使用 using 定义别名
定义别名这功能特别好,个人感觉已经完全替代以前的 typedef
功能,比如下面的代码是完全一样的。
int main() { typedef const char* PCHAR; using PCHAR2 = const char*; PCHAR ptr1 = "hello world1"; PCHAR2 ptr2 = "hello world2"; }
如果还不信的话,可以看下它们各自生成的汇编代码。
PCHAR ptr1 = "hello world1"; 00007FF79856183B lea rax,[string "hello world1" (07FF798569C10h)] 00007FF798561842 mov qword ptr [ptr1],rax PCHAR2 ptr2 = "hello world2"; 00007FF798561846 lea rax,[string "hello world2" (07FF798569CE8h)] 00007FF79856184D mov qword ptr [ptr2],rax
4. 使用 using namespace 导入
这个是最普遍的,我们对系统库的调用,无一不是用 using namespace 方式的,比如下面的代码。
using namespace std; int main() { string str = "hello world"; }
接下来我们把 V1 导入到 main 方法中,这样就可以自由自在的使用 WKS::V1
中的内容了,参考如下代码:
namespace WKS { namespace V1 { int a = 10; int b = 11; } } int main() { using namespace WKS::V1; printf("a=%d, b=%d", a, b); }
好了,这就是对 namespace 的一点理解,本篇就说这么多吧,希望对你有帮助。