• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            為生存而奔跑

               :: 首頁(yè) :: 聯(lián)系 :: 聚合  :: 管理
              271 Posts :: 0 Stories :: 58 Comments :: 0 Trackbacks

            留言簿(5)

            我參與的團(tuán)隊(duì)

            搜索

            •  

            積分與排名

            • 積分 - 328435
            • 排名 - 74

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            .Net Framework SDK文檔中,關(guān)于調(diào)用Windows API的指示比較零散,并且其中稍全面一點(diǎn)的是針對(duì)Visual Basic .net講述的。本文將C#中調(diào)用API的要點(diǎn)匯集如下,希望給未在C#中使用過(guò)API的朋友一點(diǎn)幫助。另外如果安裝了Visual Studio .net的話,在C:\Program Files\Microsoft Visual Studio .NET\FrameworkSDK\Samples\Technologies\Interop\PlatformInvoke\WinAPIs\CS目錄下有大量的調(diào)用API的例子。
            一、調(diào)用格式
             using System.Runtime.InteropServices; //
            引用此名稱空間,簡(jiǎn)化后面的代碼
             ...
             //
            使用DllImportAttribute特性來(lái)引入api函數(shù),注意聲明的是空方法,即方法體為空。
             [DllImport("user32.dll")]
             public static extern ReturnType FunctionName(type arg1,type arg2,...);
             //
            調(diào)用時(shí)與調(diào)用其他方法并無(wú)區(qū)別

             可以使用字段進(jìn)一步說(shuō)明特性,用逗號(hào)隔開,如:
             [ DllImport( "kernel32", EntryPoint="GetVersionEx" )] 
             DllImportAttribute
            特性的公共字段如下:
             1
            CallingConvention 指示向非托管實(shí)現(xiàn)傳遞方法參數(shù)時(shí)所用的 CallingConvention 值。 
              CallingConvention.Cdecl :
            調(diào)用方清理堆棧。它使您能夠調(diào)用具有 varargs 的函數(shù)。
              CallingConvention.StdCall :
            被調(diào)用方清理堆棧。它是從托管代碼調(diào)用非托管函數(shù)的默認(rèn)約定。
             2
            CharSet 控制調(diào)用函數(shù)的名稱版本及指示如何向方法封送 String 參數(shù)。
              
            此字段被設(shè)置為 CharSet 值之一。如果 CharSet 字段設(shè)置為 Unicode,則所有字符串參數(shù)在傳遞到非托管實(shí)現(xiàn)之前都轉(zhuǎn)換成 Unicode 字符。這還導(dǎo)致向 DLL EntryPoint 的名稱中追加字母“W”。如果此字段設(shè)置為 Ansi,則字符串將轉(zhuǎn)換成 ANSI 字符串,同時(shí)向 DLL EntryPoint 的名稱中追加字母“A”。大多數(shù) Win32 API 使用這種追加“W”“A”的約定。如果 CharSet 設(shè)置為 Auto,則這種轉(zhuǎn)換就是與平臺(tái)有關(guān)的(在 Windows NT 上為 Unicode,在 Windows 98 上為 Ansi)。CharSet 的默認(rèn)值為 AnsiCharSet 字段也用于確定將從指定的 DLL 導(dǎo)入哪個(gè)版本的函數(shù)。CharSet.Ansi CharSet.Unicode 的名稱匹配規(guī)則大不相同。對(duì)于 Ansi 來(lái)說(shuō),如果將 EntryPoint 設(shè)置為“MyMethod”且它存在的話,則返回“MyMethod”。如果 DLL 中沒(méi)有“MyMethod”,但存在“MyMethodA”,則返回“MyMethodA”。對(duì)于 Unicode 來(lái)說(shuō)則正好相反。如果將 EntryPoint 設(shè)置為“MyMethod”且它存在的話,則返回“MyMethodW”。如果 DLL 中不存在“MyMethodW”,但存在“MyMethod”,則返回“MyMethod”。如果使用的是 Auto,則匹配規(guī)則與平臺(tái)有關(guān)(在 Windows NT 上為 Unicode,在 Windows 98 上為 Ansi)。如果 ExactSpelling 設(shè)置為 true,則只有當(dāng) DLL 中存在“MyMethod”時(shí)才返回“MyMethod”

             3EntryPoint 指示要調(diào)用的 DLL 入口點(diǎn)的名稱或序號(hào)。 
              
            如果你的方法名不想與api函數(shù)同名的話,一定要指定此參數(shù),例如:
             [DllImport("user32.dll",CharSet="CharSet.Auto",EntryPoint="MessageBox")]
             public static extern int MsgBox(IntPtr hWnd,string txt,string caption, int type);

             4ExactSpelling 指示是否應(yīng)修改非托管 DLL 中的入口點(diǎn)的名稱,以與 CharSet 字段中指定的 CharSet 值相對(duì)應(yīng)。如果為 true,則當(dāng) DllImportAttribute.CharSet 字段設(shè)置為 CharSet Ansi 值時(shí),向方法名稱中追加字母 A,當(dāng) DllImportAttribute.CharSet 字段設(shè)置為 CharSet Unicode 值時(shí),向方法的名稱中追加字母 W。此字段的默認(rèn)值是 false 
             5
            PreserveSig 指示托管方法簽名不應(yīng)轉(zhuǎn)換成返回 HRESULT、并且可能有一個(gè)對(duì)應(yīng)于返回值的附加 [out, retval] 參數(shù)的非托管簽名。 
             6
            SetLastError 指示被調(diào)用方在從屬性化方法返回之前將調(diào)用 Win32 API SetLastError true 指示調(diào)用方將調(diào)用 SetLastError,默認(rèn)為 false。運(yùn)行時(shí)封送拆收器將調(diào)用 GetLastError 并緩存返回的值,以防其被其他 API 調(diào)用重寫。用戶可通過(guò)調(diào)用 GetLastWin32Error 來(lái)檢索錯(cuò)誤代碼。

            二、參數(shù)類型:
             1
            、數(shù)值型直接用對(duì)應(yīng)的就可。(DWORD -> int , WORD -> Int16
             2
            API中字符串指針類型 -> .netstring
             3
            API中句柄 (dWord)  -> .netIntPtr
             4
            API中結(jié)構(gòu)   -> .net中結(jié)構(gòu)或者類。注意這種情況下,要先用StructLayout特性限定聲明結(jié)構(gòu)或類
            公共語(yǔ)言運(yùn)行庫(kù)利用StructLayoutAttribute控制類或結(jié)構(gòu)的數(shù)據(jù)字段在托管內(nèi)存中的物理布局,即類或結(jié)構(gòu)需要按某種方式排列。如果要將類傳遞給需要指定布局的非托管代碼,則顯式控制類布局是重要的。它的構(gòu)造函數(shù)中用LayoutKind值初始化 StructLayoutAttribute 類的新實(shí)例。 LayoutKind.Sequential 用于強(qiáng)制將成員按其出現(xiàn)的順序進(jìn)行順序布局。
             LayoutKind.Explicit
            用于控制每個(gè)數(shù)據(jù)成員的精確位置。利用 Explicit 每個(gè)成員必須使用 FieldOffsetAttribute 指示此字段在類型中的位置。如:
             [StructLayout(LayoutKind.Explicit, Size=16, CharSet=CharSet.Ansi)]
             public class MySystemTime 
             {
                [FieldOffset(0)]public ushort wYear; 
                [FieldOffset(2)]public ushort wMonth;
                [FieldOffset(4)]public ushort wDayOfWeek; 
                [FieldOffset(6)]public ushort wDay; 
                [FieldOffset(8)]public ushort wHour; 
                [FieldOffset(10)]public ushort wMinute; 
                [FieldOffset(12)]public ushort wSecond; 
                [FieldOffset(14)]public ushort wMilliseconds; 
             }
             
            下面是針對(duì)APIOSVERSIONINFO結(jié)構(gòu),在.net中定義對(duì)應(yīng)類或結(jié)構(gòu)的例子:
            /**********************************************
            * API
            中定義原結(jié)構(gòu)聲明
            * OSVERSIONINFOA STRUCT
            *  dwOSVersionInfoSize   DWORD      ?
            *  dwMajorVersion        DWORD      ?
            *  dwMinorVersion        DWORD      ?
            *  dwBuildNumber         DWORD      ?
            *  dwPlatformId          DWORD      ?
            *  szCSDVersion          BYTE 128 dup (?)
            * OSVERSIONINFOA ENDS
            *
            * OSVERSIONINFO  equ  <OSVERSIONINFOA>
            *********************************************/

            //.net中聲明為類
            [ StructLayout( LayoutKind.Sequential )]   
            public class OSVersionInfo 
            {   
                public int OSVersionInfoSize;
                public int majorVersion; 
                public int minorVersion;
                public int buildNumber;
                public int platformId;

                [ MarshalAs( UnmanagedType.ByValTStr, SizeConst=128 )]    
                public String versionString;
            }
            //
            或者
            //.net
            中聲明為結(jié)構(gòu)
            [ StructLayout( LayoutKind.Sequential )]  
            public struct OSVersionInfo2 
            {
                public int OSVersionInfoSize;
                public int majorVersion; 
                public int minorVersion;
                public int buildNumber;
                public int platformId;

                [ MarshalAs( UnmanagedType.ByValTStr, SizeConst=128 )]    
                public String versionString;
            }

             此例中用到MashalAs特性,它用于描述字段、方法或參數(shù)的封送處理格式。用它作為參數(shù)前綴并指定目標(biāo)需要的數(shù)據(jù)類型。例如,以下代碼將兩個(gè)參數(shù)作為數(shù)據(jù)類型長(zhǎng)指針?lè)馑徒o Windows API 函數(shù)的字符串 (LPStr) 
                [MarshalAs(UnmanagedType.LPStr)]
             String existingfile;
                [MarshalAs(UnmanagedType.LPStr)]
             String newfile;

             注意結(jié)構(gòu)作為參數(shù)時(shí)候,一般前面要加上ref修飾符,否則會(huì)出現(xiàn)錯(cuò)誤:對(duì)象的引用沒(méi)有指定對(duì)象的實(shí)例。
             [ DllImport( "kernel32", EntryPoint="GetVersionEx" )] 
             public static extern bool GetVersionEx2( ref OSVersionInfo2 osvi );

            三、如何保證使用托管對(duì)象的平臺(tái)調(diào)用成功?
               
            如果在調(diào)用平臺(tái) invoke 后的任何位置都未引用托管對(duì)象,則垃圾回收器可能將完成該托管對(duì)象。這將釋放資源并使句柄無(wú)效,從而導(dǎo)致平臺(tái)invoke 調(diào)用失敗。用 HandleRef 包裝句柄可保證在平臺(tái) invoke 調(diào)用完成前,不對(duì)托管對(duì)象進(jìn)行垃圾回收。
               
            例如下面:
                    FileStream fs = new FileStream( "a.txt", FileMode.Open );
                    StringBuilder buffer = new StringBuilder( 5 );
                    int read = 0;
                    ReadFile(fs.Handle, buffer, 5, out read, 0 ); //
            調(diào)用Win API中的ReadFile函數(shù)
             
            由于fs是托管對(duì)象,所以有可能在平臺(tái)調(diào)用還未完成時(shí)候被垃圾回收站回收。將文件流的句柄用HandleRef包裝后,就能避免被垃圾站回收:
             [ DllImport( "Kernel32.dll" )]
             public static extern bool ReadFile( 
              HandleRef hndRef, 
              StringBuilder buffer, 
              int numberOfBytesToRead, 
              out int numberOfBytesRead, 
              ref Overlapped flag );
             ......
             ......
                    FileStream fs = new FileStream( "HandleRef.txt", FileMode.Open );
                    HandleRef hr = new HandleRef( fs, fs.Handle );
                    StringBuilder buffer = new StringBuilder( 5 );
                    int read = 0;
                    // platform invoke will hold reference to HandleRef until call ends
                    ReadFile( hr, buffer, 5, out read, 0 );

             

            posted on 2009-12-25 12:54 baby-fly 閱讀(243) 評(píng)論(0)  編輯 收藏 引用 所屬分類: C#
            亚洲国产成人久久综合碰碰动漫3d| 精品精品国产自在久久高清| 国产亚洲欧美成人久久片| 香蕉久久夜色精品国产2020 | 国产成人久久777777| 亚洲狠狠婷婷综合久久蜜芽| 久久久久久国产精品美女| 一本一道久久精品综合| 97久久超碰国产精品旧版| 久久亚洲国产成人精品性色| 久久精品无码一区二区WWW| 亚洲国产成人久久精品99| 久久这里有精品视频| 久久一区二区三区免费| 久久露脸国产精品| 久久青青草原精品国产软件| 久久精品国产72国产精福利| 久久国产美女免费观看精品| 国产女人aaa级久久久级| 久久精品一区二区影院| 久久婷婷五月综合色99啪ak| 人人狠狠综合久久亚洲| 日本国产精品久久| 久久精品日日躁夜夜躁欧美 | 亚洲va久久久噜噜噜久久天堂 | 久久丫精品国产亚洲av不卡| 亚洲国产精品久久电影欧美| 久久精品国产精品亚洲毛片| 狠狠色丁香婷婷久久综合不卡 | 欧美久久一区二区三区| 久久精品极品盛宴观看| 久久婷婷五月综合成人D啪| 久久夜色精品国产噜噜噜亚洲AV| 久久精品人成免费| 99久久婷婷国产综合精品草原| 久久精品一区二区三区中文字幕 | 久久综合国产乱子伦精品免费| 日本久久久精品中文字幕| 久久影院久久香蕉国产线看观看| 无码久久精品国产亚洲Av影片 | 精品久久久久中文字幕日本|