## 21.3 用 make 進行宏編譯
在本章一開始我們提到過 make 的功能是可以簡化編譯過程里面所下達的指令,同時還具有很多很方便的功能!那么下面咱們就來試看看使用 make 簡化下達編譯指令的流程吧!
### 21.3.1 為什么要用 make
先來想像一個案例,假設我的可執行文件里面包含了四個源代碼文件,分別是 main.c haha.c sin_value.c cos_value.c 這四個文件,這四個文件的目的是:
* main.c :主要的目的是讓使用者輸入角度數據與調用其他三支副程序;
* haha.c :輸出一堆有的沒有的訊息而已;
* sin_value.c :計算使用者輸入的角度(360) sin 數值;
* cos_value.c :計算使用者輸入的角度(360) cos 數值。
這四個文件你可以到 [http://linux.vbird.org/linux_basic/0520source/main.tgz](http://linux.vbird.org/linux_basic/0520source/main.tgz) 來下載。由于這四個文件里面包含了相關性,并且還用到數學函數在里面,所以如果你想要讓這個程序可以跑, 那么就需要這樣編譯:
```
# 1\. 先進行目標文件的編譯,最終會有四個 *.o 的文件名出現:
[root@study ~]# gcc -c main.c
[root@study ~]# gcc -c haha.c
[root@study ~]# gcc -c sin_value.c
[root@study ~]# gcc -c cos_value.c
# 2\. 再進行鏈接成為可執行文件,并加入 libm 的數學函數,以產生 main 可執行文件:
[root@study ~]# gcc -o main main.o haha.o sin_value.o cos_value.o -lm
# 3\. 本程序的執行結果,必須輸入姓名、360 度角的角度值來計算:
[root@study ~]# ./main
Please input your name: VBird <==這里先輸入名字
Please enter the degree angle (ex> 90): 30 <==輸入以 360 度角為主的角度
Hi, Dear VBird, nice to meet you. <==這三行為輸出的結果喔!
The Sin is: 0.50
The Cos is: 0.87
```
編譯的過程需要進行好多動作啊!而且如果要重新編譯,則上述的流程得要重新來一遍,光是找出這些指令就夠煩人的了! 如果可以的話,能不能一個步驟就給他完成上面所有的動作呢?那就利用 make 這個工具吧! 先試看看在這個目錄下創建一個名為 makefile 的文件,內容如下:
```
# 1\. 先編輯 makefile 這個規則檔,內容只要作出 main 這個可執行文件
[root@study ~]# vim makefile
main: main.o haha.o sin_value.o cos_value.o
gcc -o main main.o haha.o sin_value.o cos_value.o -lm
# 注意:第二行的 gcc 之前是 <tab> 按鍵產生的空格喔!
# 2\. 嘗試使用 makefile 制訂的規則進行編譯的行為:
[root@study ~]# rm -f main *.o <==先將之前的目標文件去除
[root@study ~]# make
cc -c -o main.o main.c
cc -c -o haha.o haha.c
cc -c -o sin_value.o sin_value.c
cc -c -o cos_value.o cos_value.c
gcc -o main main.o haha.o sin_value.o cos_value.o -lm
# 此時 make 會去讀取 makefile 的內容,并根據內容直接去給他編譯相關的文件啰!
# 3\. 在不刪除任何文件的情況下,重新執行一次編譯的動作:
[root@study ~]# make
make: `main' is up to date.
# 看到了吧!是否很方便呢!只會進行更新 (update) 的動作而已。
```
或許你會說:“如果我創建一個 shell script 來將上面的所有動作都集結在一起,不是具有同樣的效果嗎?”呵呵! 效果當然不一樣,以上面的測試為例,我們僅寫出 main 需要的目標文件,結果 make 會主動的去判斷每個目標文件相關的源代碼文件,并直接予以編譯,最后再直接進行鏈接的動作! 真的是很方便啊!此外,如果我們更動過某些源代碼文件,則 make 也可以主動的判斷哪一個源代碼與相關的目標文件文件有更新過, 并僅更新該文件,如此一來,將可大大的節省很多編譯的時間呢!要知道,某些程序在進行編譯的行為時,會消耗很多的 CPU 資源呢!所以說, make 有這些好處:
* 簡化編譯時所需要下達的指令;
* 若在編譯完成之后,修改了某個源代碼文件,則 make 僅會針對被修改了的文件進行編譯,其他的 object file 不會被更動;
* 最后可以依照相依性來更新 (update) 可執行文件。
既然 make 有這么多的優點,那么我們當然就得好好的了解一下 make 這個令人關心的家伙啦!而 make 里面最需要注意的大概就是那個規則文件,也就是 makefile 這個文件的語法啦!所以下面我們就針對 makefile 的語法來加以介紹啰。
### 21.3.2 makefile 的基本語法與變量
make 的語法可是相當的多而復雜的,有興趣的話可以到 GNU [[1]](#ps1) 去查閱相關的說明,鳥哥這里僅列出一些基本的規則,重點在于讓讀者們未來在接觸源代碼時,不會太緊張啊! 好了,基本的 makefile 規則是這樣的:
```
標的(target): 目標文件1 目標文件2
<tab> gcc -o 欲創建的可執行文件 目標文件1 目標文件2
```
那個標的 (target) 就是我們想要創建的信息,而目標文件就是具有相關性的 object files ,那創建可執行文件的語法就是以 <tab> 按鍵開頭的那一行!特別給他留意喔,“命令列必須要以 tab 按鍵作為開頭”才行!他的規則基本上是這樣的:
* 在 makefile 當中的 # 代表注解;
* <tab> 需要在命令行 (例如 gcc 這個編譯器指令) 的第一個字符;
* 標的 (target) 與相依文件(就是目標文件)之間需以“:”隔開。
同樣的,我們以剛剛上一個小節的范例進一步說明,如果我想要有兩個以上的執行動作時, 例如下達一個指令就直接清除掉所有的目標文件與可執行文件,該如何制作呢?
```
# 1\. 先編輯 makefile 來創建新的規則,此規則的標的名稱為 clean :
[root@study ~]# vi makefile
main: main.o haha.o sin_value.o cos_value.o
gcc -o main main.o haha.o sin_value.o cos_value.o -lm
clean:
rm -f main main.o haha.o sin_value.o cos_value.o
# 2\. 以新的標的 (clean) 測試看看執行 make 的結果:
[root@study ~]# make clean <==就是這里!通過 make 以 clean 為標的
rm -rf main main.o haha.o sin_value.o cos_value.o
```
如此一來,我們的 makefile 里面就具有至少兩個標的,分別是 main 與 clean ,如果我們想要創建 main 的話,輸入“make main”,如果想要清除有的沒的,輸入“make clean”即可啊!而如果想要先清除目標文件再編譯 main 這個程序的話,就可以這樣輸入:“make clean main”,如下所示:
```
[root@study ~]# make clean main
rm -rf main main.o haha.o sin_value.o cos_value.o
cc -c -o main.o main.c
cc -c -o haha.o haha.c
cc -c -o sin_value.o sin_value.c
cc -c -o cos_value.o cos_value.c
gcc -o main main.o haha.o sin_value.o cos_value.o -lm
```
這樣就很清楚了吧!但是,你是否會覺得,咦! makefile 里面怎么重復的數據這么多啊!沒錯!所以我們可以再借由 shell script 那時學到的“變量”來更簡化 makefile 喔:
```
[root@study ~]# vi makefile
LIBS = -lm
OBJS = main.o haha.o sin_value.o cos_value.o
main: ${OBJS}
gcc -o main ${OBJS} ${LIBS}
clean:
rm -f main ${OBJS}
```
與 [bash shell script](../Text/index.html) 的語法有點不太相同,變量的基本語法為:
1. 變量與變量內容以“=”隔開,同時兩邊可以具有空格;
2. 變量左邊不可以有 <tab> ,例如上面范例的第一行 LIBS 左邊不可以是 <tab>;
3. 變量與變量內容在“=”兩邊不能具有“:”;
4. 在習慣上,變量最好是以“大寫字母”為主;
5. 運用變量時,以 ${變量} 或 $(變量) 使用;
6. 在該 shell 的環境變量是可以被套用的,例如提到的 CFLAGS 這個變量!
7. 在命令行界面也可以給予變量。
由于 gcc 在進行編譯的行為時,會主動的去讀取 CFLAGS 這個環境變量,所以,你可以直接在 shell 定義出這個環境變量,也可以在 makefile 文件里面去定義,更可以在命令行當中給予這個咚咚呢!例如:
```
[root@study ~]# CFLAGS="-Wall" make clean main
# 這個動作在上 make 進行編譯時,會去取用 CFLAGS 的變量內容!
```
也可以這樣:
```
[root@study ~]# vi makefile
LIBS = -lm
OBJS = main.o haha.o sin_value.o cos_value.o
CFLAGS = -Wall
main: ${OBJS}
gcc -o main ${OBJS} ${LIBS}
clean:
rm -f main ${OBJS}
```
咦!我可以利用命令行進行環境變量的輸入,也可以在文件內直接指定環境變量,那萬一這個 CFLAGS 的內容在命令行與 makefile 里面并不相同時,以那個方式輸入的為主?呵呵!問了個好問題啊! 環境變量取用的規則是這樣的:
1. make 命令行后面加上的環境變量為優先;
2. makefile 里面指定的環境變量第二;
3. shell 原本具有的環境變量第三。
此外,還有一些特殊的變量需要了解的喔:
* $@:代表目前的標的(target)
所以我也可以將 makefile 改成:
```
[root@study ~]# vi makefile
LIBS = -lm
OBJS = main.o haha.o sin_value.o cos_value.o
CFLAGS = -Wall
main: ${OBJS}
gcc -o $@ ${OBJS} ${LIBS} <==那個 $@ 就是 main !
clean:
rm -f main ${OBJS}
```
這樣是否稍微了解了 makefile (也可能是 Makefile) 的基本語法?這對于你未來自行修改源代碼的編譯規則時,是很有幫助的喔!^_^!
- 鳥哥的Linux私房菜:基礎學習篇 第四版
- 目錄及概述
- 第零章、計算機概論
- 0.1 電腦:輔助人腦的好工具
- 0.2 個人電腦架構與相關設備元件
- 0.3 數據表示方式
- 0.4 軟件程序運行
- 0.5 重點回顧
- 0.6 本章習題
- 0.7 參考資料與延伸閱讀
- 第一章、Linux是什么與如何學習
- 1.1 Linux是什么
- 1.2 Torvalds的Linux發展
- 1.3 Linux當前應用的角色
- 1.4 Linux 該如何學習
- 1.5 重點回顧
- 1.6 本章習題
- 1.7 參考資料與延伸閱讀
- 第二章、主機規劃與磁盤分區
- 2.1 Linux與硬件的搭配
- 2.2 磁盤分區
- 2.3 安裝Linux前的規劃
- 2.4 重點回顧
- 2.5 本章習題
- 2.6 參考資料與延伸閱讀
- 第三章、安裝 CentOS7.x
- 3.1 本練習機的規劃--尤其是分區參數
- 3.2 開始安裝CentOS 7
- 3.3 多重開機安裝流程與管理(Option)
- 3.4 重點回顧
- 3.5 本章習題
- 3.6 參考資料與延伸閱讀
- 第四章、首次登陸與線上求助
- 4.1 首次登陸系統
- 4.2 文字模式下指令的下達
- 4.3 Linux系統的線上求助man page與info page
- 4.4 超簡單文書編輯器: nano
- 4.5 正確的關機方法
- 4.6 重點回顧
- 4.7 本章習題
- 4.8 參考資料與延伸閱讀
- 第五章、Linux 的文件權限與目錄配置
- 5.1 使用者與群組
- 5.2 Linux 文件權限概念
- 5.3 Linux目錄配置
- 5.4 重點回顧
- 5.5 本章練習
- 5.6 參考資料與延伸閱讀
- 第六章、Linux 文件與目錄管理
- 6.1 目錄與路徑
- 6.2 文件與目錄管理
- 6.3 文件內容查閱
- 6.4 文件與目錄的默認權限與隱藏權限
- 6.5 指令與文件的搜尋
- 6.6 極重要的復習!權限與指令間的關系
- 6.7 重點回顧
- 6.8 本章習題:
- 6.9 參考資料與延伸閱讀
- 第七章、Linux 磁盤與文件系統管理
- 7.1 認識 Linux 文件系統
- 7.2 文件系統的簡單操作
- 7.3 磁盤的分區、格式化、檢驗與掛載
- 7.4 設置開機掛載
- 7.5 內存交換空間(swap)之創建
- 7.6 文件系統的特殊觀察與操作
- 7.7 重點回顧
- 7.8 本章習題 - 第一題一定要做
- 7.9 參考資料與延伸閱讀
- 第八章、文件與文件系統的壓縮,打包與備份
- 8.1 壓縮文件的用途與技術
- 8.2 Linux 系統常見的壓縮指令
- 8.3 打包指令: tar
- 8.4 XFS 文件系統的備份與還原
- 8.5 光盤寫入工具
- 8.6 其他常見的壓縮與備份工具
- 8.7 重點回顧
- 8.8 本章習題
- 8.9 參考資料與延伸閱讀
- 第九章、vim 程序編輯器
- 9.1 vi 與 vim
- 9.2 vi 的使用
- 9.3 vim 的額外功能
- 9.4 其他 vim 使用注意事項
- 9.5 重點回顧
- 9.6 本章練習
- 9.7 參考資料與延伸閱讀
- 第十章、認識與學習BASH
- 10.1 認識 BASH 這個 Shell
- 10.2 Shell 的變量功能
- 10.3 命令別名與歷史命令
- 10.4 Bash Shell 的操作環境:
- 10.5 數據流重導向
- 10.6 管線命令 (pipe)
- 10.7 重點回顧
- 10.8 本章習題
- 10.9 參考資料與延伸閱讀
- 第十一章、正則表達式與文件格式化處理
- 11.1 開始之前:什么是正則表達式
- 11.2 基礎正則表達式
- 11.3 延伸正則表達式
- 11.4 文件的格式化與相關處理
- 11.5 重點回顧
- 11.6 本章習題
- 11.7 參考資料與延伸閱讀
- 第十二章、學習 Shell Scripts
- 12.1 什么是 Shell scripts
- 12.2 簡單的 shell script 練習
- 12.3 善用判斷式
- 12.4 條件判斷式
- 12.5 循環 (loop)
- 12.6 shell script 的追蹤與 debug
- 12.7 重點回顧
- 12.8 本章習題
- 第十三章、Linux 帳號管理與 ACL 權限設置
- 13.1 Linux 的帳號與群組
- 13.2 帳號管理
- 13.3 主機的細部權限規劃:ACL 的使用
- 13.4 使用者身份切換
- 13.5 使用者的特殊 shell 與 PAM 模塊
- 13.6 Linux 主機上的使用者訊息傳遞
- 13.7 CentOS 7 環境下大量創建帳號的方法
- 13.8 重點回顧
- 13.9 本章習題
- 13.10 參考資料與延伸閱讀
- 第十四章、磁盤配額(Quota)與進階文件系統管理
- 14.1 磁盤配額 (Quota) 的應用與實作
- 14.2 軟件磁盤陣列 (Software RAID)
- 14.3 邏輯卷軸管理員 (Logical Volume Manager)
- 14.4 重點回顧
- 14.5 本章習題
- 14.6 參考資料與延伸閱讀
- 第十五章、例行性工作調度(crontab)
- 15.1 什么是例行性工作調度
- 15.2 僅執行一次的工作調度
- 15.3 循環執行的例行性工作調度
- 15.4 可喚醒停機期間的工作任務
- 15.5 重點回顧
- 15.6 本章習題
- 第十六章、程序管理與 SELinux 初探
- 16.1 什么是程序 (process)
- 16.2 工作管理 (job control)
- 16.3 程序管理
- 16.4 特殊文件與程序
- 16.5 SELinux 初探
- 16.6 重點回顧
- 16.7 本章習題
- 16.8 參考資料與延伸閱讀
- 第十七章、認識系統服務 (daemons)
- 17.1 什么是 daemon 與服務 (service)
- 17.2 通過 systemctl 管理服務
- 17.3 systemctl 針對 service 類型的配置文件
- 17.4 systemctl 針對 timer 的配置文件
- 17.5 CentOS 7.x 默認啟動的服務簡易說明
- 17.6 重點回顧
- 17.7 本章習題
- 17.8 參考資料與延伸閱讀
- 第十八章、認識與分析登錄文件
- 18.1 什么是登錄文件
- 18.2 rsyslog.service :記錄登錄文件的服務
- 18.3 登錄文件的輪替(logrotate)
- 18.4 systemd-journald.service 簡介
- 18.5 分析登錄文件
- 18.6 重點回顧
- 18.7 本章習題
- 18.8 參考資料與延伸閱讀
- 第十九章、開機流程、模塊管理與 Loader
- 19.1 Linux 的開機流程分析
- 19.2 核心與核心模塊
- 19.3 Boot Loader: Grub2
- 19.4 開機過程的問題解決
- 19.5 重點回顧
- 19.6 本章習題
- 19.7 參考資料與延伸閱讀
- 第二十章、基礎系統設置與備份策略
- 20.1 系統基本設置
- 20.2 服務器硬件數據的收集
- 20.3 備份要點
- 20.4 備份的種類、頻率與工具的選擇
- 20.5 鳥哥的備份策略
- 20.6 災難復原的考慮
- 20.7 重點回顧
- 20.8 本章習題
- 20.9 參考資料與延伸閱讀
- 第二十一章、軟件安裝:源代碼與 Tarball
- 20.1 開放源碼的軟件安裝與升級簡介
- 21.2 使用傳統程序語言進行編譯的簡單范例
- 21.3 用 make 進行宏編譯
- 21.4 Tarball 的管理與建議
- 21.5 函數庫管理
- 21.6 檢驗軟件正確性
- 21.7 重點回顧
- 21.8 本章習題
- 21.9 參考資料與延伸閱讀
- 第二十二章、軟件安裝 RPM, SRPM 與 YUM
- 22.1 軟件管理員簡介
- 22.2 RPM 軟件管理程序: rpm
- 22.3 YUM 線上升級機制
- 22.4 SRPM 的使用 : rpmbuild (Optional)
- 22.5 重點回顧
- 22.6 本章習題
- 22.7 參考資料與延伸閱讀
- 第二十三章、X Window 設置介紹
- 23.1 什么是 X Window System
- 23.2 X Server 配置文件解析與設置
- 23.3 顯卡驅動程序安裝范例
- 23.4 重點回顧
- 23.5 本章習題
- 23.6 參考資料與延伸閱讀
- 第二十四章、Linux 核心編譯與管理
- 24.1 編譯前的任務:認識核心與取得核心源代碼
- 24.2 核心編譯的前處理與核心功能選擇
- 24.3 核心的編譯與安裝
- 24.4 額外(單一)核心模塊編譯
- 24.5 以最新核心版本編譯 CentOS 7.x 的核心
- 24.6 重點回顧
- 24.7 本章習題
- 24.8 參考資料與延伸閱讀