1、gets() 方法
問:以下代碼有個被隱藏住的問題,你能找到它嗎?
答:這個不顯眼的問題就是使用了 gets() 方法。此方法接受一個string型別參數,但是卻沒有檢測此數值是否 有足夠的空間來拷貝資料。所以這裡我們一般用 fgets() 方法將來的更好。
2、strcpy() 方法
問:密碼防護是很基本的功能,看看能否搞定下面這段代碼?
3、main() 方法的傳回型別
問:請問下面這段代碼能否通過編譯?如果能的話,那麼這段代碼中隱含什麼問題嗎?
答:答案是代碼能通過編譯,但是會留下針對main()方法的傳回型別的警告。main()方法的真正傳回型別應該為'int'而非'void'。這是因為'int'傳回型別能夠讓程式返回狀態值。尤其是當這段程式作為其他應用的附屬程式時這個狀態值將更加重要。
4、記憶體泄露
問:請問以下代碼有記憶體泄露嗎?
答:雖然上面的代碼沒有對指標 ptr 進行記憶體釋放,但實際上即使是程式結束也不會造成記憶體泄露,因為當程式結束時所有一開始被佔據的記憶體就全部清空了。但如果上面這段代碼是在 while 迴圈裡面那將會造成嚴重的問題。
5、free() 方法
問:以下代碼當使用者輸入'freeze'時會奔潰,而如果輸入'zebra'則運行正常,這是為什麼?
答:問題的根源是因為代碼在while迴圈中改變了 ptr 指標的地址。當輸入為'zebra'時,while迴圈甚至在執行 第一遍前就結束了,所以free()釋放的記憶體位址就是一開始malloc()分配的地址。但是當輸入'freeze'時, ptr記錄的地址在while迴圈中被更改,因為將會是錯誤的地址傳遞到free()方法中引起崩潰。
6、atexit with _exit
問:在以下代碼,atexit()方法並沒有被調用,你知道為什麼嗎?
答:這是因為使用了 _exit() 方法。此方法並沒有調用清除資料相關的方法,比如 atexit()等。
7、void* 與 C 結構體
問:能否設計一個方法接受任意類型的參數然後返回整數?同時是否有辦法傳遞多個這樣的參數?
答:一個能接受任意型別參數的方法像下面這個樣子:
如果需要傳遞多個參數,那麼我們可以傳遞一個包含這些參數的結構體
8、* 與 ++ 操作符
問:以下代碼將輸出什麼?為什麼?
答:以上的輸出將是:
因為++與 * 的優先順序一樣,所以 *ptr++ 將會從右向左操作。按照這個邏輯,ptr++ 會先執行然後執行*ptr。所以第一個結果是'L'。也因為 ++ 被執行了,所以下一個printf() 結果是'i'。
9、Making changes in Code segment
問:以下代碼運行時一定會崩潰,你能說出原因嗎?
答:這是因為,通過 *ptr = 'T',此行代碼嘗試更改唯讀記憶體儲存的字串'Linux'。此操作當然行不通所以才會造成崩潰。
10、Process that changes its own name
問:你能否寫一個程式在它運行時修改它的名稱?
答:以下代碼可以完成
11、局部變數的返回地址
問:下面的代碼有問題嗎?如果有,如何修改?
答:雖然上面的代碼有時運行會很好,但是在方法 inc() 中有很嚴重的隱患。當inc()方法執行後,再次使用局部變數的地址就會造成不可估量的結果。解決之道就是傳遞變數a的地址給main()。
12、處理 printf() 參數
問:以下代碼輸出請問是什麼?
答:輸出將是
這是因為參數都是從右向左處理的,然後列印出來卻是從左向右。