nm命令存在於多數後出版本的Unix類似作業系統中。nm被用來檢查二進制檔案(包括,編譯後的目標模組,共用目標文件,和獨立可執行檔案)並顯示這些檔案的內容,或儲存在其中的元資訊,特別是符號表。來自nm的輸出區分各種符號類型。例如,在一個目標文件所提供的函數和它所需要的函數之間是有區別的。nm被用作除錯的輔助工具,幫助解決名字衝突和C++名稱修飾引起的問題,和驗證工具鏈的其他部分。

nm
開發者AT&T貝爾實驗室
首次發佈1971年11月3日,​52年前​(1971-11-03
作業系統Unix類Unix
類型命令

GNU計劃GNU Binutils包中提供了一個nm的實現。objdump -treadelf -s也可以檢視目標文件的符號表。

nm輸出樣例

下面的例子展示了變數和函數在全域局部外部靜態自動初始化的不同情況下的不同符號類型。

/*
 * 文件名: test.c
 * 对于C代码如下编译: 
 * gcc -c test.c
 *
 * 对于C++代码如下编译:
 * g++ -c test.cpp
 */

int global_var;
int global_var_init = 26;

static int static_var;
static int static_var_init = 25;

extern int extern_var;
extern int extern_function(int);

static int static_function(int x, int y)
{
	int local_automatic_var;

	local_automatic_var = x + y;
	return local_automatic_var;
}

int global_function(int p)
{
	static int local_static_var;
	static int local_static_var_init=5;
	
	local_static_var = static_function(local_static_var_init, p);	
	return local_static_var;
}

int main(int argc, char** argv)
{
	static_var = 1;

	global_var = global_function(2);
	extern_var = extern_function(3);
	return 0;
}

#ifdef __cplusplus
extern "C"
#endif
void non_mangled_function(void)
{
	// I do nothing
}

如果上述代碼使用了gcc C編譯器來編譯,nm命令的輸出將是如下這樣:

$ nm test.o
                 U extern_function
                 U extern_var
000000000000001a T global_function
                 U _GLOBAL_OFFSET_TABLE_
0000000000000004 C global_var
0000000000000000 D global_var_init
0000000000000004 b local_static_var.1970
0000000000000008 d local_static_var_init.1971
000000000000004c T main
0000000000000045 T non_mangled_function
0000000000000000 t static_function
0000000000000000 b static_var
0000000000000004 d static_var_init

預設按符號字母序排序,其中,大寫字母表示全域性,小寫字母表示局部性;U表示此符號未定義,C表示此符號是公共的,T/t表示此符號在正文(代碼)段中,D/d的表示此符號在初始化數據段中,B/b表示此符號在BSS數據段中。

在使用了C++編譯器時,輸出是不同的:

$ nm test.o
                 U extern_var
                 U _GLOBAL_OFFSET_TABLE_
0000000000000000 B global_var
0000000000000000 D global_var_init
000000000000004c T main
0000000000000045 T non_mangled_function
                 U _Z15extern_functioni
000000000000001a T _Z15global_functioni
0000000000000004 b _ZL10static_var
0000000000000000 t _ZL15static_functionii
0000000000000004 d _ZL15static_var_init
0000000000000008 b _ZZ15global_functioniE16local_static_var
0000000000000008 d _ZZ15global_functioniE21local_static_var_init

在二者輸出之間的不同還展示了通過在C++代碼中使用extern "C"解決名稱修飾問題的一個例子。

參見

外部連結