这篇博客的内容是基于http://my.oschina.net/fengcunhan/blog/178155进行改造的,所以需要先看完这篇博客,然后再来看下面的内容。
Tcp消息的结束符“\r\n”,消息字段分割以“|”竖线,客户端提交数据以SS开头,服务端返回数据以RC开头。
向服务器端提交数据是类似get提交的方式 uid=2&content=hello
3、以发送心跳包的格式为例
向服务器发送:
列 |
内容 |
是否必须 |
说明 |
信息头 |
SS |
是 |
|
操作符 |
tcp.heartbeat |
是 |
服务器要执行的操作 |
内容长度 |
10 |
是 |
最后数据列的长度,为以后做数据校验预留 |
数据 |
uid:int:用户id
|
|
|
服务器返回:
列 |
内容 |
说明 |
信息头 |
RC |
|
操作符 |
tcp.heartbeat |
服务器要执行的操作 |
时间 |
2015-11-04 12:00:00 |
|
数据 |
status:int:结果状态 msg:string:服务器返回消息 |
Json数据格式 |
实例:>> SS|tcp.heartbeat|7|uid=123\r\n
<< RC|tcp.heartbeat|2015-11-04 13:30:00|{
"status": 0,
"msg": "ok",
“key”:””
}\r\n
每次数据提交中,某些参数必选,某些参数可选
某些数据提交中,key可选,key是为了确认会话(没有收到响应的key,就进行消息重发),服务器端会返回key数据
key:string:每条消息的唯一标示,只要不短时间内出现重复并且字符长度不要太长都可以,md5值,时间戳+随机数 等等,为了保证服务器端应答的唯一性
如果你需要一个未被封装的Android的raw Socket, HTTP client/server, WebSocket, and Socket.IO, AndroidAsync 正适合你。
/** * 由于移动设备的网络的复杂性,经常会出现网络断开,如果没有心跳包的检测, * 客户端只会在需要发送数据的时候才知道自己已经断线,会延误,甚至丢失服务器发送过来的数据。 * 所以需要提供心跳检测 */ public class SocketService extends Service { private static final String TAG = "SocketService"; private static final long HEART_BEAT_RATE = 30 * 1000;//目前心跳检测频率为30s private static final long RE_SEND_MSG = 5 * 1000;//重发消息间隔 public static final String HOST = "t.ing.baofeng.com"; public static final int PORT = 9099; public static final String MESSAGE_ACTION="com.storm.durian.message_ACTION"; public static final String HEART_BEAT_ACTION="com.storm.durian.heart_beat_ACTION"; public static final String RE_CONNECTION = "com.storm.durian.re_connection";//socket断了之后重新连接及做之后的操作 public Map<String,String> map = new ConcurrentHashMap<String,String>(); private ReadThread mReadThread; private LocalBroadcastManager mLocalBroadcastManager; private WeakReference<Socket> mSocket; private String socketMsgStr = ""; // For heart Beat private Handler mHandler = new Handler(); private Runnable heartBeatRunnable = new Runnable() { @Override public void run() { if (System.currentTimeMillis() - sendTime >= HEART_BEAT_RATE) { Map<String, String> m = new HashMap<String, String>(); String key = String.valueOf(System.currentTimeMillis()); m.put("key",key); boolean isSuccess = sendMsg(key,AsyncSocketUtils.spliceSendMessage(getApplicationContext(), m, UrlContainer.HEART_BEAT));//就发送一个\r\n过去 如果发送失败,就重新初始化一个socket if (!isSuccess) { reConnectSocket(); } LogHelper.e(TAG, "send a heart beat "+isSuccess); } mHandler.postDelayed(this, HEART_BEAT_RATE); } }; /** * 重连socket */ private void reConnectSocket() { mHandler.removeCallbacks(heartBeatRunnable); mHandler.removeCallbacks(reSendMsgRunnable); mReadThread.release(); releaseLastSocket(mSocket); new InitSocketThread().start(); mHandler.postDelayed(new Runnable() { @Override public void run() { Intent intent = new Intent(RE_CONNECTION); mLocalBroadcastManager.sendBroadcast(intent); } }, 2000); } private Runnable reSendMsgRunnable = new Runnable() { @Override public void run() { if(map != null &&map.size()>0){ for (String key:map.keySet()){ sendMsg(key,map.get(key)); //重发一次 map.remove(key); } } mHandler.postDelayed(this, RE_SEND_MSG); } }; private long sendTime = 0L; private ISocketService.Stub iSocketService = new ISocketService.Stub() { @Override public boolean sendMessage(String key,String message) throws RemoteException { return sendMsg(key,message); } @Override public boolean keyAuthentication(String key) throws RemoteException { return keyAuthen(key); } }; @Override public IBinder onBind(Intent arg0) { return iSocketService; } @Override public void onCreate() { super.onCreate(); new InitSocketThread().start(); mLocalBroadcastManager=LocalBroadcastManager.getInstance(this); } @Override public void onDestroy() { // TODO Auto-generated method stub super.onDestroy(); if(map !=null) { map.clear(); map = null; } LogHelper.e(TAG, "onDestroy"); mHandler.removeCallbacks(heartBeatRunnable); mHandler.removeCallbacks(reSendMsgRunnable); if(mReadThread != null){ mReadThread.release(); } if(mSocket != null){ releaseLastSocket(mSocket); } } public boolean sendMsg(String key,String msg) { if (null == mSocket || null == mSocket.get()) { return false; } Socket soc = mSocket.get(); try { if (!soc.isClosed() && !soc.isOutputShutdown()) { OutputStream os = soc.getOutputStream(); String message = msg + "\r\n"; os.write(message.getBytes()); os.flush(); sendTime = System.currentTimeMillis();//每次发送成数据,就改一下最后成功发送的时间,节省心跳间隔时间 LogHelper.e(TAG,"SuS sendMsg------> "+msg); } else { return false; } } catch (IOException e) { e.printStackTrace(); return false; } map.put(key,msg); return true; } public boolean keyAuthen(String key){ if (map.containsKey(key)) { map.remove(key); return true; } return false; } private void initSocket() {//初始化Socket try { Socket so = new Socket(HOST, PORT); mSocket = new WeakReference<Socket>(so); mReadThread = new ReadThread(so); mReadThread.start(); mHandler.postDelayed(heartBeatRunnable, HEART_BEAT_RATE);//初始化成功后,就准备发送心跳包 mHandler.postDelayed(reSendMsgRunnable,RE_SEND_MSG); } catch (UnknownHostException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } private void releaseLastSocket(WeakReference<Socket> mSocket) { try { if (null != mSocket) { Socket sk = mSocket.get(); if (sk != null && !sk.isClosed()) { sk.close(); } sk = null; mSocket = null; } } catch (IOException e) { e.printStackTrace(); } } class InitSocketThread extends Thread { @Override public void run() { super.run(); initSocket(); } } // Thread to read content from Socket class ReadThread extends Thread { private WeakReference<Socket> mWeakSocket; private boolean isStart = true; public ReadThread(Socket socket) { mWeakSocket = new WeakReference<Socket>(socket); } public void release() { isStart = false; if(mWeakSocket != null){ releaseLastSocket(mWeakSocket); } } @Override public void run() { super.run(); Socket socket = mWeakSocket.get(); if (null != socket) { try { InputStream is = socket.getInputStream(); byte[] buffer = new byte[1024 * 4]; int length = 0; while (!socket.isClosed() && !socket.isInputShutdown() && isStart && ((length = is.read(buffer)) != -1)) { if (length > 0) { String message = new String(Arrays.copyOf(buffer, length)); //trim可以去掉末尾的\r\n //每条消息以\r\n分开 //直接通过LogHelper打印message无显示,可以加trim或者replace \r\n LogHelper.e(TAG, "原始message------> " + message.replace("\r\n","||||||")); socketMsgStr = socketMsgStr + message; if (!socketMsgStr.contains("\r\n")) { LogHelper.e(TAG, "socketMsgStr without分隔符"); continue; } //返回的消息流有可能是拼接在一起的2条消息,所以需要处理 String[] a = socketMsgStr.split("\r\n"); for(int i = 0;i<a.length;i++){ Intent intent = new Intent(MESSAGE_ACTION); if(a[i].charAt(a[i].length()-1)!='}'){ continue; } intent.putExtra("message", a[i]); mLocalBroadcastManager.sendBroadcast(intent); int rIndex = socketMsgStr.indexOf("\r\n"); if (rIndex + 2 == socketMsgStr.length()) { socketMsgStr = ""; } else { socketMsgStr = socketMsgStr.substring(socketMsgStr.indexOf("\r\n") + 2); } } LogHelper.e(TAG, "截取之后的message------> " + socketMsgStr.trim()); } } } catch (Exception e) {//还会遇到SocketException } } } } }
附:如果哪位朋友有更好的Android Socket编程框架推荐或者优化方面的资源,欢迎共享,大家一起进步!