編輯:關於android開發
本篇將對“1=3”“&5”這樣無法求值的不正確的表達式進行檢查。
將檢查如下這些問題。
●為無法賦值的表達式賦值(例:1 = 2 + 2)
●使用非法的函數名調用函數(例:"string"("%d\n", i))
●操作數非法的數組引用(例:1[0])
●操作數非法的成員引用(例:1.memb)
●操作數非法的指針間接引用(例:1->memb)
●對非指針的對象取值(例:*1)
●對非左值的表達式取地址
具體例子以及問題的檢測方法如表10.1所示,其中包括了剛才列舉的問題。
非指針類型取值操作的檢查
/*非指針類型取值操作的檢查
* 表示取值運算符(*)的DereferenceNode的處理。
* 該方法檢查取值運算符的操作數的類型是否為指針。
*/
// #@@range/DereferenceNode{
public Void visit(DereferenceNode node) {
/*
* 首先,通過super.visit(node) 調用基類Visitor 的方法遍歷操作數(node.expr())
(即檢查操作數)。
*/
super.visit(node);
/*
* 接著,調用操作數node.expr() 的isPointer 方法,檢查操作數的類型是否是指針,
即檢查是否可以進行取值。如果無法取值,則調用undereferableError 方法輸出編譯錯誤。
*/
if (! node.expr().isPointer()) {
undereferableError(node.location());
}
/*
* 最後,調用handleImplicitAddress 方法對數組類型和函數類型進行特別處理。該處
理還和接下來AddressNode 的處理相關,
*/
handleImplicitAddress(node);
return null;
}
獲取非左值表達式地址的檢查

/*獲取非左值表達式地址的檢查
* 檢查操作數是否為左值。表示地址運算符的AddressNode 的處理
*/
// #@@range/AddressNode{
public Void visit(AddressNode node) {
super.visit(node);
/*
* 首先對node.expr() 調用isLvalue 方法,檢查&expr 中的expr 是否是可以進行取
址操作的表達式。
ExprNode#isLvalue 是檢查該節點的表達式是否能夠獲取地址的方法。
*/
if (! node.expr().isLvalue()) {
semanticError(node.location(), "invalid expression for &");
}
/*
* 剩余的語句用於確定AddressNode 的類型。通常node.expr().isLoadable() 會
返回true,即執行else 部分的處理。&expr 的類型是指向expr 類型的指針,因此指向
node.expr().type() 的指針類型可以作為節點整體的類型來使用。
*/
Type base = node.expr().type();
/*
* 在將puts 的類型設置為指向函數的指針的同時,還必須將&puts 的類型也設置為指向函
數的指針。
node.expr() 的類型是數組或函數的情況下進行特別處理,使得&puts 的類型
和puts 的類型相一致。
*/
if (! node.expr().isLoadable()) {
// node.expr.type is already pointer.
node.setType(base);
}
else {
node.setType(typeTable.pointerTo(base));
}
return null;
}
隱式的指針生成
單個數組類型或函數類型的變量表示數組或函數的地址。例如,假設變量puts 的類型為函數類型(一般稱為函數指針),那麼puts 和&puts 得到的值是相同的。
/*
* handleImplicitAddress 方法將數組類型或函數類型轉換為了指向
數組或函數類型的指針,即隱式地生成指針類型。
*/
private void handleImplicitAddress(LHSNode node) {
if (! node.isLoadable()) {
Type t = node.type();
if (t.isArray()) {
// int[4] ary; ary; should generate int*
node.setType(typeTable.pointerTo(t.baseType()));
}
else {
node.setType(typeTable.pointerTo(t));
}
}
}
puts 是指向函數的指針,因此它的取值運算*puts 的結果是函數類型,但這樣又會隱式地轉換為指向函數的指針。*puts 還是指向函數的指針,因此仍然可以進行取值運算,仍然會轉換為指向函數的指針。像這樣可以無限重復下去。所以C 語言中“&puts”“puts”“*puts”“**puts”“***puts”的值都是相同的。
安卓客戶端a標簽長按彈框提示解決辦法,安卓標簽
安卓客戶端a標簽長按彈框提示解決辦法,安卓標簽昨天工作時候發現一個bug,是關於a標簽的,在安卓客戶端中,如果是a標簽的話,長按會出現一個彈框,如圖所示 是因為安卓客戶端
解析BroadcastReceiver之你需要了解的一些東東,broadcastreceiver
解析BroadcastReceiver之你需要了解的一些東東,broadcastreceiver 前些天把四大組件之一的Service扯了一遍,今天就要開始談談它的弟兄
安卓7.0首批截圖公布:驚現漢堡菜單,安卓漢堡
安卓7.0首批截圖公布:驚現漢堡菜單,安卓漢堡今天谷歌在開發者頁面公布了首批Android N(安卓7.0)截圖,從截圖中能夠了解到的一點是安卓7.0設置界面中會添加漢堡
android listview多視圖嵌套多視圖,androidlistview
android listview多視圖嵌套多視圖,androidlistview筆記,listview視圖總結 1 public class HomeEduMoreA