當前位置:首頁 > 嵌入式 > 嵌入式軟件
[導讀]Linux下C應用程序開發(fā)

本文介紹了在 Linux 下能用于 C 應用程序開發(fā)和調(diào)試的工具. 本文的主旨是介紹如何在 Linux 下使用 C 編譯器和其他 C 編程工具, 而非 C 語言編程的教程.
 
  GNU C 編譯器 
  GNU C 編譯器(GCC)是一個全功能的 ANSI C 兼容編譯器. 如果你熟悉其他操作系統(tǒng)或硬件平臺上的一種 C 編譯器, 你將能很快地掌握 GCC. 本節(jié)將介紹如何使用 GCC 和一些 GCC 編譯器最常用的選項.
 
  使用 GCC 
  通常后跟一些選項和文件名來使用 GCC 編譯器. gcc 命令的基本用法如下:
 
  gcc [options] [filenames] 
  
  命令行選項指定的操作將在命令行上每個給出的文件上執(zhí)行. 下一小節(jié)將敘述一些你會最常用到的選項.
  
  GCC 選項 
  
  GCC 有超過100個的編譯選項可用. 這些選項中的許多你可能永遠都不會用到, 但一些主要的選項將會頻繁用到. 很多的 GCC 選項包括一個以上的字符. 因此你必須為每個選項指定各自的連字符, 并且就象大多數(shù) Linux 命令一樣你不能在一個單獨的連字符后跟一組選項. 例如, 下面的兩個命令是不同的:
  
  gcc -p -g test.c
  gcc -pg test.c 
  第一條命令告訴 GCC 編譯 test.c 時為 prof 命令建立剖析(profile)信息并且把調(diào)試信息加入到可執(zhí)行的文件里. 第二條命令只告訴 GCC 為 gprof 命令建立剖析信息.
當你不用任何選項編譯一個程序時, GCC 將會建立(假定編譯成功)一個名為 a.out 的可執(zhí)行文件. 例如, 下面的命令將在當前目錄下產(chǎn)生一個叫 a.out 的文件:
 
  gcc test.c 
  你能用 -o 編譯選項來為將產(chǎn)生的可執(zhí)行文件指定一個文件名來代替 a.out. 例如, 將一個叫 count.c 的 C 程序編譯為名叫 count 的可執(zhí)行文件, 你將輸入下面的命令:
 
  gcc -o count count.c
------------------------------------------------------------------------------ 
--
注意: 當你使用 -o 選項時, -o 后面必須跟一個文件名.
------------------------------------------------------------------------------ 
--
  
  GCC 同樣有指定編譯器處理多少的編譯選項. -c 選項告訴 GCC 僅把源代碼編譯為目標代碼而跳過匯編和連接的步驟. 這個選項使用的非常頻繁因為它使得編譯多個 C 程序時速度更快并且更易于管理. 缺省時 GCC 建立的目標代碼文件有一個 .o 的擴展名.
-S 編譯選項告訴 GCC 在為 C 代碼產(chǎn)生了匯編語言文件后停止編譯. GCC 產(chǎn)生的匯編語言文件的缺省擴展名是 .s . -E 選項指示編譯器僅對輸入文件進行預處理. 當這個選項被使用時, 預處理器的輸出被送到標準輸出而不是儲存在文件里.
 
  優(yōu)化選項 
  當你用 GCC 編譯 C 代碼時, 它會試著用最少的時間完成編譯并且使編譯后的代碼易于調(diào)試. 易于調(diào)試意味著編譯后的代碼與源代碼有同樣的執(zhí)行次序, 編譯后的代碼沒有經(jīng)過優(yōu)化. 有很多選項可用于告訴 GCC 在耗費更多編譯時間和犧牲易調(diào)試性的基礎上產(chǎn)生更小 
更快的可執(zhí)行文件. 這些選項中最典型的是-O 和 -O2 選項.
 
  -O 選項告訴 GCC 對源代碼進行基本優(yōu)化. 這些優(yōu)化在大多數(shù)情況下都會使程序執(zhí)行的更快. -O2 選項告訴 GCC 產(chǎn)生盡可能小和盡可能快的代碼. -O2 選項將使編譯的速度比使用 -O 時慢. 但通常產(chǎn)生的代碼執(zhí)行速度會更快.
 
  除了 -O 和 -O2 優(yōu)化選項外, 還有一些低級選項用于產(chǎn)生更快的代碼. 這些選項非常的特殊, 而且最好只有當你完全理解這些選項將會對編譯后的代碼產(chǎn)生什么樣的效果時再去使用. 這些選項的詳細描述, 請參考 GCC 的指南頁, 在命令行上鍵入 man gcc .
 
  調(diào)試和剖析選項 
  GCC 支持數(shù)種調(diào)試和剖析選項. 在這些選項里你會最常用到的是 -g 和 -pg 選項. 
-g 選項告訴 GCC 產(chǎn)生能被 GNU 調(diào)試器使用的調(diào)試信息以便調(diào)試你的程序. GCC 提供了一個很多其他 C 編譯器里沒有的特性, 在 GCC 里你能使 -g 和 -O (產(chǎn)生優(yōu)化代碼)聯(lián)用.. 這一點非常有用因為你能在與最終產(chǎn)品盡可能相近的情況下調(diào)試你的代碼. 在你同時使用這兩個選項時你必須清楚你所寫的某些代碼已經(jīng)在優(yōu)化時被 GCC 作了改動. 關于調(diào)試 
C 程序的更多信息請看下一節(jié)"用 gdb 調(diào)試 C 程序" . 
  -pg 選項告訴 GCC 在你的程序里加入額外的代碼, 執(zhí)行時, 產(chǎn)生 gprof 用的剖析信息以顯示你的程序的耗時情況. 關于 gprof 的更多信息請參考 "gprof" 一節(jié).
 
  用 gdb 調(diào)試 GCC 程序 
  Linux 包含了一個叫 gdb 的 GNU 調(diào)試程序. gdb 是一個用來調(diào)試 C 和 C++ 程序的強力調(diào)試器. 它使你能在程序運行時觀察程序的內(nèi)部結構和內(nèi)存的使用情況. 以下是 gdb 所提供的一些功能:
 
  它使你能監(jiān)視你程序中變量的值. 
  它使你能設置斷點以使程序在指定的代碼行上停止執(zhí)行. 
  它使你能一行行的執(zhí)行你的代碼.
  在命令行上鍵入 gdb 并按回車鍵就可以運行 gdb 了, 如果一切正常的話, gdb 將被啟動  
  并且你將在屏幕上看到類似的內(nèi)容:
  GNU gdb 5.0 
  Copyright 2000 Free Software Foundation, Inc. 
  GDB is free software, covered by the GNU General Public License, and you arewelcome to change   it and/or distribute copies of it under certain conditions. 
  Type "show copying" to see the conditions. 
  There is absolutely no warranty for GDB. Type "show warranty" for details. 
  This GDB was configured as "i386-redhat-linux". 
(gdb) 
  當你啟動 gdb 后, 你能在命令行上指定很多的選項. 你也可以以下面的方式來運行 gdb 
:
  gdb <fname>; 
  當你用這種方式運行 gdb , 你能直接指定想要調(diào)試的程序. 這將告訴gdb 裝入名為 fname 的可執(zhí)行文件. 你也可以用 gdb 去檢查一個因程序異常終止而產(chǎn)生的 core 文件, 
或者與一個正在運行的程序相連. 你可以參考 gdb 指南頁或在命令行上鍵入 gdb -h 得到一個有關這些選項的說明的簡單列表.
  為調(diào)試編譯代碼(Compiling Code for Debugging) 
  為了使 gdb 正常工作, 你必須使你的程序在編譯時包含調(diào)試信息. 調(diào)試信息包含你程序里的每個變量的類型和在可執(zhí)行文件里的地址映射以及源代碼的行號. gdb 利用這些信息使源代碼和機器碼相關聯(lián).
  在編譯時用 -g 選項打開調(diào)試選項.
  gdb 基本命令 
  gdb 支持很多的命令使你能實現(xiàn)不同的功能. 這些命令從簡單的文件裝入到允許你檢查所調(diào)用的堆棧內(nèi)容的復雜命令, 表27.1列出了你在用 gdb 調(diào)試時會用到的一些命令. 想了解 gdb 的詳細使用請參考 gdb 的指南頁.
  基本 gdb 命令.
  命 令 描 述 
  file 裝入想要調(diào)試的可執(zhí)行文件. 
  kill 終止正在調(diào)試的程序. 
  list 列出產(chǎn)生執(zhí)行文件的源代碼的一部分. 
  next 執(zhí)行一行源代碼但不進入函數(shù)內(nèi)部. 
  step 執(zhí)行一行源代碼而且進入函數(shù)內(nèi)部. 
  run 執(zhí)行當前被調(diào)試的程序 
  quit 終止 gdb 
  watch 使你能監(jiān)視一個變量的值而不管它何時被改變. 
  print 顯示表達式的值 
  break 在代碼里設置斷點, 這將使程序執(zhí)行到這里時被掛起. 
  make 使你能不退出 gdb 就可以重新產(chǎn)生可執(zhí)行文件. 
  shell 使你能不離開 gdb 就執(zhí)行 UNIX shell 命令.
  gdb 支持很多與 UNIX shell 程序一樣的命令編輯特征. 你能象在 bash 或 tcsh里那樣按 Tab 鍵讓 gdb 幫你補齊一個唯一的命令, 如果不唯一的話 gdb 會列出所有匹配的命令. 你也能用光標鍵上下翻動歷史命令.[!--empirenews.page--]gdb 應用舉例 
  本節(jié)用一個實例教你一步步的用 gdb 調(diào)試程序. 被調(diào)試的程序相當?shù)暮唵? 但它展示了 gdb 的典型應用.
  下面列出了將被調(diào)試的程序. 這個程序被稱為 hello , 它顯示一個簡單的問候, 再用反序?qū)⑺谐?
  #include <stdio.h>;
  static void my_print (char *); 
  static void my_print2 (char *);
  main () 
  { 
  char my_string[] = "hello world!"; 
  my_print (my_string); 
  my_print2 (my_string); 
  }
  void my_print (char *string) 
  { 
  printf ("The string is %s ", string); 
  }
  void my_print2 (char *string) 
  { 
  char *string2; 
  int size, i;
  size = strlen (string); 
  string2 = (char *) malloc (size + 1); 
  for (i = 0; i < size; i++) 
  string2[size - i] = string; 
  string2[size+1] = '';
  printf ("The string printed backward is %s ", string2); 
  } 
  用下面的命令編譯它:
  gcc -g -o hello hello.c 
  這個程序執(zhí)行時顯示如下結果: 
  ../hello 
  The string is hello world!
  The string printed backward is 
  輸出的第一行是正確的, 但第二行打印出的東西并不是我們所期望的. 我們所設想的輸出 
  應該是:
  The string printed backward is !dlrow olleh 
  由于某些原因, my_print2 函數(shù)沒有正常工作. 讓我們用 gdb 看看問題究竟出在哪兒, 
  先鍵入如下命令:
  gdb hello
------------------------------------------------------------------------------ 
--
注意: 記得在編譯 hello 程序時把調(diào)試選項打開.
------------------------------------------------------------------------------ 
--
  如果你在輸入命令時忘了把要調(diào)試的程序作為參數(shù)傳給 gdb , 你可以在 gdb 提示符下用 
  file 命令來載入它:
  (gdb) file hello 
  這個命令將載入 hello 可執(zhí)行文件就象你在 gdb 命令行里裝入它一樣.
  這時你能用 gdb 的 run 命令來運行 hello 了. 當它在 gdb 里被運行后結果大約會象這樣:
  (gdb) run
  Starting program: /root/hello
  The string is hello world!
  The string printed backward is
  Program exited with code 040 
  這個輸出和在 gdb 外面運行的結果一樣. 問題是, 為什么反序打印沒有工作? 為了找出癥結所在, 我們可以在 my_print2 函數(shù)的 for 語句后設一個斷點, 具體的做法是在 gdb 
提示符下鍵入 list 命令三次, 列出源代碼:
  (gdb) list
  (gdb) list
  (gdb) list
------------------------------------------------------------------------------ 
--
技巧: 在 gdb 提示符下按回車健將重復上一個命令.
------------------------------------------------------------------------------ 
--
  第一次鍵入 list 命令的輸出如下:
  1 #include <stdio.h>; 
  2 
  3 static void my_print (char *); 
  4 static void my_print2 (char *); 
  5 
  6 main () 
  7 { 
  8 char my_string[] = "hello world!"; 
  9 my_print (my_string); 
  10 my_print2 (my_string);
如果按下回車, gdb 將再執(zhí)行一次 list 命令, 給出下列輸出:
  11 } 
  12 
  13 void my_print (char *string) 
  14 { 
  15 printf ("The string is %s ", string); 
  16 } 
  17 
  18 void my_print2 (char *string) 
  19 { 
  20 char *string2; 
  再按一次回車將列出 hello 程序的剩余部分:
  21 int size, i; 
  22 
  23 size = strlen (string); 
  24 string2 = (char *) malloc (size + 1); 
  25 for (i = 0; i < size; i++) 
  26 string2[size - i] = string; 
  27 string2[size+1] = ''; 
  28 
  29 printf ("The string printed backward is %s ", string2); 
  30 } 
  根據(jù)列出的源程序, 你能看到要設斷點的地方在第26行, 在 gdb 命令行提示符下鍵入如下命令設置斷點:
  (gdb) break 26 
  gdb 將作出如下的響應:
  Breakpoint 1 at 0x804857c: file hello.c, line 26.
  (gdb)
  現(xiàn)在再鍵入 run 命令, 將產(chǎn)生如下的輸出:
  Starting program: /root/hello
  The string is hello world!
  Breakpoint 1, my_print2 (string=0xbffffab0 "hello world!") at hello.c:26 
  26 string2[size - i] = string; 
  你能通過設置一個觀察 string2[size - i] 變量的值的觀察點來看出錯誤是怎樣產(chǎn)生的, 
  做法是鍵入:
  (gdb) watch string2[size - i] 
  gdb 將作出如下回應:
  Hardware watchpoint 2: string2[size - i] 
  現(xiàn)在可以用 next 命令來一步步的執(zhí)行 for 循環(huán)了:
  (gdb) next 
  經(jīng)過第一次循環(huán)后, gdb 告訴我們 string2[size - i] 的值是 `h`. gdb 用如下的顯示來告訴你這個信息:
  Hardware watchpoint 2: string2[size - i]
  Old value = 0 '00' 
  New value = 104 'h' 
  my_print2 (string=0xbffffab0 "hello world!") at hello.c:25 
  25 for (i = 0; i < size; i++) 
  這個值正是期望的. 后來的數(shù)次循環(huán)的結果都是正確的. 當 i=11 時, 表達式 
  string2[size - i] 的值等于 `!`, size - i 的值等于 1, 最后一個字符已經(jīng)拷到新串里了.
  如果你再把循環(huán)執(zhí)行下去, 你會看到已經(jīng)沒有值分配給 string2[0] 了, 而它是新串的第一個字符, 因為 malloc 函數(shù)在分配內(nèi)存時把它們初始化為空(null)字符. 所以 string2 的第一個字符是空字符. 這解釋了為什么在打印 string2 時沒有任何輸出了.
 
  現(xiàn)在找出了問題出在哪里, 修正這個錯誤是很容易的. 你得把代碼里寫入 string2 的第一個字符的的偏移量改為 size - 1 而不是 size. 這是因為 string2 的大小為 12, 但起始偏移量是 0, 串內(nèi)的字符從偏移量 0 到 偏移量 10, 偏移量 11 為空字符保留.
  改正方法非常簡單. 這是這種解決辦法的代碼:
  #include <stdio.h>;
  static void my_print (char *); 
  static void my_print2 (char *);
  main () 
  { 
  char my_string[] = "hello world!"; 
  my_print (my_string); 
  my_print2 (my_string); 
  }
  void my_print (char *string) 
  { 
  printf ("The string is %s ", string); 
  }
  void my_print2 (char *string) 
  { 
  char *string2; 
  int size, i;
  size = strlen (string); 
  string2 = (char *) malloc (size + 1); 
  for (i = 0; i < size; i++) 
  string2[size -1 - i] = string; 
  string2[size] = '';
  printf ("The string printed backward is %s ", string2); 
  } 
  如果程序產(chǎn)生了core文件,可以用gdb hello core命令來查看程序在何處出錯。如在函數(shù)my_print2()中,如果忘記了給string2分配內(nèi)存 string2 = (char *) malloc (size + 1);,很可能就會  core dump.[!--empirenews.page--]另外的 C 編程工具
  xxgdb 
  xxgdb 是 gdb 的一個基于 X Window 系統(tǒng)的圖形界面. xxgdb 包括了命令行版的 gdb 上的所有特性. xxgdb 使你能通過按按鈕來執(zhí)行常用的命令. 設置了斷點的地方也用圖形來顯示.
你能在一個 Xterm 窗口里鍵入下面的命令來運行它:
  xxgdb 
  你能用 gdb 里任何有效的命令行選項來初始化 xxgdb . 此外 xxgdb 也有一些特有的命令行選項, 表 27.2 列出了這些選項.
  表 27.2. xxgdb 命令行選項.
  選 項 描 述 
  db_name 指定所用調(diào)試器的名字, 缺省是 gdb. 
  db_prompt 指定調(diào)試器提示符, 缺省為 gdb. 
  gdbinit 指定初始化 gdb 的命令文件的文件名, 缺省為 .gdbinit.
  nx 告訴 xxgdb 不執(zhí)行 .gdbinit 文件. 
  bigicon 使用大圖標.
  calls 
  你可以在 sunsite.unc.edu FTP 站點用下面的路徑: 
  /pub/Linux/devel/lang/c/calls.tar.Z 
  來取得 calls , 一些舊版本的 Linux CD-ROM 發(fā)行版里也附帶有. 因為它是一個有用的工具, 我們在這里也介紹一下. 如果你覺得有用的話, 從 BBS, FTP, 或另一張CD-ROM 上弄一個拷貝. calls 調(diào)用 GCC 的預處理器來處理給出的源程序文件, 然后輸出這些文件的里的函數(shù)調(diào)用樹圖.
  注意: 在你的系統(tǒng)上安裝 calls , 以超級用戶身份登錄后執(zhí)行下面的步驟: 1. 解壓和 untar   文件. 2. cd 進入 calls untar 后建立的子目錄. 3. 把名叫 calls 的文件移動到 /usr/bin 目錄. 4. 把名叫 calls.1 的文件移動到目錄 /usr/man/man1 . 5. 刪除 /tmp/calls 目錄. 這些步驟將把 calls 程序和它的指南頁安裝載你的系統(tǒng)上.
------------------------------------------------------------------------------ 
--
  當 calls 打印出調(diào)用跟蹤結果時, 它在函數(shù)后面用中括號給出了函數(shù)所在文件的文件名:
main [hello.c] 
  如果函數(shù)并不是向 calls 給出的文件里的, calls 不知道所調(diào)用的函數(shù)來自哪里, 則只顯示函數(shù)的名字:
  printf 
  calls 不對遞歸和靜態(tài)函數(shù)輸出. 遞歸函數(shù)顯示成下面的樣子:
  fact <<< recursive in factorial.c >;>;>; 
  靜態(tài)函數(shù)象這樣顯示:
  total [static in calculate.c] 
  作為一個例子, 假設用 calls 處理下面的程序:
  #include <stdio.h>;
  static void my_print (char *); 
  static void my_print2 (char *);
  main () 
  { 
  char my_string[] = "hello world!"; 
  my_print (my_string); 
  my_print2 (my_string); 
  my_print (my_string); 
  }
  void count_sum() 
  { 
  int i,sum=0; 
  for(i=0; i<1000000; i++) 
  sum += i; 
  }
  void my_print (char *string) 
  { 
  count_sum(); 
  ("The string is %s ", string); 
  }
  void my_print2 (char *string) 
  { 
  char *string2; 
  int size, i,sum =0;
  printf ("The string printed backward is %s ", string2); 
  } 
  將產(chǎn)生如下的輸出:
  1 __underflow [hello.c] 
  2 main 
  3 my_print [hello.c] 
  4 count_sum [hello.c] 
  5 printf 
  6 my_print2 [hello.c] 
  7 count_sum 
  8 strlen 
  9 malloc 
  10 printf 
  calls 有很多命令行選項來設置不同的輸出格式, 有關這些選項的更多信息請參考 calls 的指南頁. 方法是在命令行上鍵入 calls -h .
  calltree 
  calltree與calls類似,初了輸出函數(shù)調(diào)用樹圖外,還有其它詳細的信息??梢詮膕unsite.unc.edu FTP 站點用下面的路徑 
  :/pub/Linux/devel/lang/c/calltree.tar.gz得到calltree.
  cproto 
  cproto 讀入 C 源程序文件并自動為每個函數(shù)產(chǎn)生原型申明. 用 cproto 可以在寫程序時為你節(jié)省大量用來定義函數(shù)原型的時間. 
  如果你讓 cproto 處理下面的代碼(cproto hello.c):
  #include <stdio.h>;
  static void my_print (char *); 
  static void my_print2 (char *);
  main () 
  { 
  char my_string[] = "hello world!"; 
  my_print (my_string); 
  my_print2 (my_string); 
  }
  void my_print (char *string) 
  { 
  printf ("The string is %s ", string); 
  }
  void my_print2 (char *string) 
  { 
  char *string2; 
  int size, i;
  size = strlen (string); 
  string2 = (char *) malloc (size + 1); 
  for (i = 0; i < size; i++) 
  string2[size -1 - i] = string; 
  string2[size] = '';
  printf ("The string printed backward is %s ", string2); 
  } 
  你將得到下面的輸出:
  /* hello.c */
  int main(void);
  int my_print(char *string);
  int my_print2(char *string); 
  這個輸出可以重定向到一個定義函數(shù)原型的包含文件里.
  indent 
  indent 實用程序是 Linux 里包含的另一個編程實用工具. 這個工具簡單的說就為你的代碼產(chǎn)生美觀的縮進的格式. indent 也有很多選項來指定如何格式化你的源代碼.這些選項的更多信息請看indent 的指南頁, 在命令行上鍵入 indent -h .
  下面的例子是 indent 的缺省輸出:
  運行 indent 以前的 C 代碼:
  #include <stdio.h>;
  static void my_print (char *); 
  static void my_print2 (char *);
  main () 
  { 
  char my_string[] = "hello world!"; 
  my_print (my_string); 
  my_print2 (my_string); 
  }
  void my_print (char *string) 
  { 
  printf ("The string is %s ", string); 
  }
  void my_print2 (char *string) 
  { 
  char *string2; int size, i;
  size = strlen (string); 
  string2 = (char *) malloc (size + 1); 
  for (i = 0; i < size; i++) string2[size -1 - i] = string; 
  string2[size] = '';
  printf ("The string printed backward is %s ", string2); 
  } 
  運行 indent 后的 C 代碼:
  #include <stdio.h>; 
  static void my_print (char *); 
  static void my_print2 (char *); 
  main () 
  { 
  char my_string[] = "hello world!"; 
  my_print (my_string); 
  my_print2 (my_string); 
  } 
  void 
  my_print (char *string) 
  { 
  printf ("The string is %s ", string); 
  } 
  void 
  my_print2 (char *string) 
  { 
  char *string2; 
  int size, i; 
  size = strlen (string); 
  string2 = (char *) malloc (size + 1); 
  for (i = 0; i < size; i++) 
  string2[size - 1 - i] = string; 
  string2[size] = ''; 
  printf ("The string printed backward is %s ", string2); 
  } 
  indent 并不改變代碼的實質(zhì)內(nèi)容, 而只是改變代碼的外觀. 使它變得更可讀, 這永遠是一件好事.
  gprof 
  gprof 是安裝在你的 Linux 系統(tǒng)的 /usr/bin 目錄下的一個程序. 它使你能剖析你的程序從而知道程序的哪一個部分在執(zhí)行時最費時間.
  gprof 將告訴你程序里每個函數(shù)被調(diào)用的次數(shù)和每個函數(shù)執(zhí)行時所占時間的百分比. 你如果想提高你的程序性能的話這些信息非常有用.
  為了在你的程序上使用 gprof, 你必須在編譯程序時加上 -pg 選項. 這將使程序在每次執(zhí)行時產(chǎn)生一個叫 gmon.out 的文件. gprof 用這個文件產(chǎn)生剖析信息.
  在你運行了你的程序并產(chǎn)生了 gmon.out 文件后你能用下面的命令獲得剖析信息:
  gprof <program_name>; 
  參數(shù) program_name 是產(chǎn)生 gmon.out 文件的程序的名字.
  為了說明問題,在程序中增加了函數(shù)count_sum()以消耗CPU時間,程序如下 
  #include <stdio.h>;
  static void my_print (char *); 
  static void my_print2 (char *);
  main () 
  { 
  char my_string[] = "hello world!"; 
  my_print (my_string); 
  my_print2 (my_string); 
  my_print (my_string); 
  }
  void count_sum() 
  { 
  int i,sum=0; 
  for(i=0; i<1000000; i++) 
  sum += i; 
  }
  void my_print (char *string) 
  { 
  count_sum(); 
  printf ("The string is %s ", string); 
  }
  void my_print2 (char *string) 
  { 
  char *string2; 
  int size, i,sum =0;
  count_sum(); 
  size = strlen (string); 
  string2 = (char *) malloc (size + 1); 
  for (i = 0; i < size; i++) string2[size -1 - i] = string; 
  string2[size] = ''; 
  for(i=0; i<5000000; i++) 
  sum += i;
  printf ("The string printed backward is %s ", string2); 
  } 
  $ gcc -pg -o hello hello.c 
  $ ./hello 
  $ gprof hello | more 
  將產(chǎn)生以下的輸出 
  Flat profile:
  Each sample counts as 0.01 seconds. 
  % cumulative self self total 
  time seconds seconds calls us/call us/call name 
  69.23 0.09 0.09 1 90000.00 103333.33 my_print2 
  30.77 0.13 0.04 3 13333.33 13333.33 count_sum 
  0.00 0.13 0.00 2 0.00 13333.33 my_print
  % 執(zhí)行此函數(shù)所占用的時間占程序總 
  time 執(zhí)行時間的百分比
  cumulative 累計秒數(shù) 執(zhí)行此函數(shù)花費的時間 
  seconds (包括此函數(shù)調(diào)用其它函數(shù)花費的時間)
  self 執(zhí)行此函數(shù)花費的時間 
  seconds (調(diào)用其它函數(shù)花費的時間不計算在內(nèi))
  calls 調(diào)用次數(shù)
  self 每此執(zhí)行此函數(shù)花費的微秒時間 
  us/call
  total 每此執(zhí)行此函數(shù)加上它調(diào)用其它函數(shù) 
  us/call 花費的微秒時間
  name 函數(shù)名
  由以上數(shù)據(jù)可以看出,執(zhí)行my_print()函數(shù)本身沒花費什么時間,但是它又調(diào)用了 
count_sum()函數(shù),所以累計秒數(shù)為0.13.
  技巧: gprof 產(chǎn)生的剖析數(shù)據(jù)很大, 如果你想檢查這些數(shù)據(jù)的話最好把輸出重定向到一個文件里.
本站聲明: 本文章由作者或相關機構授權發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務中斷的風險,如企業(yè)系統(tǒng)復雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務連續(xù)性,提升韌性,成...

關鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關鍵字: 華為 12nm EDA 半導體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權最終是由生態(tài)的繁榮決定的。

關鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務引領增長 以科技創(chuàng)新為引領,提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強核心競爭優(yōu)勢...

關鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術學會聯(lián)合牽頭組建的NVI技術創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術創(chuàng)新聯(lián)...

關鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(集團)股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關鍵字: BSP 信息技術
關閉
關閉