Android 一个异步SocketHelper
发送流程:首先定义一个缓冲池,发送数据时仅仅是将待发送的数据加入到缓冲池中,再由后台的工作线程从缓冲池中取得待发送数据进行发送。可能某些情况下在数据发送完成时需要做一些处理(比如写日志),便定义了一个发送完成监听,在数据发送完成时触发此事件。
接收流程:同样定义了一个接收缓冲池,由接收数据线程将接收到的数据加入到接收缓冲池中(由于这个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的更多相关文章
- ArcGIS Runtime for Android 使用异步GP服务绘制等值线
关于基于Android上ArcGIS Server GP服务的调用,已经有前辈给出了很好的例子: http://blog.csdn.net/esrichinacd/article/details/92 ...
- Android图片异步加载之Android-Universal-Image-Loader
将近一个月没有更新博客了,由于这段时间以来准备毕业论文等各种事务缠身,一直没有时间和精力沉下来继续学习和整理一些东西.最近刚刚恢复到正轨,正好这两天看了下Android上关于图片异步加载的开源项目,就 ...
- Android图片异步加载之Android-Universal-Image-Loader(转)
今天要介绍的是Github上一个使用非常广泛的图片异步加载库Android-Universal-Image-Loader,该项目的功能十分强大,可以说是我见过的目前功能最全.性能最优的图片异步加载解决 ...
- 一个异步任务接收两个url下载两个图片
有两个url,一个是下载用户头像的url,一个是下载用户上传图片的url,想要用一个异步任务同时下载这两个图片. 程序的下载任务是这么执行的,先接受url参数,然后调用 imgUrls = infoP ...
- Android 之异步任务(AsyncTask,Handler,Message,looper)
AsyncTask: 3个类型(Params,Progress和Result),4个步骤(onPreExecute(),doInBackground(Params…),onProgressUpdate ...
- (转)ArcGIS Runtime for Android 使用异步GP服务绘制等值线
关于基于Android上ArcGIS Server GP服务的调用,已经有前辈给出了很好的例子: http://blog.csdn.net/esrichinacd/article/details/92 ...
- [Android] Android 用于异步加载 ContentProvider 中的内容的机制 -- Loader 机制 (LoaderManager + CursorLoader + LoaderManager.LoaderCallbacks)
Android 用于异步加载 ContentProvider 中的内容的机制 -- Loader 机制 (LoaderManager + CursorLoader + LoaderManager.Lo ...
- Android:异步处理之Handler、Looper、MessageQueue之间的恩怨(三)
前言 如果你在阅读本文之前,你不知道Handler在Android中为何物,我建议你先看看本系列的第一篇博文<Android:异步处理之Handler+Thread的应用(一)>:我们都知 ...
- Android:异步处理之AsyncTask的应用(二)
前言 在上一篇文章中<Android:异步处理之Handler+Thread的应用(一)>,我们知道Android的UI主线程主要负责处理用户的按键事件.用户的触屏事件以及屏幕绘图事件等: ...
随机推荐
- 洛谷 P1581 A+B Problem(升级版)
P1581 A+B Problem(升级版) 题目背景 小明这在写作业,其中有一道A+B Problem ,他想啊想啊想,就是想不出来,于是就找到了会编程的你...... 题目描述 这里的A+B是很奇 ...
- Looksery Cup 2015
A题水 C题 博弈论,如果不是CF有WA具体哪个点错了和数据的话,AC要难许多. 如果D的操作数大于奇数个数直接Win 偶数个数大于等于n – k 时,S直接Win 偶数个数小于n – k时,若D操作 ...
- 企业实战之部署Solarwinds Network八部众
企业实战之部署Solarwinds Network 网管系统八部众 Orion Network Performance Monitor是全面的带宽性能监控和故障管理软件,能监控并收集来自路由器.交换机 ...
- Vue中独立组件之间数据交互
独立组件之间数据交互:通过自定义事件 组件A中的[数据],传递给组件B 1.创建组件A,组件B 2.组件B在实例创建完成时就开始监听事件[等待接收数据]:钩子 3.组件A中触发事件,发送数据 注意:接 ...
- Java(标识符,关键字,注释,常量,变量)
标识符 在java程序中有些名字是可以自定义的,那么这些自定义的名字我们就称作为自定义的标识符. 标识符要注意的细节: 标识符的组成元素是由字母(a-zA-Z).数字(0-9).下划线(_).美元符号 ...
- LuoguP4012 深海机器人问题(费用流)
题目描述 深海资源考察探险队的潜艇将到达深海的海底进行科学考察. 潜艇内有多个深海机器人.潜艇到达深海海底后,深海机器人将离开潜艇向预定目标移动. 深海机器人在移动中还必须沿途采集海底生物标本.沿途生 ...
- Wget使用
http://www.tuicool.com/articles/A7BRny wget / curl 是两个比较方便的测试http功能的命令行工具,大多数情况下,测试http功能主要是查看请求响应 头 ...
- D3.js加载csv和json数据
1.加载数据的基本命令 D3提供了方法可以对不同的数据类型进行加载,比如d3.text(), d3.xml(), d3.json(), d3.csv(), 和d3.html(). <!DOCTY ...
- ip地址个数的计算
一个IP地址,却关联太多的知识 二进制与 8 比特 电脑中显示出来的数字是 10 进制的,键盘的每一个键都由一个 8 位的二进制编码,所以 1 字节等于 8 比特.对数字而言,1 的二进制是 0000 ...
- regular-第一课(正则表达式基础)
之前一直听说正则表达式,尤其是在学习java的时候,遇到了不少关于正则表达式的用法.例如一个输入框,你可以使用正则表达式限制输入的内容.当然,在android以后,正则表达式就几乎没有怎么用了.不过呢 ...