人人做人人澡人人爽欧美,国产主播一区二区,久久久精品五月天,羞羞视频在线观看免费

當前位置:蘿卜系統下載站 > 技術開發教程 > 詳細頁面

處理.net開發問題的最終法寶

處理.net開發問題的最終法寶

更新時間:2022-09-20 文章作者:未知 信息來源:網絡 閱讀次數:

這兩天正在網上找工作。昨天一網友問了個問題,說SendMessage在.net中調用失敗。
我看了看他的代碼是用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下內部使用的是SendMessageW,而SendMessageA的調用則是先將參數轉換后再調用SendMessageW,所以這里我是用SendMessageW。)
這個程序的功能是查找一個記事本程序,然后向他發送關閉消息。
試驗了一下,果然失敗了。
開始我懷疑是.net的問題,因為一個網友曾經說過在VB中調用成功的代碼在VB.net中調用失敗了。于是我是用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調用的部分將p[0].Handle.ToInt32()放入參數1
IL_0016: call instance int32 [mscorlib]System.IntPtr::ToInt32()
IL_001b: stloc.1
IL_001c: ldloc.1
//放入16,參數2
IL_001d: ldc.i4.s 16
//放入0,參數3
IL_001f: ldc.i4.0
//放入0,參數4
IL_0020: ldc.i4.0
IL_0021: call int64 TWin.Class1::SendMessageW(int32,
int32,
int32,
int32)
IL_0026: pop
IL_0027: ret
看不出有什么問題,本來我以為是不是某個值被裝箱之類的操作了。一看所有參數都是標準的int32類型,這個可是值啊。
我并沒有就此覺悟,反而還是執迷不悟的懷疑是.net出的問題。
于是運行程序,在SendMessageW(i,16,0,0)處設置斷點,看看匯編碼。(運行時,在代碼的旁邊有個反匯編的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函數的調用使用的方法是將參數值壓入棧中(后進現出)的原則,所以參數壓入順序為4321。
匯編中的語句
0000004f push 0
00000051 push 0
00000053 mov ecx,edi
00000055 mov edx,10h
都沒錯。唯一有可能出錯的就是
0000005a call dword ptr ds:[009C50F8h]
于是問題很清楚了,.net調用API函數的方法沒有錯,是傳遞的參數出了錯。用VC++的工具SPY++看看吧,果然發現句柄與p[0].Handle.ToInt32()不符。無意間發現p[0].MainWindowHandle與SPY++的結果相符,忽然間恍然大悟。大罵自己愚蠢分明需要給程序的窗體傳送消息,你給那個進程ID傳送,人家誰理你啊!
改用p[0].MainWindowHandle實驗,記事本被關閉了。雖然犯了傻不過倒是總結了些解決.net出的問題的一些方法,如果你有些問題感到莫名其妙,找不到方法就不如用ILDASM去反編譯一下,看看IL代碼說不定有幫助,如果還不行,干脆就用看看匯編碼,說不定問題就明白了。
告訴他后,他又問了我另一個問題,一個窗體如何得到自己的句柄呢,其實這個很簡單,this.Handle.ToInt32()就是自己的句柄了。
結尾打個廣告吧,本人軟開工作兩年想找一份月薪3000元的北京的工作。有意者請與我聯系fjl716@16

溫馨提示:喜歡本站的話,請收藏一下本站!

本類教程下載

系統下載排行

網站地圖xml | 網站地圖html
主站蜘蛛池模板: 宁明县| 大兴区| 江门市| 广昌县| 自治县| 中宁县| 保定市| 叙永县| 云和县| 龙南县| 科技| 威信县| 绥阳县| 凤城市| 光山县| 台中县| 曲阳县| 内乡县| 新津县| 隆德县| 潍坊市| 大足县| 尚义县| 怀仁县| 景谷| 栾城县| 凤山市| 墨竹工卡县| 正阳县| 含山县| 上虞市| 霍林郭勒市| 寿阳县| 达孜县| 荥阳市| 彩票| 牡丹江市| 和田县| 木兰县| 黄龙县| 贵溪市|