发送流程:首先定义一个缓冲池,发送数据时仅仅是将待发送的数据加入到缓冲池中,再由后台的工作线程从缓冲池中取得待发送数据进行发送。可能某些情况下在数据发送完成时需要做一些处理(比如写日志),便定义了一个发送完成监听,在数据发送完成时触发此事件。

接收流程:同样定义了一个接收缓冲池,由接收数据线程将接收到的数据加入到接收缓冲池中(由于这个SocketHelper只是为JT/T 794服务,因此分包逻辑我直接加入到了接收线程中),为防止阻塞接收数据线程,并未在接收线程中触发完成接收事件,而是移到了检查线程中触发。

package com.van.dev;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.InetSocketAddress;
import java.net.Socket;
import java.util.ArrayList;
import java.util.HashMap; import android.os.AsyncTask;
import android.util.Log; import com.van.jt794.MyBytesList; public class SocketHelper { /**
* @ClassName: SendDataModel
* @Description: TODO(发送数据实体)
* @author WANJJ
* @date 2011-11-17 下午02:24:27
*
*/ class SendDataModel {
public SendDataModel(byte[] bts, MySendListener sendEndListener,
ActiveSendData data) {
this.bts = bts;
this.data = data;
this.sendEndListener = sendEndListener;
} byte[] bts;
ActiveSendData data;
MySendListener sendEndListener; public void OnMyAction(Exception ex) {
if (sendEndListener != null)
sendEndListener.EndSend(data, ex);
}
} /**
* @ClassName: ReConnIn
* @Description: TODO(重连数据监听接口)
* @author WANJJ
* @date 2011-11-17 下午02:24:47
*
*/ public interface ReConnIn {
void ReConned();
} /**
* @ClassName: MySendListener
* @Description: TODO(发送数据监听接口)
* @author WANJJ
* @date 2011-11-17 下午02:26:21
*
*/ public interface MySendListener extends java.util.EventListener {
void EndSend(ActiveSendData sendData, Exception ex);
} /**
* @ClassName: MyReceiveListener
* @Description: TODO(接收数据监听接口)
* @author WANJJ
* @date 2011-11-17 下午02:26:42
*
*/ public interface MyReceiveListener extends java.util.EventListener {
void EndReceive(byte[] bts, Exception ex);
} /**
* @ClassName: SendThread
* @Description: TODO(发送线程)
* @author WANJJ
* @date 2011-11-17 下午02:22:40
*
*/ class SendThread extends AsyncTask<Integer, Void, Void> {
/**
* <p>
* Title: doInBackground
* </p>
* <p>
* Description:
* </p>
*
* @param params
* @return
* @see android.os.AsyncTask#doInBackground(Params[])
*/
@Override
protected Void doInBackground(Integer... params) {
int nowId = params[];
while (RunId == nowId) {
if (ConnedState > && sBuff.size() > ) {
SendDataModel send = sBuff.get();
try {
out.write(send.bts);
sBuff.remove(send);
send.OnMyAction(null);
continue;
} catch (IOException e) { e.printStackTrace();
send.OnMyAction(e);
}
}
mySleep();
}
return null;
}
} /**
* @ClassName: RecThread
* @Description: TODO(接受数据线程)
* @author WANJJ
* @date 2011-11-17 下午02:22:56
*
*/ class RecThread extends AsyncTask<Integer, Void, Void> { /**
* <p>
* Title: doInBackground
* </p>
* <p>
* Description:
* </p>
*
* @param params
* @return
* @see android.os.AsyncTask#doInBackground(Params[])
*/
@Override
protected Void doInBackground(Integer... params) {
int nowId = params[];
int dlen = ;
byte[] tmp = new byte[];
while (RunId == nowId) {
if (ConnedState > ) {
try {
dlen = in.read(tmp); if (dlen == -) {
changeState();
} else if (dlen > ) { int i = ;
boolean isend;// 当前为结束标记位
boolean flag7d;// 上一个字节是否为0x7d
MyBytesList lst; /*
* 问题描述:上传数据频率过快时,接收数据时可能断在某包数据内,这样造成了被断数据包无法解析。
* 解决方法: 1.定义一个全局变量PGps记录下上次未解析数据;
* 2.解析完后判断出现7E的是否为结束符
* ,如果为结束符则将PGps赋值为NULL,反之记录下未解析部分数据
* 3.开始解析时判断首字节是否为7E,如果为7E则不处理断包问题,反之继续解析
*/
if (PGps != null && ((dlen > && tmp[] != 0x7e)// 第一个字符不为7E
|| (dlen > && tmp[] == 0x7e && tmp[] == 0x7e) // 首字母为7E接下来一个也为7E
))// 处理接收断包问题
{
lst = PGps;
isend = false;
flag7d = PGps.get(PGps.size() - ) == 0x7d;
} else {
lst = new MyBytesList();
isend = true;
flag7d = false;
} while (i < dlen) {
if (tmp[i] == 0x7e)// 开始结束标记
{
isend = !isend;
if (isend)// 结束标记时解析数据
{
rBuff.add(lst);
} else {
// 清空暂存区
lst.clear();
}
} else if (flag7d)// 转义还原
{
if (tmp[i] == 0x01)// 0x7d 0x01 => 0x7d
lst.set(lst.size() - , (byte) 0x7d);
else if (tmp[i] == 0x02)// 0x7d 0x02 => 0x7e
lst.set(lst.size() - , (byte) 0x7e);
else// 出错
{
StringBuilder sb = new StringBuilder();
for (Byte byte1 : tmp) {
sb.append(Integer
.toHexString(byte1));
}
Log.d("HVT300", sb.toString());
break;
}
} else
lst.add(tmp[i]);
flag7d = tmp[i] == 0x7d;
i++;
}
PGps = isend ? null : lst;// 处理接收断包问题
}
} catch (IOException e) { e.printStackTrace();
}
} else {
mySleep();
}
}
return null;
} } /**
* @ClassName: checkThread
* @Description: TODO(检查线程)
* @author WANJJ
* @date 2011-11-17 下午02:23:28
*
*/ class checkThread extends AsyncTask<Integer, Void, Void> { /**
* <p>
* Title: doInBackground
* </p>
* <p>
* Description:
* </p>
*
* @param params
* @return
* @see android.os.AsyncTask#doInBackground(Params[])
*/ @Override
protected Void doInBackground(Integer... params) {
int nowId = params[];
while (RunId == nowId) {
if (ConnedState == ) {
disConn();
conn();
} else if (receiveListener != null) {
MyBytesList data = getData();
if (data != null)
receiveListener.EndReceive(data.toBytes(), null);
}
mySleep();
}
return null;
}
} /**
* @Fields receiveListener : TODO(接收数据监听)
*/
private MyReceiveListener receiveListener;
private Socket socket;
private OutputStream out;
private InputStream in;
private InetSocketAddress isa = null;
/**
* @Fields HOST : TODO(服务器地址 支持域名)
*/
private String HOST;
/**
* @Fields PORT : TODO(服务器端口)
*/
private int PORT ;
/**
* @Fields TIMEOUT : TODO(连接超时时间)
*/
public int TIMEOUT = ;
/**
* @Fields 0 未连接 <br/>
* 1 已连接初始化中<br/>
* 2 已连接初始化完成
*/
private int ConnedState = ;
/**
* @Fields RunId : TODO(线程执行ID)
*/
private int RunId = ;
/**
* @Fields AllSendData : TODO(主动发送的数据)
*/
HashMap<Integer, ActiveSendData> AllSendData = new HashMap<Integer, ActiveSendData>(); /**
* @Fields sBuff : TODO(发送缓存数据)
*/
private ArrayList<SendDataModel> sBuff = new ArrayList<SendDataModel>();
/**
* @Fields rBuff : TODO(接受缓存数据)
*/
private ArrayList<MyBytesList> rBuff = new ArrayList<MyBytesList>();
/**
* @Fields PGps : TODO(未解析数据)
*/
private MyBytesList PGps;
/**
* @Fields reConn : TODO(重连监听)
*/
ReConnIn reConn; /**
* <p>Title: </p>
* <p>Description: </p>
* @param HOST 服务器地址(支持域名)
* @param PORT 服务器端口
*/ public SocketHelper(String HOST, int PORT) {
this.HOST = HOST;
this.PORT = PORT;
} /**
* @Title: getConnedState
* @Description: TODO(获取连接状态)
*
* @return 0 未连接 <br/>
* 1 已连接初始化中<br/>
* 2 已连接初始化完成
*
* @version: v1.0.0
* @author: WANJJ
* @date: 2011-11-12
*
* Modification History: Date Author Version Description
* ---------------------------------------------------------*
* 2011-11-12 wanjj v1.0.0 修改原因
*/ public int getConnedState() {
return ConnedState;
} /**
* @Title: setReceiveListener
* @Description: TODO(设置接收数据监听)
*
* @param receiveListener
* @return:void
*
* @version: v1.0.0
* @author: WANJJ
* @date: 2011-11-17
*
* Modification History:
* Date Author Version Description
* ---------------------------------------------------------*
* 2011-11-17 wanjj v1.0.0 修改原因
*/ public void setReceiveListener(MyReceiveListener receiveListener) {
this.receiveListener = receiveListener;
} public void start() {
isa = new InetSocketAddress(HOST, PORT);
conn();
new SendThread().execute(RunId);
new RecThread().execute(RunId);
new checkThread().execute(RunId);
} public void stop() {
RunId++;
disConn();
} /**
* @Title: conn
* @Description: TODO(连接服务器)
*
* @return:void
*
* @version: v1.0.0
* @author: WANJJ
* @date: 2011-11-9
*
* Modification History: Date Author Version Description
* ---------------------------------------------------------*
* 2011-11-9 wanjj v1.0.0 修改原因
*/ void conn() {
try {
socket = null;
socket = new Socket();
socket.setReuseAddress(true);
socket.connect(isa, TIMEOUT);
in = socket.getInputStream();
out = socket.getOutputStream();
changeState();
if (reConn != null)
reConn.ReConned();
changeState();
} catch (IOException e) {
e.printStackTrace();
changeState();
}
} /**
* @Title: disConn
* @Description: TODO(关闭连接)
*
* @return:void
*
* @version: v1.0.0
* @author: WANJJ
* @date: 2011-11-9
*
* Modification History: Date Author Version Description
* ---------------------------------------------------------*
* 2011-11-9 wanjj v1.0.0 修改原因
*/ boolean disConn() {
boolean flag = true;
try {
if (socket != null) {
socket.shutdownInput();
socket.shutdownOutput();
try {
in.close();
out.close();
} catch (IOException e) { }
// 关闭socket
socket.close();
}
} catch (IOException e) {
e.printStackTrace();
flag = false;
}
changeState();
return flag;
} void changeState(int state) {
if (ConnedState != state) {
ConnedState = state;
// RunId++;
}
} /**
* @Title: beginSend
* @Description: TODO(开始异步发送数据)
*
* @param bts
* @param sendEndListener
* @param data
* @return:void
*
* @version: v1.0.0
* @author: WANJJ
* @date: 2011-11-11
*
* Modification History: Date Author Version Description
* ---------------------------------------------------------*
* 2011-11-11 wanjj v1.0.0 修改原因
*/ public void beginSend(byte[] bts, MySendListener sendEndListener,
ActiveSendData data) {
sBuff.add(new SendDataModel(bts, sendEndListener, data));
} /**
* @Title: SendData
* @Description: TODO(同步发送数据)
*
* @param bts
* @return
* @return:boolean
*
* @version: v1.0.0
* @author: WANJJ
* @date: 2011-11-11
*
* Modification History: Date Author Version Description
* ---------------------------------------------------------*
* 2011-11-11 wanjj v1.0.0 修改原因
*/ public boolean SendData(byte[] bts) {
if (ConnedState > ) {
try {
out.write(bts);
return true;
} catch (IOException e) { e.printStackTrace();
}
}
return false;
} /**
* @Title: getData
* @Description: TODO(获取第一包数据)
*
* @return
* @return:ArrayList<Byte>
*
* @version: v1.0.0
* @author: WANJJ
* @date: 2011-11-9
*
* Modification History: Date Author Version Description
* ---------------------------------------------------------*
* 2011-11-9 wanjj v1.0.0 修改原因
*/ public MyBytesList getData() {
if (rBuff.size() > ) {
MyBytesList tmp = rBuff.get();
rBuff.remove();
return tmp;
} else {
return null;
}
} /**
* @Title: mySleep
* @Description: TODO(当前执行线程暂停一段时间)
*
* @param time
* @return:void
*
* @version: v1.0.0
* @author: WANJJ
* @date: 2011-11-17
*
* Modification History:
* Date Author Version Description
* ---------------------------------------------------------*
* 2011-11-17 wanjj v1.0.0 修改原因
*/ public static void mySleep(long time) {
try {
Thread.sleep(time);
} catch (InterruptedException e) { e.printStackTrace();
}
} }

Android 一个异步SocketHelper的更多相关文章

  1. ArcGIS Runtime for Android 使用异步GP服务绘制等值线

    关于基于Android上ArcGIS Server GP服务的调用,已经有前辈给出了很好的例子: http://blog.csdn.net/esrichinacd/article/details/92 ...

  2. Android图片异步加载之Android-Universal-Image-Loader

    将近一个月没有更新博客了,由于这段时间以来准备毕业论文等各种事务缠身,一直没有时间和精力沉下来继续学习和整理一些东西.最近刚刚恢复到正轨,正好这两天看了下Android上关于图片异步加载的开源项目,就 ...

  3. Android图片异步加载之Android-Universal-Image-Loader(转)

    今天要介绍的是Github上一个使用非常广泛的图片异步加载库Android-Universal-Image-Loader,该项目的功能十分强大,可以说是我见过的目前功能最全.性能最优的图片异步加载解决 ...

  4. 一个异步任务接收两个url下载两个图片

    有两个url,一个是下载用户头像的url,一个是下载用户上传图片的url,想要用一个异步任务同时下载这两个图片. 程序的下载任务是这么执行的,先接受url参数,然后调用 imgUrls = infoP ...

  5. Android 之异步任务(AsyncTask,Handler,Message,looper)

    AsyncTask: 3个类型(Params,Progress和Result),4个步骤(onPreExecute(),doInBackground(Params…),onProgressUpdate ...

  6. (转)ArcGIS Runtime for Android 使用异步GP服务绘制等值线

    关于基于Android上ArcGIS Server GP服务的调用,已经有前辈给出了很好的例子: http://blog.csdn.net/esrichinacd/article/details/92 ...

  7. [Android] Android 用于异步加载 ContentProvider 中的内容的机制 -- Loader 机制 (LoaderManager + CursorLoader + LoaderManager.LoaderCallbacks)

    Android 用于异步加载 ContentProvider 中的内容的机制 -- Loader 机制 (LoaderManager + CursorLoader + LoaderManager.Lo ...

  8. Android:异步处理之Handler、Looper、MessageQueue之间的恩怨(三)

    前言 如果你在阅读本文之前,你不知道Handler在Android中为何物,我建议你先看看本系列的第一篇博文<Android:异步处理之Handler+Thread的应用(一)>:我们都知 ...

  9. Android:异步处理之AsyncTask的应用(二)

    前言 在上一篇文章中<Android:异步处理之Handler+Thread的应用(一)>,我们知道Android的UI主线程主要负责处理用户的按键事件.用户的触屏事件以及屏幕绘图事件等: ...

随机推荐

  1. 六台机器搭建RedisCluster分布式集群

    原文:六台机器搭建RedisCluster分布式集群 版权声明:m_nanle_xiaobudiu https://blog.csdn.net/m_nanle_xiaobudiu/article/de ...

  2. 深入理解javascript之高级定时器

    setTimeout()和setInterval()能够用来创建定时器.其主要的用法这里就不再做介绍了.这里主要介绍一下javascript的代码队列. 在javascript中没有不论什么代码是马上 ...

  3. C++对象模型——Inline Functions(第四章)

    4.5 Inline Functions 以下是Point class 的一个加法运算符的可能实现内容: class Point { friend Point operator+(const Poin ...

  4. python-安装xlrd xlwt 插件

    最近需要对比两个表格的内容,然后修改其中的某列内容.因为工作量太大,所以想通过python来实现.上网查了相关的操作,其中牵扯到两个功能模块,xlrd xlwt.这两个功能模块分别是对excel进行读 ...

  5. elasticsearch index之Translog

    跟大多数分布式系统一样,es也通过临时写入写操作来保证数据安全.因为lucene索引过程中,数据会首先据缓存在内存中直到达到一个量(文档数或是占用空间大小)才会写入到磁盘.这就会带来一个风险,如果在写 ...

  6. 2019.05.08 《Linux驱动开发入门与实战》

    第六章:字符设备 申请设备号---注册设备 1.字符设备的框架: 2.结构体,struct cdev: 3.字符设备的组成: 4.例子: 5.申请和释放设备号: 设备号和设备节点是什么关系.? 设备驱 ...

  7. useradd---创建的新的系统用户

    useradd命令   useradd命令用于Linux中创建的新的系统用户.useradd可用来建立用户帐号.帐号建好之后,再用passwd设定帐号的密码.而可用userdel删除帐号.使用user ...

  8. C# 读取指定文件夹中的全部文件,并按规则生成SQL语句!

    本实例的目的在于: 1 了解怎样遍历指定文件夹中的全部文件 2 控制台怎样输入和输出数据 代码: using System; using System.IO; namespace ToSql{ cla ...

  9. D. Dreamoon and Sets(Codeforces Round #272)

    D. Dreamoon and Sets time limit per test 1 second memory limit per test 256 megabytes input standard ...

  10. js 限制只能输入数字小数点

    function checkNum(e) { var re = /^\d+(?=\.{0,1}\d+$|$)/ if (e.value != "") { if (!re.test( ...