当前位置:编程学习 > JAVA >>

Mina异步传输大文件的问题

打算用mina以协议的方式一次传输大文件,其实即便是一次传输mina也会给分段传送,现在我以异步的方式传送。出现的问题是,接收端接收到的内容,不是发送过来的顺序,这可怎么办啊,这样我接收到的内容根本不知道怎么处理,原因是mina给你随机分的片,我是没办法控制的,也没办法加序号,这怎么办呢? --------------------编程问答-------------------- 自己顶一下,大家帮帮忙啊! --------------------编程问答-------------------- 你是基于UDP传输么?
基于TCP协议是不可能的,除非是你的代码有问题. --------------------编程问答-------------------- 实现Mina的 MessageDecoder 接口的 decodable 方法. 这个方法就是判断是不是该去解码了, 看我的代码吧:
public class GroupUMessageDecoder implements MessageDecoder {
private static final Logger logger = org.slf4j.LoggerFactory.getLogger(MessageDecoder.class);
private Charset charset;

public GroupUMessageDecoder(Charset charset) {
this.charset = charset;
}

/**
 * 检查给定的IoBuffer是否适合解码
 */
public MessageDecoderResult decodable(IoSession session, IoBuffer in) {
// 报头长度==6
if (in.remaining() < 6) {
return MessageDecoderResult.NEED_DATA;
}

// tag正常
short tag = in.getShort();
if (tag == 1 || tag == 2) {
logger.info("请求标识符:" + tag);
} else {
logger.error("未知的解码类型....");
return MessageDecoderResult.NOT_OK;
}

// 真实数据长度
int len = in.getInt();
int remainLen = in.remaining(); 
logger.info("dataLength:{}, remainLength:{}", len, remainLen);
if (remainLen < len) {
return MessageDecoderResult.NEED_DATA;
}
return MessageDecoderResult.OK;
}

public MessageDecoderResult decode(IoSession session, IoBuffer in, ProtocolDecoderOutput out) throws Exception {
logger.info("解码:" + in.toString());

CharsetDecoder decoder = charset.newDecoder();
BaseMessage message = null;
short tag = in.getShort(); // tag
int len = in.getInt(); // length
logger.info("解码数据长度:" + len);

byte[] temp = new byte[len];
in.get(temp); // 数据区

// ===============解析数据做准备======================
IoBuffer buf = IoBuffer.allocate(400).setAutoExpand(true);
buf.put(temp);
buf.flip(); // 为获取基本数据区长度做准备

if (tag == 1) { // 简单类型Message
SimpleMessageResponse resp = new SimpleMessageResponse();
String txtMessage = "";
if (len > 0) {
txtMessage = buf.getString(len, decoder);
}
resp.setTxtMessage(txtMessage);
message = resp;


else if (tag == 2) {
TxtImageMessageRequest resp = new TxtImageMessageRequest();
//数据部分  
// 1, 文字长度描述(32bit) + 文字内容
int txtLen = buf.getInt();//文字的长度
String txtMessage = "";
if (len > 0) {
txtMessage = buf.getString(txtLen, decoder);
}
resp.setTxtMessage(txtMessage);
// 2, 图片数量
int imageNum = buf.getInt();
logger.info("received TxtMessage length:{}, message:{}, imageNum:" + imageNum, txtLen, txtMessage);
// 2, 图片部分: 图片名称长度描述 图片长度. 图片在文字内容的位置(32bit) + 图片长度描述(32bit) + 图片数据
while (imageNum-- > 0) {
int imageNameLength = buf.getInt();//图片的名称的长度
if (imageNameLength <= 0 )
break;
String imageName = buf.getString(imageNameLength, decoder);
if (imageName == null || "".equals(imageName))
break;
logger.info("imageName:{}" , imageName);
int imagePosition = buf.getInt();//图片的位置
int imageLength = buf.getInt();//图片的长度
if (imageLength <= 0)
break;
byte[] imageData = new byte[imageLength];
buf.get(imageData);

MessageImageInfo ii = new MessageImageInfo();
ii.setImageName(imageName);
ii.setImagePositionInTxt(imagePosition);
ii.setImageData(imageData);
resp.addImageInfo(ii);
message = resp;
break;
}

}


out.write(message);
// ================解码成功=========================
return MessageDecoderResult.OK;
}

public void finishDecode(IoSession session, ProtocolDecoderOutput out)
throws Exception {
}
}



public class GroupUMessageEncoder implements MessageEncoder<BaseMessage> {

private static final Logger log = org.slf4j.LoggerFactory.getLogger(GroupUMessageEncoder.class);
private Charset charset;

public GroupUMessageEncoder(Charset charset) {
this.charset = charset;
}

public void encode(IoSession session, BaseMessage message,
ProtocolEncoderOutput out) throws Exception {

IoBuffer buf = IoBuffer.allocate(400).setAutoExpand(true);

// ===========SimpleMessageRequest 编码数据区===============
if (message instanceof SimpleMessageRequest) {
SimpleMessageRequest req = (SimpleMessageRequest) message;
buf.putShort((short) req.getTag());
int dataLength = req.getLen(charset);
buf.putInt(dataLength);
log.info("SimpleMessageRequest dataLength:" + dataLength);
buf.putString(req.getTxtMessage(), charset.newEncoder());
log.info("SimpleMessageRequest 编码完毕:" + buf.toString());


// ===========TxtImageMessageRequest 编码数据区===============
if (message instanceof TxtImageMessageRequest) {
TxtImageMessageRequest req = (TxtImageMessageRequest) message;
buf.putShort((short) req.getTag());
int dataLength = req.getLen(charset);
buf.putInt(dataLength);
System.out.println("dataLength:" + dataLength);

//数据部分  
// 1, 文字长度描述(32bit) + 文字内容
int txtLength = req.getTxtMessage().getBytes(charset).length;
buf.putInt(txtLength);
buf.putString(req.getTxtMessage(), charset.newEncoder());
// 2, 图片数量
buf.putInt(req.getImageNum());
// 3, 图片部分: 图片名称长度描述 图片长度. 图片在文字内容的位置(32bit) + 图片长度描述(32bit) + 图片数据
if (req.getImageList() != null && req.getImageList().size() > 0) {
for (MessageImageInfo mii : req.getImageList()) {
if (mii.getImageData() == null || mii.getImageData().length == 0){
log.warn("TxtImageMessageRequest Encode: image no data");
continue;
}
int imageNameLength = mii.getImageName().getBytes(charset).length;
buf.putInt(imageNameLength); //图片名称长度
buf.putString(mii.getImageName(), charset.newEncoder()); //图片名称

buf.putInt(mii.getImagePositionInTxt()); //图片位置
buf.putInt(mii.getImageData().length); //图片数据长度
buf.put(mii.getImageData()); //图片数据
}
}
log.info("TxtImageMessageRequest编码完毕:" + buf.toString());


buf.flip();
out.write(buf);
}
}

--------------------编程问答-------------------- TCP的数据传输会保证数据包的顺序正确

是不是解码的时候出现错误了? --------------------编程问答-------------------- TCP的数据传输会保证数据包的顺序正确这是没错的.
解码的时候, 你要确定你收到的数据是你Client发送的全部部分. 如果只是部分, 你就没法解码了. 比如发送图片, 接收一半就去解码成一张图片, 肯定就是个烂图.


   当然, 如果是传输大文件, 就不必要全部接收了在写磁盘去, 可以边接收,边写, 就不用占用太多内容.   --------------------编程问答-------------------- 谢谢大家,我现在就是传输打文件,边读边写磁盘,我的程序类似于kk_eclipse给的代码,但是我的文件很大是不能一次读入到内存中再发送的,我是边读边发,收是边收边写入磁盘,但是会出现偶尔服务器收不全数据库就死掉的现象,这是怎么回事呢? --------------------编程问答-------------------- “但是我的文件很大是不能一次读入到内存中再发送的,我是边读边发,收是边收边写入磁盘“,请问你这些是怎么实现的? --------------------编程问答-------------------- --------------------编程问答-------------------- 文件如何分段,排序,这得程序控制了 --------------------编程问答-------------------- 文件如何分段,是得用程序来控制啊。 --------------------编程问答-------------------- 谢谢大家的回答,我用mina发大文件就是读一点发一点的,就会出问题 --------------------编程问答-------------------- 学习了 --------------------编程问答-------------------- 楼主,现在问题解决了吗? 我现在也遇到这样大文件传输的问题了,请教一下! --------------------编程问答-------------------- kk_eclipse  写的很好,学习了 --------------------编程问答-------------------- 楼主可以发个mina的文件传输的例子给我吗?急求啊。可以发我邮箱369179509@qq.com吗
补充:Java ,  Java SE
CopyRight © 2012 站长网 编程知识问答 www.zzzyk.com All Rights Reserved
部份技术文章来自网络,