昨天寫了個關於socket接收udp包的程序,調用了socket.ReceiveFrom方法,發現了一個c#中關於ref和out參數傳遞時的問題,這裡提出來和大家共同探討一下,首先聲明,下面的結論都是本人推測,還沒有得到任何定論,若有錯誤請大家指正。
首先,ReceiveFrom方法的原型為
public int ReceiveFrom(byte[], ref EndPoint);
有一個為ref的EndPoint參數,用它來返回收到包的源地址信息,ref的語義是傳引用,即對所傳引用的修改可以反映到方法外面。我一般都使用IPEndPoint來表示地址信息,所以很自然的使用了如下的調用方法
(代碼1)
IPEndPoint IEp = new IPEndPoint(IPAddress.Any,0);
socket.ReceiveFrom(buffer,ref (EndPoint)IEp);
這時編譯時出現了一下的錯誤,“ref或out參數必須是一個lvalue”,iep怎麼會不是一個左值呢?關鍵是在調用方法時使用的強制轉換(例如(EndPoint)IEp),我改了一下代碼
(代碼2)
IPEndPoint IEp = new IPEndPoint(IPAddress.Any,0);
EndPoint ep = (EndPoint)IEp;
socket.ReceiveFrom(buffer,ref (EndPoint)IEp);
這次通過編譯了。為什麼在方法調用時會出問題?這裡要考慮類型強制轉換時的一個細節,強制轉換時編譯器會先生成一個臨時引用,然後再把這個臨時引用傳給一個和轉換類型相同的引用,這個臨時引用比較特別——不是一個左值(lvalue),不能被賦值!而使用ref參數的方法一般都要對這個引用做修改,如果直接把這個臨時引用傳進去當時編譯器會抱怨ref或out參數必須是一個lvalue。而代碼2首先將這個臨時引用賦值到一個常規引用上去,這時這個常規引用便是可以復制的了。