2013-02-12 101 views
5

更具體地說,如果一臺計算機有一個服務器(一個java.net.ServerSocket實例),我可以使用C#System.Net.Sockets.Socket實例連接到它嗎?我可以使用套接字在Java和C#之間進行通信嗎?

+9

是。他們歸結爲讀取字節,這是語言中立(減去有趣的警告,即Java沒有像C#那樣的無符號'字節')。 – pickypg 2013-02-12 02:28:40

+1

是的,看看這個答案(由藍色基因):http://stackoverflow.com/questions/5999180/sockets-send-strings-from-java-to-c – damix911 2013-02-12 02:31:29

+0

@ damix911你可以鏈接目錄到答案,例如[Blue Gene's answer](http://stackoverflow.com/a/6001758/438992)。 – 2013-02-12 02:56:34

回答

23

主要問題是您需要非常小心的發送和接收數據的編碼。這是一對一起工作的程序。 C#客戶端首先發送一個字符串作爲整數發送它的長度,然後發送字符串本身的字節。 Java服務器讀取長度,然後讀取消息並將輸出打印到控制檯。然後編寫一個echo消息,計算其長度,提取字節並將其發送回C#客戶端。客戶端讀取長度,消息並打印輸出。應該有一種方法可以避免所有的按位元素,但老實說我對這些東西有些生疏,特別是在Java方面。

Java服務器:

import java.io.IOException; 
import java.io.InputStream; 
import java.io.OutputStream; 
import java.net.ServerSocket; 
import java.net.Socket; 

public class JavaSocket { 

    public static void main(String[] args) throws IOException { 

     ServerSocket serverSocket = new ServerSocket(4343, 10); 
     Socket socket = serverSocket.accept(); 
     InputStream is = socket.getInputStream(); 
     OutputStream os = socket.getOutputStream(); 

     // Receiving 
     byte[] lenBytes = new byte[4]; 
     is.read(lenBytes, 0, 4); 
     int len = (((lenBytes[3] & 0xff) << 24) | ((lenBytes[2] & 0xff) << 16) | 
        ((lenBytes[1] & 0xff) << 8) | (lenBytes[0] & 0xff)); 
     byte[] receivedBytes = new byte[len]; 
     is.read(receivedBytes, 0, len); 
     String received = new String(receivedBytes, 0, len); 

     System.out.println("Server received: " + received); 

     // Sending 
     String toSend = "Echo: " + received; 
     byte[] toSendBytes = toSend.getBytes(); 
     int toSendLen = toSendBytes.length; 
     byte[] toSendLenBytes = new byte[4]; 
     toSendLenBytes[0] = (byte)(toSendLen & 0xff); 
     toSendLenBytes[1] = (byte)((toSendLen >> 8) & 0xff); 
     toSendLenBytes[2] = (byte)((toSendLen >> 16) & 0xff); 
     toSendLenBytes[3] = (byte)((toSendLen >> 24) & 0xff); 
     os.write(toSendLenBytes); 
     os.write(toSendBytes); 

     socket.close(); 
     serverSocket.close(); 
    } 
} 

C#客戶:

using System; 
using System.Net; 
using System.Net.Sockets; 

namespace CSharpSocket 
{ 
    class MainClass 
    { 
     public static void Main (string[] args) 
     { 
      string toSend = "Hello!"; 

      IPEndPoint serverAddress = new IPEndPoint(IPAddress.Parse("192.168.0.6"), 4343); 

      Socket clientSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp); 
      clientSocket.Connect(serverAddress); 

      // Sending 
      int toSendLen = System.Text.Encoding.ASCII.GetByteCount(toSend); 
      byte[] toSendBytes = System.Text.Encoding.ASCII.GetBytes(toSend); 
      byte[] toSendLenBytes = System.BitConverter.GetBytes(toSendLen); 
      clientSocket.Send(toSendLenBytes); 
      clientSocket.Send(toSendBytes); 

      // Receiving 
      byte[] rcvLenBytes = new byte[4]; 
      clientSocket.Receive(rcvLenBytes); 
      int rcvLen = System.BitConverter.ToInt32(rcvLenBytes, 0); 
      byte[] rcvBytes = new byte[rcvLen]; 
      clientSocket.Receive(rcvBytes); 
      String rcv = System.Text.Encoding.ASCII.GetString(rcvBytes); 

      Console.WriteLine("Client received: " + rcv); 

      clientSocket.Close(); 
     } 
    } 
} 
+0

gj它的工作,節省了大量的時間,試圖找出一個簡單的故事 – azuneca 2015-09-08 08:03:47

+0

作爲一個有希望的有趣的一邊,Java確實有一個無符號數字原語,但它是16位,而不是8.我們以前最喜歡的char是一個無符號的16位號碼畢竟。這就是爲什麼[Character.toChars(int)](https://docs.oracle.com/javase/7/docs/api/java/lang/Character.html#toChars%28int%29)返回一個字符數組,由於整個UTF-16 shenanigans的東西。 – 2016-02-28 08:10:17

相關問題