首 页 网络编程
网页制作 图形图象 操作系统 冲浪宝典
软件教学 认证考试

网络安全 网络办公 行业资讯 评测对比
您当前位置:站长天空 -> 网络编程-> Visual Basic教程
Java数据报之失序和丢包-JSP教程,Java技巧及代码
作者:网友供稿 点击:18
推荐
西部数码-全国虚拟主机10强!20余项虚拟主机管理功能,全国领先!第6代双线路虚拟主机,南北访问畅通无阻!可在线rar解压,自动数据恢复设置虚拟目录等.免费赠送访问统计,企业邮局.Cn域名注册10元/年,自助建站480元起,免费试用7天,满意再付款!P4主机租用799元/月.月付免压金
站内搜索
文章页数:[1] 
 

习惯了tcp编程,认为udp可以包办这些问题是错误的。一个udp应用程序要承担可靠性方面的全部工作,包括报文的丢失、重复、时延、乱序以及连接失效等问题。

通常我们在可靠性好,传输时延小的局域网上开发测试,一些问题不容易暴露,但在大型互联网上却会出现错误。

udp协议把递送的可靠性责任推到了上层即应用层,下面简单编写了几个类来专门处理两个问题:乱序和丢包。

四个类:datapacket 类,packetheader类,packetbody类 ,dataentry类,位于同一个文件datapacket .java中。

datapacket 类相当于一个门面模式,提供给外部使用,通信数据也在这个类中处理。

package com.skysoft.pcks;

import java.io.*;
import java.net.*;
import java.util.*;

public class  datapacket {
  inputstream is;
  outputstream os;
  packetheader header;
  packetbody body;
  arraylist al;
  public static final int dataswapsize = 64532;

  /**
   * 在接收数据报使用
   */
  public datapacket() {
    header = new packetheader();
    body = new packetbody();
    al = new arraylist();
  }
  /**
   * 在发送数据报时使用,它调用报文分割操作.
   * @param file string  硬盘文件
   */
  public datapacket(string file) {
    this();
    try {
      is = new fileinputstream(file);
      header.calcheaderinfo(is.available());
      this.madebody();
      is.close();
      //this.gereratedata();
    }
    catch (filenotfoundexception ex) {
      ex.printstacktrace();
    }
    catch (ioexception ex1) {
      ex1.printstacktrace();
    }
  }
  /**
   * 在发送数据报时使用,它调用报文分割操作.
   * @param url url url地址
   */
  public datapacket(url url) {
    this();
    try {
      //is = url.openstream();
      urlconnection conn=url.openconnection();
      is=conn.getinputstream();
      int total=conn.getcontentlength();
      header.calcheaderinfo(total);
      this.madebody();
      //system.out.println(total+":"+total);
      is.close();
    }
    catch (ioexception ex) {
      ex.printstacktrace();
    }
  }
  /**
   * 为发送构造分组,使用packageheader处理了报头格式,并为分组编序号.
   */
  private void madebody() {
    al.clear();
    byte[] buffer;
    dataentry de;
    for (int i = 0; i < header.fragmentcounter; i++) {
      try {
        bytearrayoutputstream bos = new bytearrayoutputstream();
        //is.skip(i * body.body_buffer_size);
        header.arragesort(i);
        de = new dataentry(packetbody.body_buffer_size);
        de.setsn(i);
        de.setstreamsize(header.getstreamsize());
        de.setfragmentcounter(header.getfragmentcounter());
        if (header.iswtailfragment(i)) {
          buffer = new byte[header.getminfragment()];
          is.read(buffer, 0, buffer.length);
          header.setactbytesize(header.getminfragment());
          de.setactbytesize(header.getminfragment());
        }
        else {
          buffer = new byte[body.body_buffer_size];
          is.read(buffer, 0, buffer.length);
        }
        //system.out.println("length-------"+i+" "+body.getbody().length+" "+header.getminfragment());
        body.setbody(buffer);
        //system.out.println("length:" + i + " " + header.tostring());
        bos.write(header.getbyte(), 0, header.header_buffer_size);
        bos.write(body.getbody(), 0, body.getbody().length);
        de.setbytes(bos.tobytearray());
        al.add(de);
      }
      catch (ioexception ex) {
        ex.printstacktrace();
      }
    }
  }
  /**
   * 为发送构造分组,没有考虑报头格式,也没有为分组编序号.
   */
  private void madebody1() {
    al.clear();
    for (int i = 0; i < header.fragmentcounter; i++) {
      try {
        if (header.iswtailfragment(i))
          is.read(body.getbody(), i * body.body_buffer_size,
                  header.getminfragment());
        else
          is.read(body.getbody(), i * body.body_buffer_size,
                  body.body_buffer_size);
        bytearrayoutputstream bos = new bytearrayoutputstream();
        bos.write(header.getbyte(), 0, header.header_buffer_size);
        bos.write(body.getbody(), header.header_buffer_size,
                  body.getbody().length);
        al.add(bos);
      }
      catch (ioexception ex) {
        ex.printstacktrace();
      }
    }
  }
  /**
   * 在接收到报文后,对此报文执行组装,并处理报文丢失和乱序情况.
   * @param b1 byte[]
   */
  public void add(byte[] b1) {
    byte[] buffer = (byte[]) b1.clone();
    handlertext(buffer);
    dataentry de = new dataentry(buffer, header.getactbytesize());
    de.setsn(header.getsn());
    de.setstreamsize(header.getstreamsize());
    de.setfragmentcounter(header.getfragmentcounter());
    al.add(de);
  }
  private void handlertext(byte[] buffer) {
    bytearrayoutputstream baos = new bytearrayoutputstream();
    baos.write(buffer, 0, header.header_buffer_size);
    byte[] b=new byte[header.header_buffer_size];
    system.arraycopy(buffer,0,b,0,b.length);
    bytearrayinputstream bais = new bytearrayinputstream(baos.tobytearray());
    inputstreamreader isr = new inputstreamreader(bais);
    bufferedreader br = new bufferedreader(isr);
    try {
      header = new packetheader(br.readline());
    }
    catch (exception ex) {
      ex.printstacktrace();
    }
  }
 
  private string calfilesize(int size) {
    return size / 1024 + "k";
  }

  public arraylist getdatapackets() {
    return al;
  }
/**
 * 是否接收完毕,通过序号是否等于最大段数来判断,这也许有问题,比如,正好是最后一个段丢失了,这样
 * 这个包整个就丢失了.
 * @return
 */
  public boolean isfull() {
    return this.header.getsn() == this.header.getfragmentcounter() - 1 ? true : false;
  }
/**
 * 判断是否只有一个段.
 * @return
 */
  public boolean iszero() {
    return this.header.getsn() == 0 ? true : false;
  }
/**
 * 该函数执行报文组装,不考虑丢失的报文.
 * @return
 */
  private bytearrayoutputstream fetchdatapackets() {
    bytearrayoutputstream bos = new bytearrayoutputstream();
    byte[] buffer = null;
    dataentry de;
    for (int i = 0; i < al.size(); i++) {
      try {
        de = this.getsndata(i);
        buffer = de.getbyte();
        if (header.getstreamsize() == de.getstreamsize()) {
          bos.write(de.getbyte(), header.header_buffer_size, de.getactbytesize());
          system.out.println(de.tostring() + " -- fetchdatapackets");
        }
      }
      catch (exception ex) {
        ex.printstacktrace();
      }
    }
    return bos;
  }

  /**
   * 该函数执行报文组装,对于丢失的报文,写入空报文.
   * @return bytearrayoutputstream
   */
  private bytearrayoutputstream fetchdatapackets_sn() {
    bytearrayoutputstream bos = new bytearrayoutputstream();
    byte[] buffer;
    dataentry de;
    for (int i = 0; i < header.getfragmentcounter(); i++) {
      try {
        de = this.getsndata(i);
        if (de == null) {
          de = seachdedata(i);
        }
        buffer = de.getbyte();
        //system.out.println(de.getsn() + ":" + i);
        //handlertext(buffer);
        //bos.write(buffer, header.header_buffer_size,
        //          buffer.length - header.header_buffer_size);
        if (header.getstreamsize() == de.getstreamsize()) {
          bos.write(de.getbyte(), header.header_buffer_size,
                    de.getactbytesize());
          //system.out.println(de.tostring());
        }
      }
      catch (exception ex) {
        ex.printstacktrace();
      }
    }
    return bos;
  }

  /**
   * 对缓冲的数据包进行排序处理,即按顺序提取同一帧的数据,如果没有找到该序号的帧,则返回空值.
   * @param sn int 要找的帧序号.
   * @return dataentry
   */
  private dataentry getsndata(int sn) {
    dataentry de = null;
    for (int i = 0; i < al.size(); i++) {
      de = (dataentry) al.get(i);
      if (header.getstreamsize() == de.getstreamsize()) {
        if (sn == de.getsn())
          break;
        else
          de = null;
      }
    }
    return de;
  }

  /**
   * 按序号开始向前或者是向后寻找最近的帧片段,日后可以增加请求重发功能,通过开一个通信连接.
   * @param sn int
   * @return dataentry
   */
  private dataentry seachdedata(int sn) {
    dataentry de = null;
    int initvalue, minvalue = 10000;
    dataentry back, fore = null;
    for (int i = 0; i < al.size(); i++) {
      de = (dataentry) al.get(i);
      if (header.getstreamsize() == de.getstreamsize()) {
        initvalue = math.abs(de.getsn() - sn);
        if (de.getfragmentcounter() != de.getsn() && initvalue < minvalue) {
          minvalue = initvalue;
          fore = de;
        }
      }
    }
    return fore;
  }

  /**
   * 除去最后一帧外,随机抽取一帧.
   * @return dataentry
   */
  private dataentry seachdedata() {
    dataentry de = null;
    for (int i = 0; i < al.size(); i++) {
      de = (dataentry) al.get(i);
      system.out.println("sky ::::" + de.getfragmentcounter() + ":" + de.getsn() +
                         ":" + i);
      if (header.getstreamsize() == de.getstreamsize()) {
        if (de.getfragmentcounter() != de.getsn()) {
          break;
        }
      }
    }
    return de;
  }
  /**
   * 生成组装完的结果数据.因为用图像来做测试,所以令其返回图像.
   * @return image
   */
  public java.awt.image gereratedata() {
     bytearrayinputstream bis;
     java.awt.image.bufferedimage bimage = null;
     try {
       byte[] b = fetchdatapackets_sn().tobytearray();
       //fetchdatapackets_old1()
       bis = new bytearrayinputstream(b);
       bimage = javax.imageio.imageio.read(bis);

     }
     catch (exception ex1) {
       ex1.printstacktrace();
     }
     return bimage;
  }

  public static void main(string args[]) {
    datapacket dp = new datapacket("e:\\nature\\14.jpg");
  }
}
/**
 * 数据实体,充当临时处理场所.
 * @author administrator
 *
 */
class dataentry {
  byte[] bytes;
  int fragmentcounter, sn, actbytesize;
  long streamsize;
  int minfragment;

  public dataentry() {

  }

  public dataentry(int size) {
    this.actbytesize = size;
  }

  public dataentry(byte[] b, int i) {
    this.bytes = b;
    this.actbytesize = i;
  }

  public byte[] getbyte() {
    return this.bytes;
  }

  public void setbytes(byte[] b) {
    this.bytes = b;
  }

  public void setstreamsize(long size) {
    this.streamsize = size;
  }

  public long getstreamsize() {
    return this.streamsize;
  }

  public int getminfragment() {
    return minfragment;
  }

  public synchronized void setsn(int i) {
    this.sn = i;
  }

  public synchronized int getsn() {
    return sn;
  }

  public synchronized int getfragmentcounter() {
    return fragmentcounter;
  }

  public synchronized void setfragmentcounter(int c) {
    this.fragmentcounter = c;
  }

  public void setactbytesize(int size) {
    actbytesize = size;
  }

  public int getactbytesize() {
    return actbytesize;
  }

  public string tostring() {
    return this.streamsize + "::" + this.fragmentcounter + "::" + this.sn +
        "::" + this.actbytesize + " recv dataentry";
  }
}
/**
 * 报头,处理报头格式
 * @author administrator
 *
 */
class packetheader implements serializable{
  public static final int header_buffer_size = 1024;
  int fragmentcounter, sn;
  int actbytesize = packetbody.body_buffer_size;
  byte[] header; //= new byte[header_buffer_size];
  long streamsize;
  int minfragment;

  public packetheader() {

  }

  public packetheader(long l) {
    this.setstreamsize(l);

  }

  public packetheader(string s) {
    string[] tm = s.split("::");
    this.setactbytesize(integer.parseint(tm[3]));
    this.setsn(integer.parseint(tm[2]));
    this.setfragmentcounter(integer.parseint(tm[1]));
    this.setstreamsize(long.parselong(tm[0]));
  }

  /**
   * 根据文件的段的顺序生成数据头.
   * @param sn 文件序列
   */
  public void arragesort(int sn) {
    this.setsn(sn);
    this.setbyte();
  }

  public void calcheaderinfo(long l) {
    this.setstreamsize(l);
    calcheaderinfo();
  }
  /**
   * 计算流要被分成的片段数量,并得出最小片段余量.
   */
  public void calcheaderinfo() {
    fragmentcounter = math.round( (float) streamsize /
                                 packetbody.body_buffer_size);
    float critical = (float) streamsize / packetbody.body_buffer_size;
    if (critical - fragmentcounter < 0.5 && critical - fragmentcounter > 0)
      fragmentcounter++;
    minfragment = (int) (streamsize % packetbody.body_buffer_size);
  }

  public byte[] getheader() {
    long it = new long(this.streamsize);
    return new byte[] {it.bytevalue()};
  }

  public byte[] getbyte() {
    return header; //this.tostring().getbytes();
  }
  /**
   * 生成报头字节,首先取得数据包头 流尺寸::段片数::段顺序::段实际尺寸 的字节形式,
   * 然后加入回车换行符号,对于1024字节中剩余的部分一律写入元素为0的字节数组.
   */
  public void setbyte() {
    bytearrayoutputstream bos = new bytearrayoutputstream();
    byte[] buffer = this.tobyte();
    try {
      bos.write(buffer);
      bos.write("\r\n".getbytes());
      bos.write(new byte[packetheader.header_buffer_size - buffer.length], 0,
                packetheader.header_buffer_size - buffer.length);
      header = bos.tobytearray();
    }
    catch (ioexception ex) {
      ex.printstacktrace();
    }
  }

  public void setstreamsize(long size) {
    this.streamsize = size;
  }

  public long getstreamsize() {
    return this.streamsize;
  }

  public int getminfragment() {
    return minfragment;
  }

  public synchronized void setsn(int i) {
    this.sn = i;
  }

  public int getsn() {
    return sn;
  }

  public int getfragmentcounter() {
    return fragmentcounter;
  }

  public synchronized void setfragmentcounter(int c) {
    this.fragmentcounter = c;
  }

  public void setactbytesize(int size) {
    actbytesize = size;
    setbyte();
  }

  public int getactbytesize() {
    return actbytesize;
  }
  /**
   * 数据包头的格式为:流尺寸::段片数::段顺序::段实际尺寸
   * 报头字节长度是可变化的,比如,可以加入流的具体信息如:流所属文件的名称,文件类型以及一些其他信息.
   * @return string
   */
  public string tostring() {
    return streamsize + "::" + this.fragmentcounter + "::" + this.getsn() +
        "::" + this.getactbytesize();
  }

  public byte[] tobyte() {
    return this.tostring().getbytes();
  }
  /**
   * 是否为尾段
   * @param i int
   * @return boolean
   */
  public boolean iswtailfragment(int i) {
    return (i == fragmentcounter - 1) ? true : false;
  }

}
/**
 * 用户数据区
 * @author administrator
 *
 */
class packetbody implements serializable{
  public static final int body_buffer_size = 63508; //65508
  byte[] body;

  public packetbody() {
  }

  public void setbody(byte[] b) {
    this.body = b;
  }

  public byte[] getbody() {
    return body;
  }
}

这个数据处理类,将在接下来使用。


文章整理:站长天空 网址:http://www.z6688.com/
以上信息与文章正文是不可分割的一部分,如果您要转载本文章,请保留以上信息,谢谢!

文章页数:[1] 


放大字体显示 缩小字体显示 打印文章 推荐给朋友
热门文章
·ASP.net中动态加载控件时一些问题的总结-ASP教程,ASP应用
·C#读写注册表-.NET教程,C#语言
·Visual Basic .NET中的异常处理简介(下)-.NET教程,VB.Net语言
·C#:文件的按行读/写及文件目录对话框的使用-.NET教程,C#语言
·ADO.Net:使用DataReader向数据库中插入数据-ASP教程,数据库相关
·列一张网恋赔偿清单(爆笑)
·如何用Photoshop画服装款式图-网页设计,Photoshop
·.NET下使用DataAdapter保存数据时,如何生成command语句及使用事务-.NET教程,数据库应用
·新型dc/dc电源控制芯片dpa426的应用
·ASP.NET 2.0 - Enter Key - Default Submit Button-.NET教程,Asp.Net开发
最新文章
·个人站长的网络赚钱两条新出路_网赚技巧
·adsense帐户最佳化纵深谈-adsense资深专员_网赚技巧
·google adsense容易被k的可能性列表_网赚技巧
·如何让程序被站长接受和产生利润_站长访谈
·马云,即成的中国互联网第4代霸主_站长访谈
·google关键词广告创建的十二招_google推广
·如何使google更快速收录你的新站_google推广
·几个颇有创意的网站推广方法_站长心得
·网络编辑:标题,如何让网民一见钟情(2)_网络编辑
·网站建设基础seo搜索引擎优化_seo网站优化
相关主题
  • Java数据报编程之广播-JSP教程,数据库相关
  • Java数据报编程之单播-JSP教程,数据库相关
  • Java数据报编程之组播-JSP教程,Java技巧及代码
  • Java数据报编程之概说-JSP教程,Java技巧及代码
  • Java数据报编程之测试程序-JSP教程,Java技巧及代码
  • 西部数码虚拟主机

    友情链接
    CNNIC 西部数码
    万网 自助建站
    虚拟主机 asp空间
    域名注册 域名
    域名申请 主页空间
    论坛空间 网站空间
    国际域名 虚拟空间
    空间租用 DDOS防火墙
    成都主机托管 四川主机托管
    主机租用 服务器租用
    网站目录 自助建站
    虚拟主机 网址大全
    软件下载
    自助链接
    虚拟主机资讯 特价虚拟主机
    版权申明:本站文章均来自网络,如有侵权,请联系我们,我们收到后立即删除,谢谢!
    关于我们:站长天空:专业提供最新的站长资讯、在线教程、虚拟主机权威评测、虚拟主机性能对比、网站制作教程,开发教程,站长工具。包括网页制作教程、冲浪宝典、编程参考、操作系统、软件教学、行业动态等。
    特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有。
    发表评论 打印  刷新     关闭