作者ab710912 (Leanne)
看板C_Sharp
標題[問題] 關於引用C++ dll的變數類別
時間Mon Nov 29 11:50:49 2021
工作上收到一份dll
他給的文件方法是
SnmpStatus(int& DeviceStatus,int& PrinterStatus,char* PrinterError);
所以我採用
[DllImport("MPrinterSnmp.dll", EntryPoint = "SnmpStatus", CallingConvention
= CallingConvention.StdCall,CharSet =CharSet.Unicode)]
public static extern int SnmpStatus(out int DeviceStatus, out int
PrinterStatus,ref StringBuilder PrinterError);
可是一呼叫 程式就直接結束工作,連例外都沒跳出
曾經有一次 把 StringBuilder改成Char有成功取得值一次
但是後來再試就一樣直接跳掉
StringBuilder 改試過 String、byte[2]、String[2]、IntPtr、char、char[2]
都是直接跳掉
另外 他有附他的.h給我 裡面的方法是
extern "C" __declspec(dllexport) int __stdcall SnmpStatus(int
&DeviceStatus,int &PrinterStatus,char *PrinterError);
*和&的位置和文件的位置有些微不一樣 不知道有沒有差別
--
※ 發信站: 批踢踢實業坊(ptt.cc), 來自: 60.248.154.118 (臺灣)
※ 文章網址: https://webptt.com/m.aspx?n=bbs/C_Sharp/M.1638157851.A.621.html
1F:→ ssccg: 要用IntPtr,然後用Marshal.PtrToStringAnsi 11/29 13:02
IntPtr我試過 至少要先不會跳掉才有機會可以去轉String吧?
※ 編輯: ab710912 (60.248.154.118 臺灣), 11/29/2021 13:15:47
2F:→ ssccg: 這函式看起來是傳char*(=IntPtr)進去,然後裡面會把結果直 11/30 11:23
3F:→ ssccg: 接寫到這個pointer的位置,你傳進去的IntPtr哪來的? 11/30 11:27
4F:→ ssccg: Marshal.AllocHGlobal? 11/30 11:28
5F:推 s4300026: 我覺得是 char 跟 wchar 的問題。 String^ 是wchar. 11/30 12:28
6F:→ s4300026: 話說為什麼不自己寫wrapper class,要用dllimport 11/30 12:28
7F:推 s4300026: 你char會在dll裡面改變大小嗎? 你怎麼保證不會overflow 11/30 12:29
8F:推 testPtt: unsafe 12/01 08:43
9F:推 s4300026: 話說好奇問一下,undafe的native pointer要怎麼new 和de 12/01 12:05
10F:→ s4300026: lete? 12/01 12:05
11F:→ testPtt: NativeMemory Class 12/01 13:38
12F:推 Litfal: 樓上是指IntPtr?通常是C library開api負責new和del吧 12/01 13:51
13F:→ Litfal: 然後C++的&不一定是out,用ref比較一致,雖然我覺得在api 12/01 13:54
14F:→ Litfal: 裡面用&還頗……特殊的,大部分都會用一致性更高的* 12/01 13:54
15F:推 Litfal: 你在丟StringBuilder和char[ ]時不需要ref,C#丟物件預設 12/01 13:58
16F:→ Litfal: 就是傳址了 12/01 13:58
17F:→ Litfal: StringBuilder要初始空間給他用,char[]和byte[]也是一樣 12/01 14:00
18F:→ Litfal: 。開大一點,拜託,能跑再來考慮怎麼省 12/01 14:00
19F:推 Litfal: 他api用char,應該不會是Unicode,注意一下 12/01 14:06
20F:推 s4300026: 回Litfal, 我是指*或是**,因為unsafe可以使用native pt 12/01 15:47
21F:→ s4300026: r,但c#並沒有對應的new delete關鍵字。 12/01 15:47
22F:推 s4300026: 確實我後來是在api中多宣告了new/delete的方法來解決。 12/01 15:51
23F:→ s4300026: 只是我在想,有沒有在外部new好再丟入dll內的方式。 12/01 15:51
25F:→ s4300026: 我目前是這樣寫,不是走dllimport的方式,但對於 & 和 % 12/01 16:05
26F:→ s4300026: 那裡不知道要怎麼轉會更好。 12/01 16:05
27F:推 Litfal: 有這種需求我是用Marshal.AllocHGlobal啦,不用fixed或St 12/01 16:33
28F:→ Litfal: ructureToPtr,我是用在圖像處理,C#端也有一些處理函數 12/01 16:33
29F:→ Litfal: ,對速度斤斤計較才這樣搞 12/01 16:33
感謝各位那麼認真的討論...昨天我有嘗試成功了
原因只是改成X86建置降為.net2.0就可以過了
改成這樣後無論類型帶甚麼都可以過 只是輸出值不一定有值或不一定對就是了
最後還是用了char
public static extern int SnmpStatus(out int DeviceStatus, out int
PrinterStatus, ref char PrinterError);
這樣感覺出來跟我要的結果比較像
感謝各位
※ 編輯: ab710912 (60.248.154.118 臺灣), 12/01/2021 17:22:07
※ 編輯: ab710912 (60.248.154.118 臺灣), 12/01/2021 17:22:57