明輝手游網(wǎng)中心:是一個免費提供流行視頻軟件教程、在線學(xué)習(xí)分享的學(xué)習(xí)平臺!

處理.net開發(fā)問題的最終法寶

[摘要]這兩天正在網(wǎng)上找工作。昨天一網(wǎng)友問了個問題,說SendMessage在.net中調(diào)用失敗。我看了看他的代碼是用VB.net寫的。于是我改用C#寫了個小的測試程序using System.Runtime.InteropServices;[DllImport("user32.dll"...
這兩天正在網(wǎng)上找工作。昨天一網(wǎng)友問了個問題,說SendMessage在.net中調(diào)用失敗。
我看了看他的代碼是用VB.net寫的。于是我改用C#寫了個小的測試程序
using System.Runtime.InteropServices;

[DllImport("user32.dll")]
private static extern long SendMessageW(int hwnd,int wMsg,int wParam,int lParam);

System.Diagnostics.Process[] p;
p=System.Diagnostics.Process.GetProcessesByName("notepad");
int i=p[0].Handle.ToInt32();
SendMessageW(i,16,0,0);

(因為SendMessage分兩個版本,一個是SendMessageA一個是SendMessageW,由于NT下內(nèi)部使用的是SendMessageW,而SendMessageA的調(diào)用則是先將參數(shù)轉(zhuǎn)換后再調(diào)用SendMessageW,所以這里我是用SendMessageW。)
這個程序的功能是查找一個記事本程序,然后向他發(fā)送關(guān)閉消息。
試驗了一下,果然失敗了。
開始我懷疑是.net的問題,因為一個網(wǎng)友曾經(jīng)說過在VB中調(diào)用成功的代碼在VB.net中調(diào)用失敗了。于是我是用ILDASM對該程序進行反匯編。反編譯后IL代碼如下。
.entrypoint
.custom instance void [mscorlib]System.STAThreadAttribute::.ctor() = ( 01 00 00 00 )
// 代碼大小 40 (0x28)
.maxstack 4
.locals init ([0] class [System]System.Diagnostics.Process[] p,
[1] int32 i,
[2] native int CS$00000002$00000000)
IL_0000: ldstr "notepad"
IL_0005: call class [System]System.Diagnostics.Process[] [System]System.Diagnostics.Process::GetProcessesByName(string)
IL_000a: stloc.0
IL_000b: ldloc.0
IL_000c: ldc.i4.0
IL_000d: ldelem.ref
IL_000e: callvirt instance native int [System]System.Diagnostics.Process::get_Handle()
IL_0013: stloc.2
IL_0014: ldloca.s CS$00000002$00000000
//這里是SendMessageW調(diào)用的部分將p[0].Handle.ToInt32()放入?yún)?shù)1
IL_0016: call instance int32 [mscorlib]System.IntPtr::ToInt32()
IL_001b: stloc.1
IL_001c: ldloc.1
//放入16,參數(shù)2
IL_001d: ldc.i4.s 16
//放入0,參數(shù)3
IL_001f: ldc.i4.0
//放入0,參數(shù)4
IL_0020: ldc.i4.0
IL_0021: call int64 TWin.Class1::SendMessageW(int32,
int32,
int32,
int32)
IL_0026: pop
IL_0027: ret
看不出有什么問題,本來我以為是不是某個值被裝箱之類的操作了。一看所有參數(shù)都是標準的int32類型,這個可是值啊。
我并沒有就此覺悟,反而還是執(zhí)迷不悟的懷疑是.net出的問題。
于是運行程序,在SendMessageW(i,16,0,0)處設(shè)置斷點,看看匯編碼。(運行時,在代碼的旁邊有個反匯編的tab,通過它你就可以看到匯編碼了)
0000004f push 0
00000051 push 0
00000053 mov ecx,edi
00000055 mov edx,10h
0000005a call dword ptr ds:[009C50F8h]
00000060 nop
在Win32匯編中API函數(shù)的調(diào)用使用的方法是將參數(shù)值壓入棧中(后進現(xiàn)出)的原則,所以參數(shù)壓入順序為4321。
匯編中的語句
0000004f push 0
00000051 push 0
00000053 mov ecx,edi
00000055 mov edx,10h
都沒錯。唯一有可能出錯的就是
0000005a call dword ptr ds:[009C50F8h]
于是問題很清楚了,.net調(diào)用API函數(shù)的方法沒有錯,是傳遞的參數(shù)出了錯。用VC++的工具SPY++看看吧,果然發(fā)現(xiàn)句柄與p[0].Handle.ToInt32()不符。無意間發(fā)現(xiàn)p[0].MainWindowHandle與SPY++的結(jié)果相符,忽然間恍然大悟。大罵自己愚蠢分明需要給程序的窗體傳送消息,你給那個進程ID傳送,人家誰理你。
改用p[0].MainWindowHandle實驗,記事本被關(guān)閉了。雖然犯了傻不過倒是總結(jié)了些解決.net出的問題的一些方法,如果你有些問題感到莫名其妙,找不到方法就不如用ILDASM去反編譯一下,看看IL代碼說不定有幫助,如果還不行,干脆就用看看匯編碼,說不定問題就明白了。
告訴他后,他又問了我另一個問題,一個窗體如何得到自己的句柄呢,其實這個很簡單,this.Handle.ToInt32()就是自己的句柄了。
結(jié)尾打個廣告吧,本人軟開工作兩年想找一份月薪3000元的北京的工作。有意者請與我聯(lián)系fjl716@16