昨天和一個同學討論了關於動態分配的數組調試的問題。
動態分配數組最直接也是最原始的方法就是使用動態分配記憶體空間的指標,如下代碼:
//動態分配一個行為2,列為2的2維數組,成員值分別是1,2,3,4
int rows, cols;
//行和列都設定成2
rows = cols = 2;
//數組頭指標
int **a;
//使用malloc分配空間
a = (int **)malloc(rows * sizeof(int *));
if(a == NULL)
return 1;
for(int i = 0; i < rows; i++)
{
a[i] = (int *)malloc(cols * sizeof(int));
if(a[i] == NULL)
return 1;
}
//成員賦值
a[0][0] = 1;
a[0][1] = 2;
a[1][0] = 3;
a[1][1] = 4;
當偵錯工具時,由於目標對象a不是編譯器可以知道長度的靜態數組,而是一系列的指標,所以調試時,無法顯示數組的全部成員,而僅僅是作為一個指標,顯示它指向的內容,由於a是指向指標的指標,所以a指向的空間又是一個指標,Visual Studio會再次吧這個指標所最終指向的數字1也會顯示出來。所以最初調試視窗中的本機資料中的a是這個樣子:
解決方案就是把變數加入到偵聽視窗中,在偵聽視窗中加入動態調試運算式來預覽a中的內容:
接著在偵聽視窗中,把原來的a修改成a,2(a逗號2),這告訴Visual Studio,我們要看a為起點後的兩個元素的內容(調試中還可以使用加號來修改這個起點的位置):
然後斷行符號:
此時,a指向的兩個指標,也就是數組的兩個行的頭指標(分別又指向1和3,可以從偵聽視窗中看到)就顯示出來了。
可是整個數組的全部成員還沒有顯示出來,此時則需要把這兩個頭指標再次添加偵聽,選中上面的[0]和[1],然後右鍵,添加偵聽(Add Watch):
現在Watch 1視窗中有三個資料了:
由於數組列數是2,再次在兩個數組行頭指標後加入逗號2,來查看其所指向空間兩個元素的值,最後,整個數組的成員就出來了:
這裡還有一個技巧,當第一個把a加入到偵聽視窗中,介面是這樣的(此時介面從Locals轉到Watch 1):
Watch不同於Locals是我們可以修改運算式的值,當然也可以添加任意多的運算式,看中的藍色選擇框,Locals裡不會有的,多出一行正是為了添加調試運算式的。
那麼此時我們也可以用運算式來偵聽a中的資料:
依次鍵入:*a,2和*(a+1),2結果就出來了:
可以看到,這樣動態分配數組,不僅寫起來不方便,調試起來也不方便(日後維護起來也會不方便的),做一還是推薦用vector這樣專門為動態數組和生的C++標準庫中的內部資料結構。調試起來也很方便的:
下面代碼,用vector聲明上述二維數組:
//+ #include <vector>
//動態分配一個行為2,列為2的2維vector<int>,成員值分別是1,2,3,4
int rows, cols;
//行和列都設定成2
rows = cols = 2;
vector<vector<int>> a;
//調整大小
a.resize(rows);
for(int i = 0; i < rows; i++)
a[i].resize(cols);
//成員賦值
a[0][0] = 1;
a[0][1] = 2;
a[1][0] = 3;
a[1][1] = 4;
然後調試起來非常方便,不需要另作手腳,資料一目瞭然: