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主线程主要负责处理用户的按键事件.用户的触屏事件以及屏幕绘图事件等: ...
随机推荐
- php,javascript设置和读取cookie
1.php设置cookie setcookie($name,$value,expire,path,domain,secrue); //$name:指的是cookie的名字:$value指的是cooki ...
- 【Henu ACM Round #13 A】 Hulk
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 模拟. [代码] #include <bits/stdc++.h> using namespace std; int m ...
- FZU 1968 Twinkling lights III
Twinkling lights III Time Limit: 8000ms Memory Limit: 131072KB This problem will be judged on FZU. O ...
- python调用Java代码,完毕JBPM工作流application
1.缘由 有一庞大Python django webproject,要引入工作流引擎,像OA一样.方便的流程控制与管理.Python或django关于工作流的开源插件,稀少,并且弱爆了,终于选用jav ...
- vue ---- 实现手机端(左滑 删除。右划 正常)
touchstart: // 手指放到屏幕上的时候触发 touchmove: // 手指在屏幕上移动的时候触发 touchend: // 手指从屏幕上拿起的时候触发 touchcancel: // 系 ...
- 深入理解Android(3)——Eclipse集成javah和NDK-Builder
在上一篇文章中我们使用了javah工具来生成了native java文件所对应的C++头文件,但是这样生成比较麻烦,我们这一篇来介绍如何在eclipse中集成javah和NDK-Builder. 一. ...
- IOS 一句代码搞定启动引导页
前言引导页,一个酷炫的页面,自从微博用了之后一下就火起来了,对于现在来说一个app如果没有引导页似乎总显那么不接地气,那么为了让我们的app也“高大上”一次,我写了一个demo来实现启动引导页的实现, ...
- JavaScript--数据结构与算法之图
图和图的算法:图的定义:由边的集合及顶点的集合组成. 例如地图,每个城镇是顶点,道路是边,由顶点对来定义(城镇1,城镇2)简称(v1,v2)顶点也有权重——成本.基本概念: 有向图:图的顶点对是有序的 ...
- GetInvocationList 委托链表
最近发现C#程序初始化时在构造函数中,偶尔出现事件注册不成功.后查资料发现有GetInvocationList 这么一个获取类中的委托链表的函数, 使用方法如下: 1.在需委托的类(Class1)中增 ...
- ZJU 2425 Inversion
Inversion Time Limit: 2000ms Memory Limit: 65536KB This problem will be judged on ZJU. Original ID: ...