在整理自己的代碼的時候,考慮到我寫的代碼從一至終都是在一個cpp文件裡面。於是,想把自己的代碼中的各個模塊分離開來,以便更好地閱讀和管理。可在分離的時候出現了xxx變數已經在*.obj中定義的問題,即我定義的全局變數出現了重覆定義的現象。深究編譯鏈接的過程,發現static關鍵字的方法不可行,唯獨好... ...
前言
今天,在整理自己的代碼的時候,考慮到我寫的代碼從一至終都是在一個cpp文件裡面。於是,想把自己的代碼中的各個模塊分離開來,以便更好地閱讀和管理。
遇到的問題
我的做法是:
- 巨集定義、結構體定義、函數聲明以及全局變數定義放到一個head.h頭文件中
- 函數的定義放到head.cpp中
- main函數放到main.cpp中
然而卻報錯了,提示xxx變數在*.obj文件中已定義
問題出現的原因
為什麼會出現這種情況呢?
- 首先單個文件的編譯是獨立的。在head.cpp編譯到head.obj,main.cpp編譯到main.obj。這個過程沒有報錯,也就是說明編譯過程是沒有問題的。
- 接下來是obj的鏈接。在鏈接main.obj與head.obj的時候,此時編譯器發現head.obj為這些全局變數分配了記憶體空間,而在main.obj中也為這些全局變數分配了記憶體空間。
- 同樣一個變數卻出現了兩個不同的記憶體地址。於是編譯器報錯。
不是辦法的辦法
把head.h裡面的頭文件的全局變數都加上static。編譯便可通過,可是卻會不經意出現了其他問題。
static只是把變數的生存周期延長,同時也把該變數限定於當前的文件。而之所以能用於main.cpp中,是因為在編譯的時候複製了一個變數名相同的變數給main.cpp而已。那麼main.cpp裡面的“全局變數”的改變,並不能改變原來head.h裡面的全局變數的值。
這樣子雖然編譯通過了,但是程式是錯誤的。
真正的解決方法
- 把全局變數定義放到head.cpp文件中。
- 在head.h存放全局變數的聲明,同時每個聲明前用
extern
去修飾。
我的個人想法
我覺得為了能更加分離全局變數,可以做的一個做法是:
- 全局變數定義依舊放在head.cpp中。
- 新建一個global.h的頭文件,存放全局變數的聲明,同時每個聲明前用
extern
去修飾。 - 在其他文件需要用到全局變數的時候,將global.h頭文件#include進來。
結言
這個問題的出現,很大原因是C語言太久沒有使用過了。而且,在使用c語言或者c++語言的時候,往往因為實驗以及課設所需要寫的代碼不太多,於是養成了一種習慣,一個main.cpp寫到結尾。當真正自己去分離自己的模塊代碼的時候,發現因為定義的全局變數導致編譯鏈接出現錯誤,實屬不該。故寫下此文警惕自己!文中可能有不對的地方,希望大家能指正!
文章出自kwongtai'blog,轉載請標明出處