簡體   English   中英

C ++客戶端和C#服務器通過網絡流之間的通信

[英]Communication between C++ Client and C# Server trough Network Stream

我有一個小shell程序,其中客戶端用c ++編寫,服務器用c#編寫。 客戶端應該只向服務器編寫一些cmd命令。 除了一件事,其他所有東西都運轉良好:服務器輸出總的紅寶石!

這是C#中的服務器:

    static NetworkStream Stream;
    static byte[] AWNSER = new byte[255];

    static void Main(string[] args)
    {
        TcpListener listen = new TcpListener(IPAddress.Any, 123);
        listen.Start();
        Stream = listen.AcceptTcpClient().GetStream();

        Stream.BeginRead(AWNSER, 0, 255, HNDLR, null);
        while (true)
        {
            var str = Console.ReadLine();
            byte[] MESSAGE = Encoding.UTF8.GetBytes(str + "&& cd");
            Stream.Write(MESSAGE, 0, MESSAGE.Length);
        }
    }
    static void HNDLR(IAsyncResult a)
    {
        Stream.EndRead(a);
        Stream.BeginRead(AWNSER, 0, 255, HNDLR, null);

        Console.WriteLine(Encoding.UTF8.GetString(AWNSER));
    }

這是C ++中的客戶端代碼

FILE *fp;
char AWNSER[255];
WSAData wsaData;
WORD DllVersion = MAKEWORD(2, 1);

if (WSAStartup(DllVersion, &wsaData) != 0) //no error
{
    cout << "Winsock startup failed";
}
SOCKADDR_IN addr;
int addrlen = sizeof(addr);
addr.sin_port = htons(123);
addr.sin_family = AF_INET;
addr.sin_addr.s_addr = inet_addr("199.166.177.22");
SOCKET Connection = socket(AF_INET, SOCK_STREAM, NULL);
if (connect(Connection, (SOCKADDR*)&addr, addrlen) != 0)
{   /*ERROR*/
}
else
{
        fp = _popen("dir", "r");
        while (fgets(AWNSER, 255, fp) != NULL) {
            cout << AWNSER;
            send(Connection, AWNSER, 255, NULL);
            AWNSER[255];
        }
        _pclose(fp);
}

這是服務器輸出的示例:

 Volumeseriennummer: BAF0-6BB2
Windows


    ?????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????

 olumeseriennummer: ABF1-7AA3
 Windows
 ??????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????

 olumeseriennummer: ABF1-7AA3
 Windows
 ??????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????
 Verzeichnis von C:\Users\User\Source\MyProblem
 ?????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????

 erzeichnis von C:\Users\User\Source\MyProblem
 ?????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????
30.07.2018  19:08    <DIR>          .
 myProblem

非常感謝您的幫助

即使沒有完全填寫,您的客戶端也會發送整個AWNSER數組。您應該只發送實際輸入的內容。

服務器將輸出其接收到的任何內容,但不考慮其實際接收到的字節數。 您需要考慮接收的字節數。

TCP是一種流傳輸。 發送和讀取之間沒有一對一的關系。 因此,您需要緩沖傳入的數據,並以某種方式構造數據,以使客戶端可以在完成發送消息后發出信號,而服務器可以知道一條完整的消息在哪里結束而下一條從哪里開始。 在簡單的示例中,空終止符或換行符就足夠了。

不要顯示部分字符串。 等待完整的字符串到達​​,然后再顯示它。 否則,在解碼尚未准備好解碼的內容時,就有可能損壞數據。

在服務器處理程序中,請勿在調用Console.WriteLine()之前調用Stream.BeginRead() Console.WriteLine() 使用AWNSER完成后調用它。 否則,您有機會在使用之前將AWNSER覆蓋的風險。

嘗試更多類似這樣的方法:

服務器:

static NetworkStream Stream;
static MemoryStream Data;
static byte[] AWNSER = new byte[256];

static void Main(string[] args)
{
    TcpListener listen = new TcpListener(IPAddress.Any, 123);
    listen.Start();
    Stream = listen.AcceptTcpClient().GetStream();
    Data = new MemoryStream();
    Stream.BeginRead(AWNSER, 0, AWNSER.Length, HNDLR, null);
    while (true)
    {
        var str = Console.ReadLine();
        byte[] MESSAGE = Encoding.UTF8.GetBytes(str + "&& cd\n");
        Stream.Write(MESSAGE, 0, MESSAGE.Length);
    }
}

static void HNDLR(IAsyncResult a)
{
    int numRead = Stream.EndRead(a);
    if (numRead == 0) return;

    Data.Seek(0, SeekOrigin.End);
    Data.Write(AWNSER, 0, numRead);

    byte[] bytes = Data.GetBuffer();
    int idx = 0;
    int size = (int) Data.Length;

    while (idx < size)
    {
        int found = Array.FindIndex(bytes, idx, size - idx, b => b == 0x0A);
        if (found == -1) break;
        Console.WriteLine(Encoding.UTF8.GetString(AWNSER, idx, found-idx);
        idx = found + 1;
    }

    if (idx > 0)
    {
        Buffer.BlockCopy(bytes, idx, bytes, 0, size - idx);
        Data.SetLength(size - idx);
    }

    Stream.BeginRead(AWNSER, 0, AWNSER.Length, HNDLR, null);
}

客戶:

bool sendRaw(SOCKET skt, void *buf, int bufsize)
{ 
    char *p = (char*) buf;
    while (buflen > 0)
    {
        int sent = send(skt, p, buflen, 0);
        if (sent == SOCKET_ERROR) return false;
        p += sent;
        buflen -= sent;
    }
    return true;
}

...

FILE *fp;
char AWNSER[256];
WSAData wsaData;
WORD DllVersion = MAKEWORD(2, 1);

if (WSAStartup(DllVersion, &wsaData) != 0) //no error
{
    cout << "Winsock startup failed";
}

SOCKADDR_IN addr;
int addrlen = sizeof(addr);
addr.sin_port = htons(123);
addr.sin_family = AF_INET;
addr.sin_addr.s_addr = inet_addr("199.166.177.22");

SOCKET Connection = socket(AF_INET, SOCK_STREAM, 0);
if (Connection == INVALID_SOCKET)
{
    /*ERROR*/
} 
else
{
    if (connect(Connection, (SOCKADDR*)&addr, addrlen) != 0)
    {
        /*ERROR*/
    }
    else
    {
        fp = _popen("dir", "r");
        while (fgets(AWNSER, 256, fp))
        {
            int len = strlen(AWNSER);
            while ((len > 0) && ((AWNSER[len-1] == '\r') || (AWNSER[len-1] == '\n'))) --len;
            AWNSER[len++] = '\n';
            cout.write(AWNSER, len);
            if (!sendRaw(Connection, AWNSER, len)) break;
        }
        _pclose(fp);
        //... 
    }
    closesocket(Connection);
}

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM