Socket是應用層與TCP/IP協議族通訊的中間軟件抽象層,它是一組接口。java
講了這麼久,終於要開始講socket的具體實現了,iOS提供了Socket網絡編程的接口CFSocket,不過這裏使用BSD Socket。android
tcp和udp的socket是有區別的,這裏給出這兩種的設計框架編程
基本TCP客戶—服務器程序設計基本框架數組
基本UDP客戶—服務器程序設計基本框架流程圖
服務器
這裏咱們利用Linux C POSIX Socket API進行NDK Socket編程網絡
AbstractEchoActivityapp
package com.apress.echo; import android.app.Activity; import android.os.Bundle; import android.os.Handler; import android.util.Log; import android.view.View; import android.view.View.OnClickListener; import android.widget.Button; import android.widget.EditText; import android.widget.ScrollView; import android.widget.TextView; /** * 客戶端和服務端的抽象父類 共同有一個啓動按鈕,顯示日誌的TextView,端口設置EditText * */ public abstract class AbstractEchoActivity extends Activity implements OnClickListener { protected static final int TCP = 1; protected static final int UDP = 2; protected EditText editPort;// Port number protected Button btnStart;// server button protected ScrollView scrollLog;// protected TextView tvLog;// log view private final int layoutID; public AbstractEchoActivity(int layoutID) { this.layoutID = layoutID; } @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(layoutID); editPort = (EditText) findViewById(R.id.port_edit); btnStart = (Button) findViewById(R.id.start_button); scrollLog = (ScrollView) findViewById(R.id.scroll_view); tvLog = (TextView) findViewById(R.id.log_view); btnStart.setOnClickListener(this); } @Override public void onClick(View v) { if (v == btnStart) { onStartButtonClicked(); } else { Log.v("onClick", "onClick no done."); } } /** * 獲取端口 * * @return */ protected Integer getPort() { Integer port; try { port = Integer.valueOf(editPort.getText().toString()); } catch (Exception e) { e.printStackTrace(); port = null; } return port; } protected void logMessage(final String message) { runOnUiThread(new Runnable() { @Override public void run() { logMessageDirect(message); } }); } protected void logMessageDirect(final String message) { tvLog.append(message); tvLog.append("\n"); scrollLog.fullScroll(View.FOCUS_DOWN); } protected abstract void onStartButtonClicked(); /** * 這個thread抽象出onBackground()方法做爲線程的執行方法,在啓動前先設置控件狀態爲不可用,同時清空日誌。執行完畢後設置控件可用。 * */ protected abstract class AbstractEchoTask extends Thread { private final Handler handler; public AbstractEchoTask() { handler = new Handler(); } protected void onPreExecute() { btnStart.setEnabled(false); // 清空日誌 tvLog.setText(""); } /* * */ @Override public synchronized void start() { // 這裏start是由主線程來調用的。調用以前先設置控件狀態。 onPreExecute(); super.start(); } @Override public void run() { // run是在新線程中運行的 onBackground(); // 用handler來修改控件 handler.post(new Runnable() { @Override public void run() { onPostExecute(); } }); } /** * 線程的執行體 */ protected abstract void onBackground(); /** * */ protected void onPostExecute() { btnStart.setEnabled(true); } } static { System.loadLibrary("Echo"); } }
客戶端 EchoClientActivity 框架
package com.apress.echo; import android.os.Bundle; import android.widget.EditText; public class EchoClientActivity extends AbstractEchoActivity { private EditText editIp; private EditText editMessage; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); editIp = (EditText) findViewById(R.id.ip_edit); editMessage = (EditText) findViewById(R.id.message_edit); } public EchoClientActivity() { super(R.layout.activity_echo_client); } @Override protected void onStartButtonClicked() { String ip = editIp.getText().toString(); Integer port = getPort(); String message = editMessage.getText().toString(); if (0 != ip.length() && port != null && (0 != message.length())) { new ClientTask(ip, port, message).start(); } } private native void nativeStartTcpClient(String ip, int port, String message) throws Exception; private class ClientTask extends AbstractEchoTask { private final String ip; private final int port; private final String message; public ClientTask(String ip, int port, String message) { this.ip = ip; this.port = port; this.message = message; } @Override protected void onBackground() { logMessage("Starting client"); try { nativeStartTcpClient(ip, port, message); } catch (Exception e) { logMessage(e.getMessage()); } logMessage("Client terminated."); } } }
服務端SocketServerdom
EchoServerActivitysocket
package com.apress.echo; public class EchoServerActivity extends AbstractEchoActivity { public EchoServerActivity() { super(R.layout.activity_echo_server); } @Override protected void onStartButtonClicked() { Integer port = getPort(); if (port != null) { new ServerTask(port, TCP).start(); } else { logMessage("port error"); } } /** * 啓動tcp服務 * * @param port * @throws Exception */ private native void nativeStartTcpServer(int port) throws Exception; /** * 啓動udp服務 * * @param port * @throws Exception */ private native void nativeStartUdpServer(int port) throws Exception; private class ServerTask extends AbstractEchoTask { private final int port; private final int protocol; /** * @param port端口 * @param protocol * 使用的協議 */ public ServerTask(int port, int protocol) { this.port = port; this.protocol = protocol; } @Override protected void onBackground() { logMessage("Starting server."); logMessage("server ip:" + Commons.getIpAddress()); try { if (protocol == TCP) { nativeStartTcpServer(port); } else if (protocol == UDP) { nativeStartUdpServer(port); } else { logMessage("protocol error."); } } catch (Exception e) { logMessage(e.getMessage()); } logMessage("Server terminated."); } } }
清單文件
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.apress.echo" android:versionCode="1" android:versionName="1.0" > <uses-sdk android:minSdkVersion="8" android:targetSdkVersion="19" /> <application android:allowBackup="true" android:icon="@drawable/ic_launcher" android:label="@string/app_name" android:theme="@style/AppTheme" > <!-- 服務端app --> <!-- <activity --> <!-- android:name=".EchoServerActivity" --> <!-- android:label="@string/title_activity_echo_server" --> <!-- android:launchMode="singleTop" > --> <!-- <intent-filter> --> <!-- <action android:name="android.intent.action.MAIN" /> --> <!-- <category android:name="android.intent.category.LAUNCHER" /> --> <!-- </intent-filter> --> <!-- </activity> --> <!-- 客戶端app --> <activity android:name=".EchoClientActivity" android:label="@string/title_activity_echo_client" android:launchMode="singleTop" > <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> </application> <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" /> <uses-permission android:name="android.permission.INTERNET" /> </manifest>
實現NDK編程
native接口文件
/* DO NOT EDIT THIS FILE - it is machine generated */ #include <jni.h> /* Header for class com_apress_echo_EchoServerActivity */ #ifndef _Included_com_apress_echo_EchoServerActivity #define _Included_com_apress_echo_EchoServerActivity #ifdef __cplusplus extern "C" { #endif #undef com_apress_echo_EchoServerActivity_TCP #define com_apress_echo_EchoServerActivity_TCP 1L #undef com_apress_echo_EchoServerActivity_UDP #define com_apress_echo_EchoServerActivity_UDP 2L /* * Class: com_apress_echo_EchoServerActivity * Method: nativeStartTcpServer * Signature: (I)V */ JNIEXPORT void JNICALL Java_com_apress_echo_EchoServerActivity_nativeStartTcpServer (JNIEnv *, jobject, jint); /* * Class: com_apress_echo_EchoServerActivity * Method: nativeStartUdpServer * Signature: (I)V */ JNIEXPORT void JNICALL Java_com_apress_echo_EchoServerActivity_nativeStartUdpServer (JNIEnv *, jobject, jint); #ifdef __cplusplus } #endif #endif
實現native Socket
SocketTool.cpp
#include <stdio.h> #include <stdarg.h> //errno #include <errno.h> #include <string.h> #include <sys/types.h> #include <sys/socket.h> //sockaddr_un #include <sys/un.h> //htons,sockaddr_in #include <netinet/in.h> //inet_ntop #include <arpa/inet.h> //close,unlink #include <unistd.h> //offsetof #include <stddef.h> #ifndef __SOCKET_UTILS__ #define __SOCKET_UTILS_ //MAX log message length #define MAX_LOG_MESSAGE_LENGTH 256 //MAX data buffer size #define MAX_BUFFER_SIZE 80 //打印日誌到java環境中 static void LogMessage(JNIEnv* env, jobject obj, const char* format, ...) { //cache log method ID static jmethodID methodID = NULL; if (methodID == NULL) { jclass clazz = env->GetObjectClass(obj); methodID = env->GetMethodID(clazz, "logMessage", "(Ljava/lang/String;)V"); env->DeleteLocalRef(clazz); } if (methodID != NULL) { char buffer[MAX_BUFFER_SIZE]; //將可變參數輸出到字符數組中 va_list ap; va_start(ap, format); vsnprintf(buffer, MAX_LOG_MESSAGE_LENGTH, format, ap); va_end(ap); //轉換成java字符串 jstring message = env->NewStringUTF(buffer); if (message != NULL) { env->CallVoidMethod(obj, methodID, message); env->DeleteLocalRef(message); } } } //經過異常類和異常信息拋出異常 static void ThrowException(JNIEnv* env, const char* className, const char* message) { jclass clazz = env->FindClass(className); if (clazz != NULL) { env->ThrowNew(clazz, message); env->DeleteLocalRef(clazz); } } //經過異常類和錯誤號拋出異常 static void ThrowErrnoException(JNIEnv* env, const char* className, int errnum) { char buffer[MAX_LOG_MESSAGE_LENGTH]; //經過錯誤號得到錯誤消息 if (-1 == strerror_r(errnum, buffer, MAX_LOG_MESSAGE_LENGTH)) { strerror_r(errno, buffer, MAX_LOG_MESSAGE_LENGTH); } ThrowException(env, className, buffer); } //sock用到的一些公用方法 //建立一個socket:socket() static int NewTcpSocket(JNIEnv* env, jobject obj) { LogMessage(env, obj, "Constructing a new TCP socket..."); int tcpSocket = socket(PF_INET, SOCK_STREAM, 0); if (-1 == tcpSocket) { ThrowErrnoException(env, "java/io/IOException", errno); } return tcpSocket; } //綁定 bind() static void BindSocketToPort(JNIEnv* env, jobject obj, int sd, unsigned short port) { struct sockaddr_in address; //清空結構體 memset(&address, 0, sizeof(address)); address.sin_family = PF_INET; //Bind to all address address.sin_addr.s_addr = htonl(INADDR_ANY); //Convert port to network byte order address.sin_port = htons(port); //Bind socket LogMessage(env, obj, "Binding to port %hu.", port); //sockaddr方便函數傳遞, sockaddr_in方便用戶設定, 因此須要的時候在這2者之間進行轉換 if (-1 == bind(sd, (struct sockaddr*) &address, sizeof(address))) { ThrowErrnoException(env, "java/io/IOException", errno); } } //返回當前socket綁定的端口 static unsigned short GetSocketPort(JNIEnv* env, jobject obj, int sd) { unsigned short port = 0; struct sockaddr_in address; socklen_t addressLength = sizeof(address); if (-1 == getsockname(sd, (struct sockaddr*) &address, &addressLength)) { ThrowErrnoException(env, "java/io/IOException", errno); } else { port = ntohs(address.sin_port); LogMessage(env, obj, "Binding to the random port %hu.", port); } return port; } //監聽 listen() static void ListenOnSocket(JNIEnv*env, jobject obj, int sd, int backlog) { LogMessage(env, obj, "Listening on socket with a baklog of %d pending connections.", backlog); //listen()用來等待參數s 的socket 連線. 參數backlog 指定同時能處理的最大鏈接要求, //若是鏈接數目達此上限則client 端將收到ECONNREFUSED 的錯誤. //Listen()並未開始接收連線, 只是設置socket 爲listen 模式, 真正接收client 端連線的是accept(). //一般listen()會在socket(), bind()以後調用, 接着才調用accept(). if (-1 == listen(sd, backlog)) { ThrowErrnoException(env, "java/io/IOException", errno); } } //根據地址打印IP和端口 static void LogAddress(JNIEnv* env, jobject obj, const char* message, const struct sockaddr_in* address) { char ip[INET_ADDRSTRLEN]; if (NULL == inet_ntop(PF_INET, &(address->sin_addr), ip, INET_ADDRSTRLEN)) { ThrowErrnoException(env, "java/io/IOException", errno); } else { unsigned short port = ntohs(address->sin_port); LogMessage(env, obj, "%s %s:%hu", message, ip, port); } } //accept() static int AcceptOnSocket(JNIEnv* env, jobject obj, int sd) { struct sockaddr_in address; socklen_t addressLength = sizeof(address); LogMessage(env, obj, "Waiting for a client connection..."); int clientSocket = accept(sd, (struct sockaddr*) &address, &addressLength); if (-1 == clientSocket) { ThrowErrnoException(env, "java/io/IOException", errno); } else { LogAddress(env, obj, "Client connection from ", &address); } return clientSocket; } //接收 recv() static ssize_t ReceiveFromSocket(JNIEnv* env, jobject obj, int sd, char* buffer, size_t bufferSize) { LogMessage(env, obj, "Receiving from the socket... "); ssize_t recvSize = recv(sd, buffer, bufferSize - 1, 0); if (-1 == recvSize) { ThrowErrnoException(env, "java/io/IOException", errno); } else { //字符串截斷 buffer[recvSize] = NULL; if (recvSize > 0) { //接收成功,打印 LogMessage(env, obj, "Received %d bytes:%s", bufferSize, buffer); } else { LogMessage(env, obj, "Client disconnected."); } } return recvSize; } //發送消息:send() static ssize_t SendToSocket(JNIEnv *env, jobject obj, int sd, const char* buffer, size_t bufferSize) { LogMessage(env, obj, "Sending to the socket... "); ssize_t sentSize = send(sd, buffer, bufferSize, 0); if (-1 == sentSize) { ThrowErrnoException(env, "java/io/IOException", errno); } else { if (sentSize > 0) { LogMessage(env, obj, "Send %d bytes: %s", sentSize, buffer); } else { LogMessage(env, obj, "Client disconnected."); } } return sentSize; } //連接到服務器 connect() static void ConnectToAddress(JNIEnv*env, jobject obj, int sd, const char*ip, unsigned short port) { LogMessage(env, obj, "Connecting to %s:%hu...", ip, port); struct sockaddr_in address; memset(&address, 0, sizeof(address)); address.sin_family = PF_INET; //轉換ip if (0 == inet_aton(ip, &(address.sin_addr))) { ThrowErrnoException(env, "java/io/IOException", errno); } else { address.sin_port = htons(port); } if (-1 == connect(sd, (const sockaddr*) &address, sizeof(address))) { ThrowErrnoException(env, "java/io/IOException", errno); } else { LogMessage(env, obj, "Connected."); } } //----------------udp //建立udp socket static int NewUdpSocket(JNIEnv* env, jobject obj) { LogMessage(env, obj, "Constructing a new UDP socket..."); int udpSocket = socket(PF_INET, SOCK_DGRAM, 0); if (-1 == udpSocket) { ThrowErrnoException(env, "java/io/IOException", errno); } return udpSocket; } #endif __SOCKET_UTILS_
實現Native 接口
#include <jni.h> #include "com_apress_echo_EchoServerActivity.h" #include "com_apress_echo_EchoClientActivity.h" #include "SocketTool.cpp" //服務端:啓動監聽 //流程:socket()->listen()->accept()->recv()->send()_close() void JNICALL Java_com_apress_echo_EchoServerActivity_nativeStartTcpServer( JNIEnv *env, jobject obj, jint port) { int serverSocket = NewTcpSocket(env, obj); if (NULL == env->ExceptionOccurred()) { //綁定 BindSocketToPort(env, obj, serverSocket, (unsigned short) port); if (NULL != env->ExceptionOccurred()) { goto exit; } //若是端口是0,打印出當前隨機分配的端口 if (0 == port) { GetSocketPort(env, obj, serverSocket); if (NULL != env->ExceptionOccurred()) { goto exit; } } //監聽 連接4 ListenOnSocket(env, obj, serverSocket, 4); if (NULL != env->ExceptionOccurred()) { goto exit; } // int clientSocket = AcceptOnSocket(env, obj, serverSocket); if (NULL != env->ExceptionOccurred()) { goto exit; } char buffer[MAX_BUFFER_SIZE]; ssize_t recvSize; ssize_t sentSize; while (1) { //接收 recvSize = ReceiveFromSocket(env, obj, clientSocket, buffer, MAX_BUFFER_SIZE); if ((0 == recvSize) || (NULL != env->ExceptionOccurred())) { break; } //發送 sentSize = SendToSocket(env, obj, clientSocket, buffer, (size_t) recvSize); if ((0 == sentSize) || (NULL != env->ExceptionOccurred())) { break; } } //close the client socket close(clientSocket); } exit: if (serverSocket > 0) { close(serverSocket); } } //客戶端:鏈接 void JNICALL Java_com_apress_echo_EchoClientActivity_nativeStartTcpClient( JNIEnv *env, jobject obj, jstring ip, jint port, jstring message) { int clientSocket = NewTcpSocket(env, obj); if (NULL == env->ExceptionOccurred()) { const char* ipAddress = env->GetStringUTFChars(ip, NULL); if (NULL == ipAddress) { goto exit; } ConnectToAddress(env, obj, clientSocket, ipAddress, (unsigned short) port); //釋放ip env->ReleaseStringUTFChars(ip, ipAddress); //connect exception check if (NULL != env->ExceptionOccurred()) { goto exit; } const char* messageText = env->GetStringUTFChars(message, NULL); if (NULL == messageText) { goto exit; } //這裏的size不用release?? jsize messageSize = env->GetStringUTFLength(message); SendToSocket(env, obj, clientSocket, messageText, messageSize); // env->ReleaseStringUTFChars(message, messageText); if (NULL != env->ExceptionOccurred()) { goto exit; } char buffer[MAX_BUFFER_SIZE]; ReceiveFromSocket(env, obj, clientSocket, buffer, MAX_BUFFER_SIZE); } exit: if (clientSocket > -1) { close(clientSocket); } } //啓動udp服務端 void JNICALL Java_com_apress_echo_EchoServerActivity_nativeStartUdpServer( JNIEnv *, jobject, jint) { }