From: 011netservice@gmail.com Date: 2024-02-13, 週5 Subject: Readme-TcpIP.txt #### TCPIPServerClient.zip https://www.codeproject.com/Articles/1215517/TCPIP-Server-and-Client-example 原版為 .NET Framework 4.5 曾(2023-12-15)用 VS 2022, Windows 10, 已安裝 .NET Framework 4.5, 成功載入可執行. 後來(2024-02-13)用 VS 2022, Windows 11, 未安裝 .NET Framework 4.5, 無法載入. 錯誤點為 TCPIPServer.frmServer.cs.OnCommunications.如下 Properties.Settings.Default 已不支援 string style = String.Empty; if (iNK.Equals(INK.CLR_GREEN)) style = Properties.Settings.Default.StyleGreen; else if (iNK.Equals(INK.CLR_BLUE)) style = Properties.Settings.Default.StyleBlue; else if (iNK.Equals(INK.CLR_RED)) style = Properties.Settings.Default.StyleRed; else if (iNK.Equals(INK.CLR_PURPLE)) style = Properties.Settings.Default.StylePurple; else style = Properties.Settings.Default.StyleBlack; #### 進行中: 2024-02-13 整合進 ZLib2024.DTCPIP, .NET 7.0, 並改用 Microsoft.Extensions.Configuration 取代以上的 Properties.Settings 2024-02-19 1. 原版使用元件 WebBrowser 在 .NET 7.0 已不支援! 2. 嘗試將元件 WebBrowser 不成功: WebBrowse 元件在其他專案可正常運作, 但是放在本專案中卻會衝突. 因此, 繼續嘗試以下步驟: a. 先在原專案中, .NET Framwork 4.5 環境, 暫時將 WebBrowser 換成 multi-line textbox. ----> 資料夾 NoWebBrowser b. 確認功能正常以後, 再升級到 .NET 7.0. ----> 資料夾 Upgrade70 2024-02-26 從 Upgrade70 轉入 ZLib 整合為 a. Sample.TCPClient + Sample.TCPServer. ----> 資料夾 Sample\FormBase\TCPClient b. NoWarning, 包括 nullable 調整. ----> 資料夾 2024-03-05 Sending message with unicode string will result an exception. Unicode character incompatable to store in a MarshalAs(UnmanagedType.ByValArray...Char[] type property. Sending message with unicode string, e.g., "This string contains the unicode character Pi (Π)", will result an exception. string unicodeString = "This string contains the unicode character Pi (\u03a0)"; Console.WriteLine("Original string: {0}", unicodeString); // Original string: This string contains the unicode character Pi (Π) Maybe the Unicode character is incompatable to store in all MarshalAs(UnmanagedType.ByValArray...Char[] type property. [MarshalAs(UnmanagedType.ByValArray, SizeConst = 300)] public Char[] szStringDataA = new Char[300]; I'm not a native English speaker and it's hard to describe all the details in fluent English. #### Sample receive the result when we send the message to server. How to receive the result when we send the message to server ? answer: In the CommonClassLibs project, (Definitions.cs) define a couple of enumerations in the PACKETTYPES structure that defines a response you want from the server, like: TYPE_ServerAsk = 17, TYPE_ServerAnswer = 18 In the client project create a function to fill out a PACKET_DATA class and set the Packet_Type to your new enumeration, like so: PACKET_DATA xData = new PACKET_DATA(); xData.Packet_Type = (uint16)PACKETTYPES.TYPE_ServerAsk; and set idTo to 0. This tells the server that the message is not to be forwarded to another client. xData.idTo = 0; private void AskServerForSomeData() { PACKET_DATA xdata = new PACKET_DATA(); xdata.Packet_Type = (UInt16)PACKETTYPES.TYPE_ServerAsk; xdata.Data_Type = 0; xdata.Packet_Size = 16; xdata.maskTo = 0; xdata.idTo = 0; xdata.idFrom = 0; xdata.Data1 = 2;// Get me data code 2 byte[] byData = PACKET_FUNCTIONS.StructureToByteArray(xdata); client.SendMessage(byData);//fire this off to the HostServer } In the server project's ProcessReceivedData() function, make a case statement that catches the 'TYPE_ServerAsk'. Create a function that returns whatever data back to the client who sent that packet. The client id is set in the 'FullPacket' classes member iFromClient. case (UInt16)PACKETTYPES.TYPE_ServerAsk: ReturnDataToCallingClient(fp.iFromClient, fp.ThePacket); break; Fill in the PACKET_DATA class object with whatever data you want to send back to the client... NOTE the TYPE_ServerAnswer as the Packet_Type private void ReturnDataToCallingClient(uint32 clientID, byte[] message) { PACKET_DATA IncomingData = new PACKET_DATA(); IncomingData = (PACKET_DATA)PACKET_FUNCTIONS.ByteArrayToStructure(message, typeof(PACKET_DATA)); PACKET_DATA xData = new PACKET_DATA(); xData.Packet_Type = (uint16)PACKETTYPES.TYPE_ServerAnswer; xData.idTo = clientID; // fill in data if(IncomingData.Data1 == 2) { xData.Data1 = 500; xData.Data5 = 912300; } else if(IncomingData.Data1 == 3) { xData.Data1 = 13500; xData.Data5 = 1300; } string ExampleString = "Here is how you send a string, but make sure the string doesn't exceed the size of the PACKET_DATA's 'szStringDataB' field. In this case szStringDataB length is defined as 300 characters long"; ExampleString.CopyTo(0, xdata.szStringDataB, 0, ExampleString.Length); // Sen the packet to the client byte[] byData = PACKET_FUNCTIONS.StructureToByteArray(xdata); svr.SendMessage(byData); } Back in the client project, in the ProcessRecievedServerData() function, create a case statement that catches the servers 'TYPE_ServerAnswer' packet. case (UInt16)PACKETTYPES.TYPE_ServerAnswer: { DataFromTheServer(fp.ThePacket); } break; private void DataFromTheServer(byte[] message) { PACKET_DATA IncomingData = new PACKET_DATA(); IncomingData = (PACKET_DATA)PACKET_FUNCTIONS.ByteArrayToStructure(message, typeof(PACKET_DATA)); Console.WriteLine($"Data1: {IncomingData.Data1}"); Console.WriteLine($"Data5: {IncomingData.Data5}"); string GetStringFromPacket = new string(IncomingData.szStringDataB).TrimEnd('\0'); Console.WriteLine($"szStringDataB: {GetStringFromPacket}"); } #### Stange characters with connection from Android (Xamarin) Many thanks for this useful code. I am using the code on a server (computer) / client (Android Xamarin). I can connect but with every response from Android, in some strings, unwanted characters are added. At the first connection i send "TestName" as machine name, on the server i get "TestNamefÿÿÿÿÿÿÿÿÿÿÿÿÿÿÿÿv" On the second connection, and for subsequent ones, i receive "TestName1.51" Where could the problem be? reply: Hmmm, I wonder if you have to initialize the 1024 byte packet to '0'. Unfortunately same result. A little clarification: if I disconnect and then reconnect, the answer becomes "TestName1.59" where "1.59" is the final part of the IP address 192.168.1.59 of the server. It appears that the xdata.szStringDataA variable is not being cleaned (very strange). I think the problem is the fact that asynchronism on Xamarin has to be handled differently. Or maybe it's in the method Marshal.Copy, or Marshal.FreeHGlobal(buffer) that does not clean the memory. It is a pity not to be able to use this beautiful code reply: I don't have a lot of experience with Android Xamarin but Im sure there is a way of handling that. On the server when you initialize the PACKET_DATA class... PACKET_DATA xdata = new PACKET_DATA(); Maybe try: Array.Clear(xdata, 0, xdata.Length); Unfortunately it didn't work, everything is set up as the TCPIPClient and each variable is initialized correctly. I think the problem could be converting from packet to byte with the marshal method. Someone had problems with data types. I want to try converting with a different method. If I find the solution I come to communicate it. Thanks for the help. Okay, maybe the problem is in the Marshal.Copy function. I solved it, if anyone needs it: I put these two functions in PACKET_FUNCTIONS in addition to the other two. From Xamarin I use the new functions while for the windows client I use the existing functions. The first tests give the desired results. Thanks again for the support: public static byte[] GetBytes(T str) { int size = Marshal.SizeOf(str); byte[] arr = new byte[size]; GCHandle h = default(GCHandle); try { h = GCHandle.Alloc(arr, GCHandleType.Pinned); Marshal.StructureToPtr(str, h.AddrOfPinnedObject(), false); } finally { if (h.IsAllocated) { h.Free(); } } return arr; } public static T FromBytes(byte[] arr) where T : struct { T str = default(T); GCHandle h = default(GCHandle); try { h = GCHandle.Alloc(arr, GCHandleType.Pinned); str = Marshal.PtrToStructure(h.AddrOfPinnedObject()); } finally { if (h.IsAllocated) { h.Free(); } } return str; }