欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java利用Socket實(shí)現(xiàn)網(wǎng)絡(luò)通信功能

 更新時(shí)間:2023年11月22日 08:27:16   作者:一一哥Sun  
在早期的網(wǎng)絡(luò)編程中,Socket是很常見的實(shí)現(xiàn)技術(shù)之一,比如早期的聊天室,就是基于這種技術(shù)進(jìn)行實(shí)現(xiàn)的,另外現(xiàn)在有些消息推送,也可以基于Socket實(shí)現(xiàn),本文小編給大家介紹了Java利用Socket實(shí)現(xiàn)網(wǎng)絡(luò)通信功能的示例,需要的朋友可以參考下

一. Socket編程

1. 簡(jiǎn)介

Socket編程是基于TCP/IP協(xié)議的網(wǎng)絡(luò)編程技術(shù),它給我們提供了一種可以用于網(wǎng)絡(luò)通信的機(jī)制,讓我們能在不同的計(jì)算機(jī)之間進(jìn)行數(shù)據(jù)交換。我們可以利用Socket編程實(shí)現(xiàn)客戶端/服務(wù)器程序的開發(fā),如聊天室、FTP客戶端等項(xiàng)目。

2. 通信流程

對(duì)于Socket編程,我們要重點(diǎn)搞清楚客戶端與服務(wù)器端之間的通信流程。

在Socket編程中有兩種類型的Socket:服務(wù)器Socket和客戶端Socket。服務(wù)器Socket可以在服務(wù)器上創(chuàng)建用于監(jiān)聽客戶端請(qǐng)求的端口,客戶端Socket則可以在客戶端上創(chuàng)建用于連接服務(wù)器的Socket??蛻舳薙ocket向服務(wù)器Socket發(fā)送請(qǐng)求,服務(wù)器Socket可以接收該請(qǐng)求,并創(chuàng)建一個(gè)新的Socket用于與客戶端通信。通過這種方式,客戶端和服務(wù)器端之間就可以進(jìn)行數(shù)據(jù)交換。

在Socket編程中,客戶端和服務(wù)器之間會(huì)通過TCP/IP協(xié)議進(jìn)行通信??蛻舳薙ocket首先回連接到服務(wù)器Socket的IP地址和端口號(hào),這樣就會(huì)建立起一個(gè)TCP連接。一旦連接建立,客戶端和服務(wù)器之間就可以進(jìn)行數(shù)據(jù)傳輸。數(shù)據(jù)會(huì)被分割成數(shù)據(jù)包,并通過TCP/IP協(xié)議在客戶端和服務(wù)器之間傳輸。

然后服務(wù)器Socket會(huì)在服務(wù)器端創(chuàng)建一個(gè)用于監(jiān)聽客戶端請(qǐng)求的端口,客戶端Socket則在客戶端上創(chuàng)建用于連接服務(wù)器的Socket。客戶端Socket向服務(wù)器Socket發(fā)送連接請(qǐng)求,服務(wù)器Socket接收該請(qǐng)求并創(chuàng)建一個(gè)新的Socket用于與客戶端通信。通過這種方式,客戶端和服務(wù)器端之間就可以進(jìn)行數(shù)據(jù)交換。

這樣,客戶端和服務(wù)器端之間就通過TCP/IP協(xié)議實(shí)現(xiàn)了Socket通信。

3. 核心API

我們可以使用java.net包中的API來實(shí)現(xiàn)Socket編程,這些常用的API包括:

  • ServerSocket類:用于創(chuàng)建服務(wù)器端Socket,監(jiān)聽客戶端發(fā)來的請(qǐng)求。
  • Socket類:用于創(chuàng)建客戶端Socket,可以連接服務(wù)器Socket。
  • InputStream和OutputStream類:用于在Socket之間傳輸數(shù)據(jù)的輸入輸出流。

4. 基本案例-單向通信

接下來先給大家編寫一個(gè)可以實(shí)現(xiàn)單向通信的基本案例,包括一個(gè)服務(wù)器端和一個(gè)客戶端。這個(gè)案例中,客戶端給服務(wù)器端發(fā)送一條消息,然后服務(wù)器把客戶端發(fā)來的消息打印出來,代碼如下:

4.1 服務(wù)器端

下面是服務(wù)器端的代碼案例。在本案例中,主要是利用ServerSocket對(duì)象定義了一個(gè)服務(wù)器,并通過accept()方法來獲取與該服務(wù)器綁定的Socket客戶端對(duì)象。這里要注意端口號(hào)“1234”是我們自己隨意定義的,只要不與其他程序的端口號(hào)重合即可。然后我們可以在Socket客戶端對(duì)象上通過getInputStream()方法來獲取一個(gè)InputStream輸入流,進(jìn)而讀取客戶端發(fā)來的消息。最后大家要記得把各種IO流資源釋放。

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.ServerSocket;
import java.net.Socket;

/**
 * @author 一一哥Sun
 * @company 千鋒教育
 */
public class MySocketServer {
    public static void main(String[] args) throws IOException {
        //設(shè)置服務(wù)器的端口號(hào)
        int portNumber = 1234;
        // 創(chuàng)建服務(wù)器套接字并綁定端口號(hào)
        ServerSocket serverSocket = new ServerSocket(portNumber);
        //獲取與ServerSocket關(guān)聯(lián)的Socket客戶端對(duì)象
        Socket acceptSocket = serverSocket.accept();

        // 服務(wù)器接收客戶端發(fā)來的消息
        InputStream serverInputStream = acceptSocket.getInputStream();
        BufferedReader in = new BufferedReader(new InputStreamReader(serverInputStream));
        String response = in.readLine();
        if (response != null) {
            System.out.println("服務(wù)器接收客戶端發(fā)來的消息===>: " + response);
        }

        // 關(guān)閉套接字和流
        serverSocket.close();
        acceptSocket.close();
        serverInputStream.close();
    }
}

大家要注意,在今天的案例中,沒有使用循環(huán)來一直進(jìn)行消息的收發(fā),如果我們想實(shí)現(xiàn)不間斷的消息收發(fā),可以把相關(guān)代碼寫在循環(huán)體中。所以在今天的案例中,消息收發(fā)一次之后,項(xiàng)目就會(huì)停掉。

4.2 客戶端

接著又定義了一個(gè)客戶端程序。在該程序中,定義了一個(gè)Socket客戶端對(duì)象,該對(duì)象通過IP地址和端口號(hào)來關(guān)聯(lián)服務(wù)端對(duì)象。因?yàn)?strong>小編的項(xiàng)目,客戶端和服務(wù)端是在同一臺(tái)機(jī)器上,所以這里的IP地址我們可以使用localhost,當(dāng)然也可以用服務(wù)端的實(shí)際IP地址。然后利用字符流和字節(jié)輸出流給服務(wù)端發(fā)送信息,最后釋放IO流資源。

import java.io.IOException;
import java.io.PrintWriter;
import java.net.Socket;
import java.util.Scanner;

/**
 * @author 一一哥Sun
 * @company 千鋒教育
 */
public class MySocketClient {
    public static void main(String[] args) throws IOException {
        try {
            // 創(chuàng)建Socket對(duì)象并連接到服務(wù)器,關(guān)聯(lián)服務(wù)器的ip地址與端口號(hào)
            Socket socket = new Socket("localhost", 1234);

            // 創(chuàng)建輸入輸出流,通過套接字發(fā)送和接收數(shù)據(jù)
            Scanner scanner = new Scanner(System.in);
            String nextLine = scanner.nextLine();
            // 輸出流,客戶端向服務(wù)器發(fā)送消息
            PrintWriter out = new PrintWriter(socket.getOutputStream(), true);
            out.println(nextLine);

            // 關(guān)閉套接字和流
            out.close();
            scanner.close();
            socket.close();
            } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

在這個(gè)案例中,主要是利用ServerSocket、Socket和IO流這三種API,就實(shí)現(xiàn)了客戶端給服務(wù)端發(fā)送消息的功能。但是這個(gè)案例中,我們只能是客戶端給服務(wù)端發(fā)送消息,服務(wù)端卻不能給客戶端回復(fù)消息,所以接下來要把這個(gè)案例改進(jìn)一下,實(shí)現(xiàn)客戶端與服務(wù)端之間互相傳遞消息。

5. 改進(jìn)案例-雙向通信

接下來我們就把上面的案例改進(jìn)一下,實(shí)現(xiàn)客戶端與服務(wù)器端之間的雙向通信,即客戶端給服務(wù)器端發(fā)送消息,服務(wù)器端收到消息之后,再給客戶端返回消息。

5.1 服務(wù)器端

在下面的代碼中,小編把服務(wù)器端的代碼改進(jìn)了一下,在if語句中增加了利用OutputStream向客戶端返回信息的代碼,如下所示:

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;

/**
 * @author 一一哥Sun
 * @company 千鋒教育
 */
public class MyServer {
    public static void main(String[] args) throws IOException {
        int portNumber = 1234;
        // 創(chuàng)建服務(wù)器套接字并綁定端口號(hào)
        ServerSocket serverSocket = new ServerSocket(portNumber);
        // 注意:服務(wù)器只能接受一次客戶端!用該socket對(duì)象既可以接收客戶端發(fā)來的消息,也可以給客戶端回復(fù)消息
        Socket acceptSocket = serverSocket.accept();

        // 服務(wù)器接收客戶端發(fā)來的消息
        InputStream serverInputStream = acceptSocket.getInputStream();
        BufferedReader in = new BufferedReader(new InputStreamReader(serverInputStream));
        String response = in.readLine();
        if (response != null) {
            System.out.println("服務(wù)器接收客戶端發(fā)來的消息===>: " + response);

            // 服務(wù)器向客戶端發(fā)送響應(yīng)信息
            OutputStream serverOutputStream = acceptSocket.getOutputStream();
            String serverResponse = "我是服務(wù)器,你的消息已收到!";
            serverOutputStream.write(serverResponse.getBytes());
            serverOutputStream.flush();
            serverOutputStream.close();
        }

        // 關(guān)閉套接字和流
        serverSocket.close();
        acceptSocket.close();
        serverInputStream.close();
        //serverOutputStream.close();
    }
}

5.2 客戶端

在下面的代碼中,把客戶端的代碼也改進(jìn)了一下,主要是增加了利用BufferedReader和InputStream,讀取服務(wù)器返回信息的代碼,如下所示:

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.Socket;
import java.util.Scanner;

/**
 * @author 一一哥Sun
 * @company 千鋒教育
 */
public class MyClient {
    public static void main(String[] args) throws IOException {
        try {
            //創(chuàng)建Socket對(duì)象并連接到服務(wù)器
            Socket socket = new Socket("localhost", 1234);

            //創(chuàng)建輸入輸出流,通過套接字發(fā)送和接收數(shù)據(jù)
            Scanner scanner=new Scanner(System.in);
            String nextLine = scanner.nextLine();
            //輸出流,客戶端向服務(wù)器發(fā)送消息
            PrintWriter out = new PrintWriter(socket.getOutputStream(), true);
            out.println(nextLine);

            //輸入流,客戶端從服務(wù)器接收消息
            BufferedReader in = new BufferedReader(new InputStreamReader(socket.getInputStream()));
            String response = in.readLine();
            System.out.println("Server服務(wù)器返回的響應(yīng)信息===>: " + response);

            //關(guān)閉套接字和流
            out.close();
            in.close();
            scanner.close();
            socket.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

經(jīng)過以上代碼的改造,我們就實(shí)現(xiàn)了服務(wù)器與客戶端之間的雙向通信,大家可以把小編的代碼案例運(yùn)行起來看看效果。

6. 注意事項(xiàng)

大家要注意,在進(jìn)行Socket通信開發(fā)時(shí),稍有不慎可能就會(huì)出現(xiàn)各種問題,所以我們要注意以下事項(xiàng):

  • 在每個(gè)Socket連接中,我們都應(yīng)該嚴(yán)格按照協(xié)議規(guī)定的格式進(jìn)行數(shù)據(jù)的發(fā)送和接收,否則就可能會(huì)導(dǎo)致數(shù)據(jù)傳輸失敗或被誤解。
  • 在Socket編程中,應(yīng)該使用多線程或異步機(jī)制等技術(shù)來避免阻塞。如果阻塞時(shí)間過長(zhǎng),可能會(huì)導(dǎo)致客戶端或服務(wù)器崩潰。在上面的案例中,是把收發(fā)消息的代碼之間寫在了主線程中,其實(shí)我們可以把這種耗時(shí)的操作放在Thread或線程池中進(jìn)行實(shí)現(xiàn)。
  • 服務(wù)器端應(yīng)該可以同時(shí)處理多個(gè)客戶端的請(qǐng)求,否則有可能會(huì)導(dǎo)致客戶端的請(qǐng)求被拒絕或服務(wù)器崩潰。
  • 我們要保證Socket編程的保安全性,例如防止黑客攻擊、拒絕服務(wù)攻擊等惡意行為。
  • 我們還應(yīng)該有適當(dāng)?shù)腻e(cuò)誤處理,例如必須處理網(wǎng)絡(luò)連接失敗、數(shù)據(jù)傳輸?shù)氖〉犬惓G闆r。

除了以上幾點(diǎn)注意事項(xiàng),還有其他一些細(xì)節(jié)需要注意。在Socket編程中,必須對(duì)網(wǎng)絡(luò)通信有深入的理解和掌握,才能確保程序的正確性和安全性。

二. 結(jié)語

今天的文章,主要是給大家介紹了Socket編程的實(shí)現(xiàn)過程。其實(shí)Socket通信主要就是分為服務(wù)器端和客戶端,有著比較清晰的實(shí)現(xiàn)邏輯,學(xué)習(xí)起來并不難,大家重點(diǎn)理解和掌握Socket通信的實(shí)現(xiàn)流暢即可。

相關(guān)文章

  • Spring核心方法refresh的使用解析

    Spring核心方法refresh的使用解析

    這篇文章主要介紹了Spring核心方法refresh的使用,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • SpringBoot中MyBatis-Plus 查詢時(shí)排除某些字段的操作方法

    SpringBoot中MyBatis-Plus 查詢時(shí)排除某些字段的操作方法

    這篇文章主要介紹了SpringBoot中MyBatis-Plus 查詢時(shí)排除某些字段的操作方法,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-08-08
  • win10安裝JDK14.0.2的詳細(xì)安裝過程

    win10安裝JDK14.0.2的詳細(xì)安裝過程

    這篇文章主要介紹了win10安裝JDK14.0.2的詳細(xì)安裝過程的相關(guān)資料,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-09-09
  • Elasticsearch寫入瓶頸導(dǎo)致skywalking大盤空白

    Elasticsearch寫入瓶頸導(dǎo)致skywalking大盤空白

    這篇文章主要為大家介紹了Elasticsearch寫入瓶頸導(dǎo)致skywalking大盤空白的解決方案,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步
    2022-02-02
  • SpringBoot實(shí)現(xiàn)elasticsearch 查詢操作(RestHighLevelClient 的案例實(shí)戰(zhàn))

    SpringBoot實(shí)現(xiàn)elasticsearch 查詢操作(RestHighLevelClient 

    這篇文章主要給大家介紹了SpringBoot如何實(shí)現(xiàn)elasticsearch 查詢操作,文中有詳細(xì)的代碼示例和操作流程,具有一定的參考價(jià)值,需要的朋友可以參考下
    2023-07-07
  • Java 實(shí)現(xiàn)限流器處理Rest接口請(qǐng)求詳解流程

    Java 實(shí)現(xiàn)限流器處理Rest接口請(qǐng)求詳解流程

    在工作中是否會(huì)碰到這樣的場(chǎng)景,高并發(fā)的請(qǐng)求但是無法全部執(zhí)行,需要一定的限流。如果你是使用的微服務(wù)框架,比如SpringCloud,可以使用Gateway增加限流策略來解決。本篇文章是在沒有框架的情況實(shí)現(xiàn)限流器
    2021-11-11
  • 一個(gè)@Component注解引發(fā)的大坑

    一個(gè)@Component注解引發(fā)的大坑

    這篇文章主要介紹了一個(gè)@Component注解引發(fā)的大坑,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-09-09
  • SpringBoot啟動(dòng)時(shí)自動(dòng)執(zhí)行代碼的幾種實(shí)現(xiàn)方式

    SpringBoot啟動(dòng)時(shí)自動(dòng)執(zhí)行代碼的幾種實(shí)現(xiàn)方式

    這篇文章主要給大家介紹了關(guān)于SpringBoot啟動(dòng)時(shí)自動(dòng)執(zhí)行代碼的幾種實(shí)現(xiàn)方式,文中通過實(shí)例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2022-02-02
  • Spring?AI?+?混元帶你實(shí)現(xiàn)企業(yè)級(jí)穩(wěn)定可部署的AI業(yè)務(wù)智能體

    Spring?AI?+?混元帶你實(shí)現(xiàn)企業(yè)級(jí)穩(wěn)定可部署的AI業(yè)務(wù)智能體

    我們深入探討了Spring?AI在智能體構(gòu)建中的實(shí)際應(yīng)用,特別是在企業(yè)環(huán)境中的價(jià)值與效能,通過逐步實(shí)現(xiàn)一個(gè)本地部署的智能體解決方案,我們不僅展示了Spring?AI的靈活性與易用性,還強(qiáng)調(diào)了它在推動(dòng)AI技術(shù)與業(yè)務(wù)深度融合方面的潛力,感興趣的朋友一起看看吧
    2024-11-11
  • 聊聊springboot2.2.3升級(jí)到2.4.0單元測(cè)試的區(qū)別

    聊聊springboot2.2.3升級(jí)到2.4.0單元測(cè)試的區(qū)別

    這篇文章主要介紹了springboot 2.2.3 升級(jí)到 2.4.0單元測(cè)試的區(qū)別,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-10-10

最新評(píng)論