Android Socket保持心跳長(zhǎng)連接逼蒙,斷線重連

昨天三點(diǎn)鐘才睡覺(jué)的贡茅,現(xiàn)在胸口感覺(jué)悶悶的,兄弟們其做,我是不是要GG了顶考?如果我G了,求大佬們給我燒個(gè)女朋友妖泄,

ss.gif

1.在使用Socket連接客戶端和服務(wù)器端的時(shí)候驹沿,如果服務(wù)端斷開(kāi)了連接,我們客戶端是收不到任何回調(diào)消息的蹈胡,只是在你發(fā)送消息給服務(wù)器的時(shí)候渊季,會(huì)走異常,表示發(fā)送失敗罚渐。
2.所以要判斷服務(wù)器是否在線却汉,就需要客戶端不停的發(fā)送心跳消息給服務(wù)器,服務(wù)器收到心跳消息荷并,就立馬回復(fù)給你消息合砂,這樣就 能知道雙方是否都在線。
3.如果在一段時(shí)間內(nèi)源织,還是沒(méi)有收到服務(wù)器回復(fù)的消息翩伪,就表示服務(wù)器可能已經(jīng)死了,這時(shí)候你可能需要去做一些提示信息給Android前臺(tái)谈息。
4.在這一段時(shí)間內(nèi)缘屹,你可以不停的嘗試重新建立Socket連接,即斷線重連侠仇。

上代碼吧:
首先正常創(chuàng)建一個(gè)Activity轻姿,并創(chuàng)建一個(gè)TcpService服務(wù),在服務(wù)中去進(jìn)行Socket的相關(guān)操作逻炊。在connection中回調(diào)的clientBinder 對(duì)象互亮,就是Activity和Service通訊的橋梁。上一篇我們是在Activity里去進(jìn)行Socket測(cè)試的嗅骄,用Service顯然要比用Activity好的多胳挎。

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Intent intent = new Intent(this,TcpService.class);
        bindService(intent,connection,BIND_AUTO_CREATE);
    }

    ServiceConnection connection = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            TcpService.ClientBinder clientBinder = (TcpService.ClientBinder) service;
            clientBinder.startConnect();
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {
        }
    };
}

在TcpService的onBind()方法中,我們首先需要返回ClientBinder這個(gè)對(duì)象溺森,然后調(diào)用clientBinder.startConnect()建立Socket連接慕爬。

public void startConnect() {
            //在子線程進(jìn)行網(wǎng)絡(luò)操作
            // Service也是運(yùn)行在主線程窑眯,千萬(wàn)不要以為Service意思跟后臺(tái)運(yùn)行很像,就以為Service運(yùn)行在后臺(tái)子線程
            if (mExecutorService == null) {
                mExecutorService = Executors.newCachedThreadPool();
            }
            mExecutorService.execute(connectRunnable);
        }

        private Runnable connectRunnable = new Runnable() {
            @Override
            public void run() {
                try {
                    // 建立Socket連接
                    mSocket = new Socket();
                    mSocket.connect(new InetSocketAddress("192.168.1.186", 8292), 10);
                    bis = new BufferedInputStream(mSocket.getInputStream());
                    bos = new BufferedOutputStream(mSocket.getOutputStream());
                    // 創(chuàng)建讀取服務(wù)器心跳的線程
                    mReadThread = new ReadThread();
                    mReadThread.start();
                    //開(kāi)啟心跳,每隔3秒鐘發(fā)送一次心跳
                    mHandler.post(mHeartRunnable);
                    tryCount = 1;
                } catch (Exception e) {
                    tryCount ++ ;
                    e.printStackTrace();
                    Log.d(TAG, "Socket連接建立失敗,正在嘗試第"+ tryCount + "次重連");
                    mHandler.postDelayed(new Runnable() {
                        @Override
                        public void run() {
                            mExecutorService.execute(connectRunnable);
                        }
                    },mHeart_spacetime);
                }
            }
        };

這里我創(chuàng)建了一個(gè)線程池医窿,用線程池去處理Socket的需要聯(lián)網(wǎng)的Runnable磅甩。然后就是正常的創(chuàng)建Socket連接,新建讀取線程以及開(kāi)啟mHeartRunnable姥卢。在異常處理里面卷要,如果建立Socket失敗,就發(fā)送一個(gè)延時(shí)消息独榴,重新去創(chuàng)建連接僧叉。下面我們看一下ReadThread。

public class ReadThread extends Thread {
            @Override
            public void run() {
                int size;
                byte[] buffer = new byte[1024];
                try {
                    while ((size = bis.read(buffer)) != -1) {
                        String str = new String(buffer, 0, size);
                        Log.d(TAG,"我收到來(lái)自服務(wù)器的消息: " +str);
                        //收到心跳消息以后棺榔,首先移除斷連消息瓶堕,然后創(chuàng)建一個(gè)新的60秒后執(zhí)行斷連的消息。
                        //這樣每次收到心跳后都會(huì)重新創(chuàng)建一個(gè)60秒的延時(shí)消息症歇,在60秒后還沒(méi)收到心跳消息郎笆,表明服務(wù)器已死,就會(huì)執(zhí)行斷開(kāi)Socket連接
                        //在60秒鐘內(nèi)如果收到過(guò)一次心跳消息忘晤,就表明服務(wù)器還活著宛蚓,可以繼續(xù)與之通訊。
                        mHandler.removeCallbacks(disConnectRunnable);
                        mHandler.postDelayed(disConnectRunnable, mHeart_spacetime * 40);
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }

ReadThread主要處理的是disConnectRunnable设塔,接收到服務(wù)器的心跳消息就移除斷連任務(wù)凄吏,然后重新創(chuàng)建一個(gè)新的斷連任務(wù)。在指定的時(shí)間內(nèi)沒(méi)有收到服務(wù)端的心跳消息壹置,斷連任務(wù)就會(huì)執(zhí)行竞思。反之表谊,則會(huì)又進(jìn)入一個(gè) "移除舊的—?jiǎng)?chuàng)建新的" 的循環(huán)钞护。當(dāng)然,這個(gè)發(fā)送心跳消息的時(shí)間間隔(mHeart_spacetime )肯定是要小于這個(gè)斷連任務(wù)延時(shí)時(shí)間的(mHeart_spacetime * 40)爆办。接下來(lái)难咕,看一下mHeartRunnable,心跳發(fā)送失敗以后立馬執(zhí)行重連操作距辆。

 private Runnable mHeartRunnable = new Runnable() {
            @Override
            public void run() {
                sendData();
            }
        };

        private void sendData() {
            mExecutorService.execute(new Runnable() {
                @Override
                public void run() {
                    try {
                        bos.write("給你一張過(guò)去的CD,聽(tīng)聽(tīng)那時(shí)我們的愛(ài)情余佃!".getBytes());
                        //一定不能忘記這步操作
                        bos.flush();
                        //發(fā)送成功以后,重新建立一個(gè)心跳消息
                        mHandler.postDelayed(mHeartRunnable, mHeart_spacetime);
                        Log.d(TAG, "我發(fā)送給服務(wù)器的消息: 給你一張過(guò)去的CD,聽(tīng)聽(tīng)那時(shí)我們的愛(ài)情跨算!");
                    } catch (Exception e) {
                        e.printStackTrace();
                        Log.d(TAG, "心跳任務(wù)發(fā)送失敗爆土,正在嘗試第"+ tryCount + "次重連");
                        //mExecutorService.schedule(connectRunnable,mHeart_spacetime, TimeUnit.SECONDS);
                        mExecutorService.execute(connectRunnable);
                    }
                }
            });
        }

下面來(lái)看一下效果圖:


客戶端和服務(wù)器正常收發(fā)心跳≈畈希客戶端每隔3秒鐘發(fā)送一次心跳步势,服務(wù)器收到心跳后立馬回復(fù)心跳氧猬。此時(shí)雙方都正常在線。

當(dāng)我停止服務(wù)端連接的時(shí)候坏瘩,程序開(kāi)始自動(dòng)重連

然后我又重新開(kāi)啟了服務(wù)盅抚,可以看到在重連到第10次的時(shí)候,Socket連接重新建立倔矾,并正常收發(fā)心跳消息

最后妄均,附上TcpService類(lèi)和AppServer類(lèi)的代碼:

客戶端代碼

/**
 * Create by Fiora on 2018/10/24 0024
 */
public class TcpService extends Service {
    public static final String TAG = TcpService.class.getSimpleName();

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return new ClientBinder();
    }

    public class ClientBinder extends Binder {
        private int mHeart_spacetime = 3 * 1000; //心跳間隔時(shí)間
        private BufferedInputStream bis;
        private BufferedOutputStream bos;
        private ReadThread mReadThread;
        private Handler mHandler = new Handler();
        private Socket mSocket;
        private ExecutorService mExecutorService;
        private int tryCount = 0;//重試次數(shù)

        public void startConnect() {
            //在子線程進(jìn)行網(wǎng)絡(luò)操作
            // Service也是運(yùn)行在主線程,千萬(wàn)不要以為Service意思跟后臺(tái)運(yùn)行很像哪自,就以為Service運(yùn)行在后臺(tái)子線程
            if (mExecutorService == null) {
                mExecutorService = Executors.newCachedThreadPool();
            }
            mExecutorService.execute(connectRunnable);
        }

        private Runnable connectRunnable = new Runnable() {
            @Override
            public void run() {
                try {
                    // 建立Socket連接
                    mSocket = new Socket();
                    mSocket.connect(new InetSocketAddress("192.168.1.186", 8292), 10);
                    bis = new BufferedInputStream(mSocket.getInputStream());
                    bos = new BufferedOutputStream(mSocket.getOutputStream());
                    // 創(chuàng)建讀取服務(wù)器心跳的線程
                    mReadThread = new ReadThread();
                    mReadThread.start();
                    //開(kāi)啟心跳,每隔15秒鐘發(fā)送一次心跳
                    mHandler.post(mHeartRunnable);
                    tryCount = 1;
                } catch (Exception e) {
                    tryCount ++ ;
                    e.printStackTrace();
                    Log.d(TAG, "Socket連接建立失敗,正在嘗試第"+ tryCount + "次重連");
                    mHandler.postDelayed(new Runnable() {
                        @Override
                        public void run() {
                            mExecutorService.execute(connectRunnable);
                        }
                    },mHeart_spacetime);
                }
            }
        };

        public class ReadThread extends Thread {
            @Override
            public void run() {
                int size;
                byte[] buffer = new byte[1024];
                try {
                    while ((size = bis.read(buffer)) != -1) {
                        String str = new String(buffer, 0, size);
                        Log.d(TAG,"我收到來(lái)自服務(wù)器的消息: " +str);
                        //收到心跳消息以后丰包,首先移除斷連消息,然后創(chuàng)建一個(gè)新的60秒后執(zhí)行斷連的消息壤巷。
                        //這樣每次收到心跳后都會(huì)重新創(chuàng)建一個(gè)60秒的延時(shí)消息烫沙,在60秒后還沒(méi)收到心跳消息,表明服務(wù)器已死隙笆,就會(huì)執(zhí)行斷開(kāi)Socket連接
                        //在60秒鐘內(nèi)如果收到過(guò)一次心跳消息锌蓄,就表明服務(wù)器還活著,可以繼續(xù)與之通訊撑柔。
                        mHandler.removeCallbacks(disConnectRunnable);
                        mHandler.postDelayed(disConnectRunnable, mHeart_spacetime * 40);
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }

        private Runnable mHeartRunnable = new Runnable() {
            @Override
            public void run() {
                sendData();
            }
        };

        private void sendData() {
            mExecutorService.execute(new Runnable() {
                @Override
                public void run() {
                    try {
                        bos.write("給你一張過(guò)去的CD,聽(tīng)聽(tīng)那時(shí)我們的愛(ài)情瘸爽!".getBytes());
                        //一定不能忘記這步操作
                        bos.flush();
                        //發(fā)送成功以后,重新建立一個(gè)心跳消息
                        mHandler.postDelayed(mHeartRunnable, mHeart_spacetime);
                        Log.d(TAG, "我發(fā)送給服務(wù)器的消息: 給你一張過(guò)去的CD,聽(tīng)聽(tīng)那時(shí)我們的愛(ài)情铅忿!");
                    } catch (Exception e) {
                        e.printStackTrace();
                        Log.d(TAG, "心跳任務(wù)發(fā)送失敗剪决,正在嘗試第"+ tryCount + "次重連");
                        //mExecutorService.schedule(connectRunnable,mHeart_spacetime, TimeUnit.SECONDS);
                        mExecutorService.execute(connectRunnable);
                    }
                }
            });
        }

        private Runnable disConnectRunnable = new Runnable() {
            @Override
            public void run() {
                disConnect();
            }
        };

        private void disConnect() {
            mExecutorService.execute(new Runnable() {
                @Override
                public void run() {
                    try {
                        Log.d(TAG, "正在執(zhí)行斷連: disConnect");
                        //執(zhí)行Socket斷連
                        mHandler.removeCallbacks(mHeartRunnable);
                        if (mReadThread != null) {
                            mReadThread.interrupt();
                        }

                        if (bos != null) {
                            bos.close();
                        }

                        if (bis != null) {
                            bis.close();
                        }

                        if (mSocket != null) {
                            mSocket.shutdownInput();
                            mSocket.shutdownOutput();
                            mSocket.close();
                        }
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            });
        }
    }
}

服務(wù)端代碼:

/**
 * Create by Fiora on 2018/10/24 0024
 */
public class AppServer {
    public static final String TAG = AppServer.class.getSimpleName();
    private static BufferedOutputStream bos;
    private static BufferedInputStream bis;
    private static Socket acceptSocket;

    public static void main (String args[]){
        try{
            ServerSocket serverSocket  = new ServerSocket(8292);
            while(true) {
                acceptSocket = serverSocket.accept();
                bos = new BufferedOutputStream(acceptSocket.getOutputStream());
                bis = new BufferedInputStream(acceptSocket.getInputStream());

                ReadThread readThread = new ReadThread();
                readThread.start();
            }
        }catch (Exception e){
            e.printStackTrace();
        }
    }

    private static class ReadThread extends Thread {
        @Override
        public void run() {
            while (true) {
                byte[] data = new byte[1024];
                int size = 0;
                try {
                    while ((size = bis.read(data)) != -1) {
                        String str = new String(data,0,size);
                        System.out.println(TAG+"----"+str);
                        //收到客戶端發(fā)送的請(qǐng)求后,立馬回一條心跳給客戶端
                        bos.write("有時(shí)會(huì)突然忘了檀训,我依然愛(ài)著你柑潦!".getBytes());
                        bos.flush();
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                    break;
                }
            }
        }
    }
}

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市峻凫,隨后出現(xiàn)的幾起案子渗鬼,更是在濱河造成了極大的恐慌,老刑警劉巖荧琼,帶你破解...
    沈念sama閱讀 206,839評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件譬胎,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡命锄,警方通過(guò)查閱死者的電腦和手機(jī)堰乔,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)脐恩,“玉大人镐侯,你說(shuō)我怎么就攤上這事∈幻埃” “怎么了苟翻?”我有些...
    開(kāi)封第一講書(shū)人閱讀 153,116評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵搭伤,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我袜瞬,道長(zhǎng)怜俐,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,371評(píng)論 1 279
  • 正文 為了忘掉前任邓尤,我火速辦了婚禮拍鲤,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘汞扎。我一直安慰自己季稳,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,384評(píng)論 5 374
  • 文/花漫 我一把揭開(kāi)白布澈魄。 她就那樣靜靜地躺著景鼠,像睡著了一般。 火紅的嫁衣襯著肌膚如雪痹扇。 梳的紋絲不亂的頭發(fā)上铛漓,一...
    開(kāi)封第一講書(shū)人閱讀 49,111評(píng)論 1 285
  • 那天,我揣著相機(jī)與錄音鲫构,去河邊找鬼浓恶。 笑死,一個(gè)胖子當(dāng)著我的面吹牛结笨,可吹牛的內(nèi)容都是我干的包晰。 我是一名探鬼主播,決...
    沈念sama閱讀 38,416評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼炕吸,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼伐憾!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起赫模,我...
    開(kāi)封第一講書(shū)人閱讀 37,053評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤树肃,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后嘴瓤,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體扫外,經(jīng)...
    沈念sama閱讀 43,558評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,007評(píng)論 2 325
  • 正文 我和宋清朗相戀三年廓脆,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片磁玉。...
    茶點(diǎn)故事閱讀 38,117評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡停忿,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出蚊伞,到底是詐尸還是另有隱情席赂,我是刑警寧澤吮铭,帶...
    沈念sama閱讀 33,756評(píng)論 4 324
  • 正文 年R本政府宣布,位于F島的核電站颅停,受9級(jí)特大地震影響谓晌,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜癞揉,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,324評(píng)論 3 307
  • 文/蒙蒙 一纸肉、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧喊熟,春花似錦柏肪、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,315評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至壁拉,卻和暖如春谬俄,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背弃理。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,539評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工凤瘦, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人案铺。 一個(gè)月前我還...
    沈念sama閱讀 45,578評(píng)論 2 355
  • 正文 我出身青樓蔬芥,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親控汉。 傳聞我的和親對(duì)象是個(gè)殘疾皇子笔诵,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,877評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)姑子,斷路器乎婿,智...
    卡卡羅2017閱讀 134,600評(píng)論 18 139
  • 鈞鈺反思日志0610-2017 必讀:12條人生原則 早睡早起——晚不超過(guò)22:30,早不晚于6:00街佑,醒后不睡回...
    鈞鈺閱讀 167評(píng)論 0 0
  • 中心圖是一個(gè)穿裙子的小女孩谢翎,因?yàn)樽约罕容^喜歡穿裙子,感覺(jué)這個(gè)卡通人物很符合我沐旨。 四個(gè)分叉點(diǎn)主要概括了我的愛(ài)好森逮、專業(yè)...
    美院王之?huà)?/span>閱讀 486評(píng)論 1 0
  • 判斷用到的例子 var a = "iamstring.";var b = 222;var c= [1,2,3];v...
    瘋?cè)嗽傅寞傃辕傉Z(yǔ)閱讀 242評(píng)論 0 0
  • 老人們常說(shuō)“人越閑,做的事情越是無(wú)用功” 以前不懂 總覺(jué)得“回味”一件事是會(huì)讓人反省的 現(xiàn)在卻慢慢明白了 有很多事...
    不停雜貨鋪閱讀 97評(píng)論 0 0