通过TCP套接字从C#服务器向Java客户端发送图像

3

我在服务器上有一张图片,我想通过socket将其发送到我的Java客户端。在C#中,我将其转换为字节数组,然后尝试通过socket发送。但是,在C#中,字节数组是无符号的,因此我尝试发送有符号的字节数组sbyte[],但是它不能通过clientSocket.Send()方法发送。在Java客户端方面,我需要将接收到的字节数组转换为一个图像对象。下面是我得到的异常跟踪信息,在Image image = reader.read(0, param)处。请帮助我解决这个问题。

Exception in thread "main" javax.imageio.IIOException: Bogus marker length
at com.sun.imageio.plugins.jpeg.JPEGImageReader.readImageHeader(Native Method)
at com.sun.imageio.plugins.jpeg.JPEGImageReader.readNativeHeader(Unknown Source)
at com.sun.imageio.plugins.jpeg.JPEGImageReader.checkTablesOnly(Unknown Source)
at com.sun.imageio.plugins.jpeg.JPEGImageReader.gotoImage(Unknown Source)
at com.sun.imageio.plugins.jpeg.JPEGImageReader.readHeader(Unknown Source)
at com.sun.imageio.plugins.jpeg.JPEGImageReader.readInternal(Unknown Source)
at com.sun.imageio.plugins.jpeg.JPEGImageReader.read(Unknown Source)
at ServerByteStreamWithoutOIS.main(ServerByteStreamWithoutOIS.java:54)

这是我的C#服务器代码:

class Program
{
static void Main(String[] args)
{
Socket sListen = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);

// 2. Fill IP
IPAddress IP = IPAddress.Parse("147.174.117.187");
IPEndPoint IPE = new IPEndPoint(IP, 20229);

// 3. binding
sListen.Bind(IPE);

// 4. Monitor
Console.WriteLine("Service is listening ...");
sListen.Listen(2);

// 5. loop to accept client connection requests
while (true)
{
Socket clientSocket;
try
{
clientSocket = sListen.Accept();
}
catch
{
throw;
}


// send the file
byte[] buffer = ReadImageFile("1.jpg");

clientSocket.Send(buffer, buffer.Length, SocketFlags.None);
clientSocket.Close();
Console.WriteLine("Send success!");
}
}


private static byte[] ReadImageFile(String img)
{
FileInfo fileInfo = new FileInfo(img);
byte[] buf = new byte[fileInfo.Length];
FileStream fs = new FileStream(img, FileMode.Open, FileAccess.Read);
fs.Read(buf, 0, buf.Length);
fs.Close();
//fileInfo.Delete ();
GC.ReRegisterForFinalize(fileInfo);
GC.ReRegisterForFinalize(fs);
return buf;
}
}
}

以下是我的Java客户端:

public class ServerByteStreamWithoutOIS {
public static void main(String[] args) throws IOException, ClassNotFoundException{
int port = 20229;
Socket sock = null; 
InetAddress addr = null;
addr = InetAddress.getByName("147.174.117.187");    
sock = new Socket(addr, port);
System.out.println("created socket!");
int count = 0;
while(true){
String line = "";
String realLine = "";
BufferedReader bReader = new BufferedReader(new InputStreamReader(sock.getInputStream()));
byte[] buffer = null;
while((line=bReader.readLine() )!=null){
realLine = realLine + line;
System.out.println(line.getBytes());
}

buffer = realLine.getBytes();

//buffer = (byte[])ois.readObject();

ByteArrayInputStream bis = new ByteArrayInputStream(buffer);
Iterator<?> readers = ImageIO.getImageReadersByFormatName("jpg");

ImageReader reader = (ImageReader) readers.next();
Object source = bis; // File or InputStream, it seems file is OK

ImageInputStream iis = ImageIO.createImageInputStream(source);
//Returns an ImageInputStream that will take its input from the given Object

reader.setInput(iis, true);
ImageReadParam param = reader.getDefaultReadParam();

Image image = reader.read(0, param);
//got an image file

BufferedImage bufferedImage = new BufferedImage(image.getWidth(null), image.getHeight(null), BufferedImage.TYPE_INT_RGB);
//bufferedImage is the RenderedImage to be written
Graphics2D g2 = bufferedImage.createGraphics();
g2.drawImage(image, null, null);
File imageFile = new File("image.bmp");
ImageIO.write(bufferedImage, "bmp", imageFile); 

System.out.println(imageFile.getPath());

//Thread.sleep(100);
System.out.println("Done saving");
}
}

}
2个回答

3
我认为错误是因为您将Java服务器接收到的字节转换为字符串表示形式。这可能会导致错误,因为jpg是二进制数据,当某些二进制数据无法转换为字符串中的字符时,会发生一些转换,当您使用getBytes()函数时会导致错误。
如果您改为使用read(byte[],int,int])函数从输入流中读取字节,我认为您应该没问题了。
http://download.oracle.com/javase/1.5.0/docs/api/java/io/InputStream.html#read(byte[], int, int)
编辑,添加一个有效的代码示例
Java拥有带符号字节的“问题”不是问题。在二进制中,它们是相同的位,因此当写入文件时,仍然写入相同的位,因为它们仍然是相同的顺序。
我编写了一个C#客户端和Java服务器的示例,我已经使其工作。我相信您会发现它有用。
服务器 - Java
import java.io.BufferedInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;

public class ImageServer {
    public static void main(String[] args) {
        try {
            ServerSocket server = new ServerSocket(8000);
            Socket accept = server.accept();
            InputStream inputStream = accept.getInputStream();
            BufferedInputStream stream = new BufferedInputStream(inputStream);
            int length = readInt(inputStream);
            byte[] buf = new byte[length];
            for (int read = 0; read < length; ) {
                read += stream.read(buf, read, buf.length - read);
            }
            stream.close();

            FileOutputStream fos = new FileOutputStream("image.png");
            fos.write(buf, 0, buf.length);
            fos.flush();
            fos.close();
        }
        catch (IOException e) {
            e.printStackTrace();
        }
    }

    private static int readInt(InputStream inputStream) throws IOException {
        byte[] buf = new byte[4];
        for (int read = 0; read < 4; ) {
            read += inputStream.read(buf, 0, 4);
        }
        return toInt(buf);
    }

    public static int toInt(byte[] b) {
        return (b[0] << 24)
                + ((b[1] & 0xFF) << 16)
                + ((b[2] & 0xFF) << 8)
                + (b[3] & 0xFF);
    }
}

客户端 - C#

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

namespace Test.SendImageClient {
    public class Program {
        public static void Main(string[] args) {
            if (args.Length == 0) {
                Console.WriteLine("usage: client imagefile");
                return;
            }
            FileStream stream = File.OpenRead(args[0]);

            Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
            socket.Connect("localhost", 8000);
            int length = IPAddress.HostToNetworkOrder((int)stream.Length);
            socket.Send(BitConverter.GetBytes(length), SocketFlags.None);

            byte[] buffer = new byte[1024];
            int read;
            while ((read = stream.Read(buffer, 0, buffer.Length)) > 0) {
                socket.Send(buffer, 0, read, SocketFlags.None);
            }
            socket.Close();
        }
    }
}

谢谢回复。我用了你的方法,加上了这段代码:byte[] buffer = new byte[3000]; inputStream.read(buffer, 0, 3000);现在我得到了异常信息 Not a jpeg file starts with 0x04 0x48。我认为这是因为 C# 发送的是无符号字节数组,而 Java 将其视为有符号字节数组。你知道我可以通过套接字将 sbyte 数组发送到 Java 的任何方式吗?谢谢。 - mojorisinify
@mojorisinify:你能验证一下是否是因为字节序的问题吗?请查看https://dev59.com/BkXRa4cB1Zd3GeqPpjPk#92983,了解如何将int转换为网络顺序的示例。如果在Visual Studio中设置断点,您可以检查C#字节数组中的第一个字节,并验证Java服务器中接收到的第一个字节。 - Patrick
顺便说一下,不要创建一个固定大小的字节数组,而是从你的C#客户端发送字节数组长度,在你的Java服务器中读取那么多字节。使用读取函数的返回值来知道已经读取了多少字节,并迭代直到你拥有所有的字节。 - Patrick
好消息是,现在使用相同的inputstream.read()方法,它可以从C#获取图像,但是图像已经损坏了。一些像素不正确。但是大部分图像是可见的。错误仍然是相同的“Not a jpeg file: starts with 0xd9 0xff”。如果是因为字节序问题,那么图像就不会被保存在Java客户端中了,对吧? - mojorisinify
@mojorisinify 当然可以保存,但正如你所说,它已经损坏了。你可以尝试验证一下是不是字节顺序(大小端)的问题,如果你检查接收到的字节与发送的字节是否具有相同的值。 - Patrick
显示剩余4条评论

1

看起来接收到的字节与JPEG规范不匹配(无法正确反序列化)。您确定服务器上的文件存在并且C# byte[]被正确填充了吗?也许在通过套接字发送之前尝试在服务器上写入文件,以确保您实际上正在读取有效的JPEG。


是的,文件存在于服务器上并且C#字节已经填充。我已经测试过它在C#中确实可以工作。问题出现在将其发送到Java客户端之后。我认为这是因为我不知道如何通过套接字发送sbyte。Java获取了未签名的字节,并认为它是一个有符号的字节数组。 - mojorisinify

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接