Android Socket通信的簡單實現(xiàn)
更新時間:2021年09月26日 14:00:52 作者:c小旭
這篇文章主要為大家詳細介紹了Android Socket通信的簡單實現(xiàn),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
公司要實現(xiàn)一個簡單的聊天功能,提前研究一下Socket通信,而公司的服務端功能又沒有實現(xiàn),所以這里就把服務端的功能放到一個界面實現(xiàn)了。
直接上代碼:
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity">
<EditText
android:id="@+id/et_ip"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:hint="IP:端口"
android:text="127.0.0.1:8081"
android:layout_margin="10dp"
android:padding="10dp"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toLeftOf="@+id/btn_connect"
app:layout_constraintTop_toTopOf="parent" />
<Button
android:id="@+id/btn_connect"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintRight_toRightOf="parent"
android:layout_margin="10dp"
android:text="連接"/>
<TextView
android:id="@+id/tv_receive"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="接受數(shù)據(jù):"
app:layout_constraintTop_toBottomOf="@+id/et_ip"
app:layout_constraintLeft_toLeftOf="parent"
android:textSize="14sp"
android:layout_margin="10dp"/>
<ScrollView
android:id="@+id/sv_content"
android:layout_width="match_parent"
android:layout_height="200dp"
app:layout_constraintTop_toBottomOf="@+id/tv_receive"
app:layout_constraintLeft_toLeftOf="parent">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView
android:id="@+id/tv_content"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="聊天內(nèi)容"
android:textColor="#000"
android:textSize="16sp"
android:layout_margin="10dp"/>
</LinearLayout>
</ScrollView>
<EditText
android:id="@+id/et_input"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="發(fā)送內(nèi)容"
android:layout_margin="10dp"
android:padding="10dp"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintTop_toBottomOf="@+id/sv_content" />
<Button
android:id="@+id/btn_service"
android:layout_width="0dp"
android:layout_height="wrap_content"
app:layout_constraintTop_toBottomOf="@+id/et_input"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toLeftOf="@+id/btn_send"
android:layout_margin="10dp"
android:text="服務端發(fā)送"/>
<Button
android:id="@+id/btn_send"
android:layout_width="0dp"
android:layout_height="wrap_content"
app:layout_constraintTop_toBottomOf="@+id/et_input"
app:layout_constraintLeft_toRightOf="@+id/btn_service"
app:layout_constraintRight_toRightOf="parent"
android:layout_margin="10dp"
android:text="發(fā)送"/>
</androidx.constraintlayout.widget.ConstraintLayout>
主要代碼:
package com.app.socketdemo;
import android.annotation.SuppressLint;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.text.Html;
import android.text.TextUtils;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ScrollView;
import android.widget.TextView;
import android.widget.Toast;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.InetAddress;
import java.net.ServerSocket;
import java.net.Socket;
import androidx.appcompat.app.AppCompatActivity;
import butterknife.BindView;
import butterknife.ButterKnife;
import butterknife.OnClick;
public class MainActivity extends AppCompatActivity {
@BindView(R.id.et_ip)
EditText etIp;
@BindView(R.id.btn_connect)
Button btnConnect;
@BindView(R.id.tv_receive)
TextView tvReceive;
@BindView(R.id.tv_content)
TextView tvContent;
@BindView(R.id.et_input)
EditText etInput;
@BindView(R.id.btn_send)
Button btnSend;
@BindView(R.id.sv_content)
ScrollView svContent;
@BindView(R.id.btn_service)
Button btnService;
private StringBuffer strMsg = new StringBuffer();
private final int MESSAGE_ERROR = 0;
private final int MESSAGE_SUCCEED = 1;
private final int MESSAGE_RECEIVE = 2;
private Socket sock;
private OutputStream outx;
private InputStream inx;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.bind(this);
//啟動服務端
new Thread(() -> new Server().startService()).start();
}
@OnClick({R.id.btn_connect, R.id.btn_service, R.id.btn_send})
public void onViewClicked(View view) {
switch (view.getId()) {
case R.id.btn_connect://連接服務端
String strip = etIp.getText().toString().trim();
if (strip.indexOf(":") >= 0) {
//啟動連接
new Socket_thread(strip).start();
}
break;
case R.id.btn_service:
if (!TextUtils.isEmpty(etInput.getText().toString())) {
sendString("服務端:" + etInput.getText().toString().trim());
etInput.setText("");
} else {
Toast.makeText(this, "輸入不可為空", Toast.LENGTH_SHORT).show();
}
break;
case R.id.btn_send:
if (!TextUtils.isEmpty(etInput.getText().toString())) {
sendStrSocket("客戶端:" + etInput.getText().toString().trim());
etInput.setText("");
} else {
Toast.makeText(this, "輸入不可為空", Toast.LENGTH_SHORT).show();
}
break;
}
}
/**
* 連接服務器
*/
class Socket_thread extends Thread {
private String IP = "";//ip地址
private int PORT = 0;//端口號
public Socket_thread(String strip) {
//如: 127.0.0.1:8081
String[] stripx = strip.split(":");
this.IP = stripx[0];
this.PORT = Integer.parseInt(stripx[1]);
}
@Override
public void run() {
try {
disSocket();
//連接服務器,此處會一直處于阻塞,直到連接成功
sock = new Socket(this.IP, this.PORT);
//阻塞停止,表示連接成功
setMessage("連接成功", MESSAGE_SUCCEED);
} catch (Exception e) {
setMessage("連接服務器時異常", MESSAGE_ERROR);
e.printStackTrace();
return;
}
try {
//獲取到輸入輸出流
outx = sock.getOutputStream();
inx = sock.getInputStream();
} catch (Exception e) {
setMessage("獲取輸入輸出流異常", MESSAGE_ERROR);
e.printStackTrace();
return;
}
new Inx().start();
}
}
/**
* 循環(huán)接收數(shù)據(jù)
*/
class Inx extends Thread {
@Override
public void run() {
while (true) {
byte[] bu = new byte[1024];
try {
int conut = inx.read(bu);//設備重啟,異常 將會一直停留在這
if (conut == -1) {
setMessage("服務器斷開", MESSAGE_ERROR);
disSocket();
return;
}
String strread = new String(bu, "GBK").trim();
setMessage(strread, MESSAGE_RECEIVE);
} catch (IOException e) {
System.out.println(e);
}
}
}
}
/**
* 斷開連接
*/
private void disSocket() {
if (sock != null) {
try {
outx.close();
inx.close();
sock.close();
sock = null;
} catch (Exception e) {
setMessage("斷開連接時發(fā)生錯誤", MESSAGE_ERROR);
}
}
}
@SuppressLint("HandlerLeak")
private Handler handler = new Handler() {
@Override
public void handleMessage(Message msg) {
switch (msg.arg1) {
case MESSAGE_ERROR:
disSocket();
strMsg.append(msg.obj + "<br>");
tvContent.setText(Html.fromHtml(strMsg.toString()));
break;
case MESSAGE_SUCCEED:
strMsg.append(msg.obj + "<br>");
tvContent.setText(Html.fromHtml(strMsg.toString()));
break;
case MESSAGE_RECEIVE:
//收到數(shù)據(jù)
strMsg.append(msg.obj);
if (!strMsg.toString().substring(strMsg.length() - 4, strMsg.length()).equals("<br>")) {
strMsg.append("<br>");
}
tvContent.setText(Html.fromHtml(strMsg.toString()));
svContent.fullScroll(ScrollView.FOCUS_DOWN);
break;
default:
break;
}
}
};
/**
* 發(fā)送消息
*/
private void sendStrSocket(final String senddata) {
new Thread(new Runnable() {
@Override
public void run() {
try {
String str = "<font color='#EE2C2C'>" + senddata + "</font>";
outx.write(str.getBytes("gbk"));//"utf-8"
} catch (Exception e) {
setMessage("數(shù)據(jù)發(fā)送異常", MESSAGE_ERROR);
}
}
}).start();
}
/**
* 消息處理
*/
private void setMessage(String obj, int arg1){
Message message = new Message();
message.arg1 = arg1;
message.obj = obj;
handler.sendMessage(message);
}
/*************************************************************服務端(用于測試)**********************************************************************/
private String msg = "";
public class Server {
ServerSocket serverSocket = null;
public final int port = 8081;
public Server() {
//輸出服務器的IP地址
try {
InetAddress addr = InetAddress.getLocalHost();
serverSocket = new ServerSocket(port);
} catch (IOException e) {
e.printStackTrace();
}
}
public void startService() {
try {
while (true) {
Socket socket = null;
socket = serverSocket.accept();//等待一個客戶端的連接,在連接之前,此方法是阻塞的
new ConnectThread(socket).start();
new ConnectThread1(socket).start();
}
} catch (IOException e) {
e.printStackTrace();
}
}
/**
* 向客戶端發(fā)送信息
*/
class ConnectThread extends Thread {
Socket socket = null;
public ConnectThread(Socket socket) {
super();
this.socket = socket;
}
@Override
public void run() {
try {
DataOutputStream out = new DataOutputStream(socket.getOutputStream());
while (true) {
Thread.sleep(1000);
if (!TextUtils.isEmpty(msg)) {
String str = "<font color='#4F94CD'>" + msg + "</font>";
out.write(str.getBytes("gbk"));
out.flush();
msg = "";
}
}
} catch (IOException e) {
e.printStackTrace();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
/**
* 接收客戶端信息
*/
class ConnectThread1 extends Thread {
Socket socket = null;
public ConnectThread1(Socket socket) {
super();
this.socket = socket;
}
@Override
public void run() {
try {
DataInputStream inp = new DataInputStream(socket.getInputStream());
while (true) {
byte[] bu = new byte[1024];
int conut = inp.read(bu);//設備重啟,異常 將會一直停留在這
if (conut == -1) {
setMessage("服務器斷開", MESSAGE_ERROR);
return;
}
String strread = new String(bu, "GBK").trim();
setMessage(strread, MESSAGE_RECEIVE);
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
private void sendString(String str) {
msg = str;
}
}
運行效果:

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android App中使用SurfaceView制作多線程動畫的實例講解
這篇文章主要介紹了Android App中使用SurfaceView制作多線程動畫的實例講解,SurfaceView經(jīng)常被用來制作游戲中的動畫,不過同時要注意畫面閃爍的問題,需要的朋友可以參考下2016-04-04
android調(diào)用web service(cxf)實例應用詳解
Google為ndroid平臺開發(fā)Web Service提供了支持,提供了Ksoap2-android相關(guān)架包接下來介紹android調(diào)用web service(cxf),感興趣的朋友可以了解下2013-01-01
React-Native Android 與 IOS App使用一份代碼實現(xiàn)方法
這篇文章主要介紹了React-Native Android 與 IOS App使用一份代碼實現(xiàn)方法的相關(guān)資料,這里舉例說明,該如何實現(xiàn)IOS和Android APP 都使用一樣的代碼,需要的朋友可以參考下2016-12-12

