一、UDP協議全稱是用戶數據報協議 ,在網絡中它與TCP協議一樣用於處理數據包,是一種無連接的協議。
1、UDP是一個無連接協議,傳輸數據之前源端和終端不建立連接;
2、不維護連接狀態,包括收發狀態等,因此一台服務機可同時向多個客戶機傳輸相同的消息;
3、UDP信息包的標題很短,8個字節,對於TCP的20個字節信息包的額外開銷很小;
4、吞吐量不受擁擠控制算法的調節,只受應用軟件生成數據的速率、傳輸帶寬、源端和終端主機性能的限制;
5、不保證可靠交付;
6、UDP是面向報文的。
二、UDP的代碼編程需要用到DatagramSocket類,Java使用DatagramSocket代表UDP協議的Socket,先了解一下如何使用這個類。
server服務端(接收數據方)創建步驟
1、生成DatagramSocket對象,參數為端口號
DatagramSocket socket = new DatagramSocket (8888);
2、byte數組用來接收數據
byte data[] = new byte[1024];
3、DatagramPackage以包形式裝載byte數組
DatagramPacket packet = new DatagramPacket(data, data.length);
4、使用DatagramPacket的receive方法接收發送方所發送的數據,是一個阻塞的方法
socket.receive(packet);
5、接收到的數據轉變化成可讀字符串
String result = new String(packet.getData(), packet.getOffset(), packet.getLength());
client客戶端(發送數據方)創建步驟
1、生成DatagramSocket對象,參數為端口號
DatagramSocket socket = new DatagramSocket (8888);
2、創建一個InetAddress對象,用於確定server客戶端(數據接收方),參數為IP地址,如我的IP地址為192.168.1.103
InetAddress serverAddress = InetAddress.getByName("192.168.1.103");
3、需要發送的數據轉化為byte數組
String sendData= "hello word"; byte data[] = sendData.getBytes();
4、生成一個DatagramPacket對象,包含需要發送的byte數組,byte數組長度,已設置IP地址的serverAddress,和服務端一致的端口號(這里是8888)
DatagramPacket packet = new DatagramPacket(data, data.length, serverAddress, 8888);
5、發送數據,調用DatagramSocket對象的send方法
socket.send(packet);
三、代碼demo演示
server服務端,可以直接運行在eclipse即可
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
public class UDPServer {
public static void main(String[] args) throws IOException {
DatagramSocket socket = null;
try {
socket = new DatagramSocket(8888);
} catch (Exception e) {
e.printStackTrace();
}
while (true) {
byte data[] = new byte[1024];
DatagramPacket packet = new DatagramPacket(data, data.length);
socket.receive(packet);
String result = new String(packet.getData(), packet.getOffset(), packet.getLength());
System.out.println("receive client's data: " + result);
}
}
}
若多次運行會提示錯誤:Address already in use: Cannot bind,關掉之前開啟的即可。
Android客戶端代碼demo演示,需要注意的是端口號是與服務端一致的,IP地址是服務端的IP地址。
import android.app.Activity;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
public class UDPActivity extends Activity {
private Button bt_send_data = null;
DatagramSocket socket = null;
InetAddress serverAddress = null;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
bt_send_data = new Button(this);
bt_send_data.setText("發送");
setContentView(bt_send_data);
try {
socket = new DatagramSocket(8888);
serverAddress = InetAddress.getByName("192.168.1.101");
} catch (Exception e) {
e.printStackTrace();
}
bt_send_data.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View arg0) {
new Thread(new Runnable() {
@Override
public void run() {
try {
String sendData = "hello world";
byte data[] = sendData.getBytes();
DatagramPacket packet = new DatagramPacket(data, data.length, serverAddress, 8888);
socket.send(packet);
} catch (Exception e) {
e.printStackTrace();
}
}
}).start();
}
});
}
}
在AndroidManifest.xml添加權限和注冊
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.libill.demos"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk
android:minSdkVersion="8"
android:targetSdkVersion="10" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="android.permission.CHANGE_NETWORK_STATE"/>
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE"/>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>
<uses-permission android:name="android.permission.INTERNET"/>
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppStartTheme" >
<activity
android:name="com.liqw.demos.activity.UDPServerActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
這樣即可跑起demo做測試了。先運行服務端,再點擊發送4次,服務端的log如下:

參考博客:
http://www.cnblogs.com/lee0oo0/archive/2012/04/04/2431907.html
http://wang09si.blog.163.com/blog/static/170171804201303004343304/
