2010-04-23 71 views
5

我正在用C++編寫注入的.dll文件,我想使用命名管道與C#應用程序進行通信。如何使用命名管道在C++ .dll和C#應用程序之間發送消息?

現在,我在C#應用程序中使用內置的System.IO.Pipe .net類,並且使用C++中的常規函數​​。我沒有太多的C++經驗(閱讀:這是我的第一個C++代碼..),我在C#中經驗豐富。

看來,與服務器和客戶端的連接正在工作,唯一的問題是messaged沒有被髮送。我試圖讓.dll服務器,C#應用程序服務器,使管道方向InOut(雙工),但似乎沒有工作。

當我試圖讓.dll文件服務器,這將消息發送到C#應用程序,我使用的代碼是這樣的:

DWORD ServerCreate() // function to create the server and wait till it successfully creates it to return. 
{ 
    hPipe = CreateNamedPipe(pipename,//The unique pipe name. This string must have the following form: \\.\pipe\pipename 
    PIPE_ACCESS_DUPLEX, 
    PIPE_TYPE_MESSAGE | PIPE_READMODE_MESSAGE | PIPE_NOWAIT, //write and read and return right away 
    PIPE_UNLIMITED_INSTANCES,//The maximum number of instances that can be created for this pipe 
    4096 , // output time-out 
    4096 , // input time-out 
    0,//client time-out 
    NULL); 

    if(hPipe== INVALID_HANDLE_VALUE) 
    { 
    return 1;//failed 
    } 
    else 
    return 0;//success 
} 

void SendMsg(string msg) 
{ 
    DWORD cbWritten; 
    WriteFile(hPipe,msg.c_str(), msg.length()+1, &cbWritten,NULL); 
} 

void ProccesingPipeInstance() 
{ 
    while(ServerCreate() == 1)//if failed 
{ 
    Sleep(1000);  
} 

//if created success, wait to connect 
ConnectNamedPipe(hPipe, NULL); 
for(;;) 
{ 
    SendMsg("HI!"); 
    if(ConnectNamedPipe(hPipe, NULL)==0) 
     if(GetLastError()==ERROR_NO_DATA) 
     { 
      DebugPrintA("previous closed,ERROR_NO_DATA"); 
      DisconnectNamedPipe(hPipe); 
      ConnectNamedPipe(hPipe, NULL); 
     } 
    Sleep(1000); 

} 

和C#cliend這樣的:

static void Main(string[] args) 
    { 
     Console.WriteLine("Hello!"); 

     using (var pipe = new NamedPipeClientStream(".", "HyprPipe", PipeDirection.In)) 
     { 
      Console.WriteLine("Created Client!"); 

      Console.Write("Connecting to pipe server in the .dll ..."); 
      pipe.Connect(); 

      Console.WriteLine("DONE!"); 

      using (var pr = new StreamReader(pipe)) 
      { 
       string t; 
       while ((t = pr.ReadLine()) != null) 
       { 
        Console.WriteLine("Message: {0}",t); 
       } 
      } 
     } 
    } 

我看到連接到.dll的C#客戶端,但它不會收到任何消息。 我嘗試以相反的方式執行此操作,正如前面所述,並試圖使C#將消息發送給.dll。 DLL,它會在消息框中顯示它們。 .dll被注入並連接到C#服務器,但是當它收到一條消息時,它剛剛崩潰了它注入的應用程序。

請幫助我,或者指導我如何使用C++和C#應用程序

回答

15

有幾件事情之間的命名管道。

1 - 您是否使用相同的管道名稱
C++: 「\\。\管\ HyperPipe」
C#: 「HyperPipe」

2 - 我想在C#方面,它可能最好使用ReadToEnd(),我只在C++中使用了命名管道,但我假設ReadToEnd()將讀取一條消息,因爲您正在使用基於消息 的管道。

3-在C++方面,您嘗試使用非阻塞管道並輪詢連接和數據等的管道。我會建議三件事之一。

 
    a - Use a blocking pipe on a separate thread or 
    b - Use non blocking pipes using Overlapped IO 
    c - Use non blocking pipes using IOCompletion ports 

第一種選擇將是最簡單和這聽起來像你正在做的,它會判罰尺度。 (a) http://msdn.microsoft.com/en-us/library/aa365588(VS.85).aspx

4-請確保您的編碼匹配雙方。例如,如果您的C++代碼是爲Unicode編譯的,則必須使用Unicode編碼在C#端讀取Pipe流。 有些事情如下。

using (StreamReader rdr = new StreamReader(pipe, System.Text.Encoding.Unicode)) 
{ 
    System.Console.WriteLine(rdr.ReadToEnd()); 
} 

更新:由於我沒有在C#與這個工作,我想我會寫一個小的測試。只需使用阻塞管道,不需要線程或任何東西,只是爲了確認基本工作,以下是非常粗糙的測試代碼。

C++服務器

#include <tchar.h> 
#include <windows.h> 

int _tmain(int argc, _TCHAR* argv[]) 
{ 
    HANDLE hPipe = ::CreateNamedPipe(_T("\\\\.\\pipe\\HyperPipe"), 
    PIPE_ACCESS_DUPLEX, 
    PIPE_TYPE_MESSAGE | PIPE_READMODE_MESSAGE, 
    PIPE_UNLIMITED_INSTANCES, 
    4096, 
    4096, 
    0, 
    NULL); 


    ConnectNamedPipe(hPipe, NULL); 

    LPTSTR data = _T("Hello"); 
    DWORD bytesWritten = 0; 
    WriteFile(hPipe, data, _tcslen(data) * sizeof(TCHAR), &bytesWritten, NULL); 
    CloseHandle(hPipe); 
    return 0; 
} 

C#的客戶

using System; 
using System.Text; 
using System.IO; 
using System.IO.Pipes; 

namespace CSPipe 
{ 
    class Program 
    { 
    static void Main(string[] args) 
    { 
     NamedPipeClientStream pipe = new NamedPipeClientStream(".", "HyperPipe", PipeDirection.InOut); 
     pipe.Connect(); 
     using (StreamReader rdr = new StreamReader(pipe, Encoding.Unicode)) 
     { 
     System.Console.WriteLine(rdr.ReadToEnd()); 
     } 

     Console.ReadKey(); 
    } 
    } 
} 
+0

哦,你應該把pipenames之間在C++和C#中大膽的差異。我浪費了時間解決這個問題,直到我找到了答案。 – 2017-10-10 16:46:37

+0

@BrianFairservice - 完成:)。我很高興這篇文章有幫助! – 2017-10-12 03:59:05

相關問題