溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Android中如何實(shí)現(xiàn)Socket通信

發(fā)布時(shí)間:2021-09-26 15:49:15 來源:億速云 閱讀:123 作者:小新 欄目:開發(fā)技術(shù)

小編給大家分享一下Android中如何實(shí)現(xiàn)Socket通信,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

公司要實(shí)現(xiàn)一個(gè)簡(jiǎn)單的聊天功能,提前研究一下Socket通信,而公司的服務(wù)端功能又沒有實(shí)現(xiàn),所以這里就把服務(wù)端的功能放到一個(gè)界面實(shí)現(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="服務(wù)端發(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);
 
        //啟動(dòng)服務(wù)端
        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://連接服務(wù)端
                String strip = etIp.getText().toString().trim();
                if (strip.indexOf(":") >= 0) {
                    //啟動(dòng)連接
                    new Socket_thread(strip).start();
                }
                break;
            case R.id.btn_service:
                if (!TextUtils.isEmpty(etInput.getText().toString())) {
                    sendString("服務(wù)端:" + 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;
        }
    }
 
    /**
     * 連接服務(wù)器
     */
    class Socket_thread extends Thread {
        private String IP = "";//ip地址
        private int PORT = 0;//端口號(hào)
 
        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();
                //連接服務(wù)器,此處會(huì)一直處于阻塞,直到連接成功
                sock = new Socket(this.IP, this.PORT);
                //阻塞停止,表示連接成功
                setMessage("連接成功", MESSAGE_SUCCEED);
            } catch (Exception e) {
                setMessage("連接服務(wù)器時(shí)異常", 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);//設(shè)備重啟,異常 將會(huì)一直停留在這
                    if (conut == -1) {
                        setMessage("服務(wù)器斷開", 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("斷開連接時(shí)發(fā)生錯(cuò)誤", 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);
    }
 
 
    /*************************************************************服務(wù)端(用于測(cè)試)**********************************************************************/
    private String msg = "";
 
    public class Server {
        ServerSocket serverSocket = null;
        public final int port = 8081;
 
        public Server() {
            //輸出服務(wù)器的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();//等待一個(gè)客戶端的連接,在連接之前,此方法是阻塞的
                    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);//設(shè)備重啟,異常 將會(huì)一直停留在這
                        if (conut == -1) {
                            setMessage("服務(wù)器斷開", 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;
    }
}

運(yùn)行效果:

Android中如何實(shí)現(xiàn)Socket通信

以上是“Android中如何實(shí)現(xiàn)Socket通信”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對(duì)大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI