json传二进制文件(转)

 

前几天,项目中需要在socket中传输二进制文件. 
这本来是很简单的事,因为我们知道socket传输的就是字节流.所以非常简单. 
java的实现: 
Java代码 

import java.io.FileOutputStream;   
import java.io.InputStream;   
import java.net.ServerSocket;   
import java.net.Socket;   
  
public class FileReceive {   
  
    public static void main(String[] args) throws Exception {   
        ServerSocket serverSocket = new ServerSocket(8800);   
        System.out.println("server started!");   
        while (true) {   
            Socket socket = serverSocket.accept();   
            InputStream inputStream = socket.getInputStream();   
            FileOutputStream fileOutputStream = new FileOutputStream("d:/b.png");   
            int i;   
            while ((i = inputStream.read()) != -1) {   
                fileOutputStream.write(i);   
            }   
            fileOutputStream.flush();   
            fileOutputStream.close();   
            inputStream.close();   
            serverSocket.close();   
            break;   
        }   
    }   
  
}   
  
  
  
  
import java.io.FileInputStream;   
import java.io.OutputStream;   
import java.net.Socket;   
  
public class FileSend {   
  
    public static void main(String[] args) throws Exception {   
        // 打开文件流   
        FileInputStream fileInputStream = new FileInputStream("d:/a.png");   
        Socket socket = new Socket("127.0.0.1", 8800);   
        OutputStream outputStream = socket.getOutputStream();   
        int i;   
        // 读取并写入socket   
        while ((i = fileInputStream.read()) != -1) {   
            outputStream.write(i);   
  
        }   
        // 结束   
        outputStream.flush();   
        socket.close();   
    }   
  
}
import java.io.FileOutputStream;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;

public class FileReceive {

	public static void main(String[] args) throws Exception {
		ServerSocket serverSocket = new ServerSocket(8800);
		System.out.println("server started!");
		while (true) {
			Socket socket = serverSocket.accept();
			InputStream inputStream = socket.getInputStream();
			FileOutputStream fileOutputStream = new FileOutputStream("d:/b.png");
			int i;
			while ((i = inputStream.read()) != -1) {
				fileOutputStream.write(i);
			}
			fileOutputStream.flush();
			fileOutputStream.close();
			inputStream.close();
			serverSocket.close();
			break;
		}
	}

}




import java.io.FileInputStream;
import java.io.OutputStream;
import java.net.Socket;

public class FileSend {

	public static void main(String[] args) throws Exception {
		// 打开文件流
		FileInputStream fileInputStream = new FileInputStream("d:/a.png");
		Socket socket = new Socket("127.0.0.1", 8800);
		OutputStream outputStream = socket.getOutputStream();
		int i;
		// 读取并写入socket
		while ((i = fileInputStream.read()) != -1) {
			outputStream.write(i);

		}
		// 结束
		outputStream.flush();
		socket.close();
	}

}


测试的时候先运行FileReceive,再运行FileSend 

这样你就能看到在D:/下有一个文件是"b.png".跟原来的一模一样.代表传输成功! 

不过我们的项目有个要求,需要用json的协议来传输. 

我们都知道json是文本格式.所以我马上想到的是把文件编码成文本,再进行传输. 

所以关键就是要把一个二进制文件翻译成文本,再翻译回去.如果可以,也就代表这个方案是可行的. 

马上我就写出了第一个版本: 

Java代码 

// 读取文件   
FileInputStream fileInputStream = new FileInputStream("d:/a.png");   
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();   
int i;   
while ((i = fileInputStream.read()) != -1) {   
    byteArrayOutputStream.write(i);   
}   
fileInputStream.close();   
// 把文件存在一个字节数组中   
byte[] filea = byteArrayOutputStream.toByteArray();   
byteArrayOutputStream.close();   
  
String fileaString = new String(filea);   
System.out.println(fileaString);   
// 写入文件   
FileOutputStream fileOutputStream = new FileOutputStream("d:/b.png");   
fileOutputStream.write(fileaString.getBytes());   
fileOutputStream.flush();   
fileOutputStream.close();
// 读取文件
		FileInputStream fileInputStream = new FileInputStream("d:/a.png");
		ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
		int i;
		while ((i = fileInputStream.read()) != -1) {
			byteArrayOutputStream.write(i);
		}
		fileInputStream.close();
		// 把文件存在一个字节数组中
		byte[] filea = byteArrayOutputStream.toByteArray();
		byteArrayOutputStream.close();

		String fileaString = new String(filea);
		System.out.println(fileaString);
		// 写入文件
		FileOutputStream fileOutputStream = new FileOutputStream("d:/b.png");
		fileOutputStream.write(fileaString.getBytes());
		fileOutputStream.flush();
		fileOutputStream.close();


发现b.png根本无法打开,很是奇怪?而且大小也不一样了. 

检查代码后发现把数组转化成string的时候后面还可以加个参数charset[编码名称].

于是马上改了一下: 

....... 
String fileaString = new String(filea,"uft-8"); 
....... 
fileOutputStream.write(fileaString.getBytes("utf-8"));

结果还是不行,又改了一下: 

....... 
String fileaString = new String(filea,"ISO-8859-1"); 
....... 
fileOutputStream.write(fileaString.getBytes("ISO-8859-1"));

结果成功了. 

结合以前的编码知识,总算是想明白了. 

把字节数组转换成字符串时,如果charset为空,那就表示用你的默认编码进行转换. 

而不管gb2312和utf8都不是用单字节表示一个字符的.只要他不认识的,他就会默认转换成一个特定字符,所以这样的转换是不可逆的. 

只有像ascii这样的单字节编码的转换才是可逆的. 

明白了这个道理.我们就可以用json协议来传任何的数据了.^_^,大功告成!