使用json协议传输二进制文件
2016-05-03 13:55
423 查看
前几天,项目中需要在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();
}
}
测试的时候先运行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();
发现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协议来传任何的数据了
这本来是很简单的事,因为我们知道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协议来传任何的数据了
相关文章推荐
- 解决js输出汉字乱码问题
- extjs 遍历树 根据子节点修改父节点图标
- 原生JS封装Ajax插件(同域&&jsonp跨域)
- jsp实现单刷时间显示,刷新显示时间不刷新页面
- 简单粗暴地理解js原型链--js面向对象编程
- 手机中的js事件
- js所有函数集合
- jsp
- C# litJson 使用方法
- JavaScript 模块插件程序之返回顶部插件
- JavaScript权威指南 第二章 词法结构
- mojo 接口返回键值对的json格式
- mojo 接口返回键值对的json格式
- mojo 接口返回键值对的json格式
- Web前端初步——Javascript(0)
- 加载js代码
- JS 二维数组排序
- JFreeChart画图+jsp页面显示
- fastjson生成json时Null属性不显示
- JSONP跨域处理实例