scoket用法

来源:https://www.cnblogs.com/lihongtaoya/archive/2023/03/31/17276968.html
-Advertisement-
Play Games

一.scoket基本介紹 1.scoket簡介(以下是來自chatgpt回答) 1)Socket(套接字)是電腦網路中用於描述主機之間通信的一種機制。它定義了一種標準的介面, 使得應用程式可以利用網路傳輸層提供的服務(如TCP或UDP)進行通信。 2)Socket的作用是在網路應用程式之間提供數據 ...


一.scoket基本介紹

1.scoket簡介(以下是來自chatgpt回答)

1)Socket(套接字)是電腦網路中用於描述主機之間通信的一種機制。它定義了一種標準的介面,

      使得應用程式可以利用網路傳輸層提供的服務(如TCP或UDP)進行通信。

2)Socket的作用是在網路應用程式之間提供數據傳輸服務。通過Socket,應用程式可以將數據發送

      到網路上的另一個應用程式,並從網路上的另一個應用程式接收數據。

3)Socket還提供了一種機制,使得應用程式可以接收來自網路上的請求,並對這些請求進行響應。

     常見的網路應用程式,如Web伺服器、郵件伺服器、FTP伺服器等都使用Socket技術。此外,許多P2P

   (點對點)應用程式,如BitTorrent、eMule等也使用Socket技術。Socket還可以用於實現各種網路應用

      程式,如網路游戲、聊天程式等。

2.建立TCP/IP連接

1)創建一個scoket對象

socket_stream = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

第一個參數表示通信協議類型

AF_INET:適用於網路通信

AF_UNIX:適用於本地進程間通信

第二個參數表示創建什麼連接

SOCK_STREAM:表示tcp/ip連線

SOCK_DGRAM:表示udp連接

2)與服務端建立連接

# 建立連接
ip_port = ("ip", "port")  # 連接的服務端ip和埠
socket_stream.connect(ip_port)  # 建立連接,出錯不返回錯誤碼
socket_stream.connect_ex(ip_port)  # 出錯時返回錯誤碼,不拋異常

connect()和connect_ex()選一個即可

3)發送數據

# 發送數據
data = 'hello'
socket_stream.send(data.encode("utf-8"))  # 發送TCP數據,當send在待發送數據量大於己端緩存區剩餘空間時,數據丟失,不會發完
# or
socket_stream.sendall(data.encode("utf-8"))  # 發送完整的TCP數據(迴圈調用send,sendall在待發送數據量大於己端緩存區剩餘空間時,數據不丟失,迴圈調用send直到發完)

當發送的數據較大時建議使用sendall發送

4)接收TCP數據

# 接收TCP數據。1024(變數)表示每次最多接受1M位元組的數據,recv()函數是一個阻塞函數,沒有要接收的數據時,會一直等待,直到接收到數據或出現錯誤才會返回
response_data = socket_stream.recv(1024)

5)關閉連接

# 關閉連接
socket_stream.close()

3.建立UDP連接

1)創建scoket對象

client_dgram = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)

2)發送數據

# 發送數據
ip_port = ('127.0.0.1', 8080)
data = 'Hello'
client_dgram.sendto(data.encode("utf-8"), ip_port)

這裡發送用的是sendto()方法

3)接收數據

# 接收UDP數據數據
data = client_dgram.recvfrom(4096)

4)關閉連接

# 關閉連接
client_dgram.close()

二.使用scoket編寫聊天程式

以下部分代碼參考:https://www.yuque.com/imhelloworld/nov9az/bffcea259d3c96fb17a130acebc12801

聊天程式的話涉及兩個端,既server和click。這裡直接貼代碼了,相關註釋都是代碼里

1)server

import socket
import threading

# 伺服器IP地址和埠號
SERVER_IP = 'IP地址'
SERVER_PORT = 埠號
# 創建一個socket對象
server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 綁定IP地址和埠號
server_socket.bind((SERVER_IP, SERVER_PORT))
# 設置最大連接數,超過10後拒絕。0的話表示不接受連接,直接拒絕
server_socket.listen(10)
# 客戶端列表
clients = []


def handle_client(client_socket):
    while True:
        try:
            # 接收客戶端發送的消息
            message = client_socket.recv(10240).decode('utf-8')
            print("列印", message)
            if message == 'quert':
                client_socket.send(message.encode('utf-8'))
            else:
                # 廣播消息給所有客戶端
                for c in clients:
                    if c != client_socket:
                        c.send(message.encode('utf-8'))
        except:
            # 發生異常時關閉連接
            client_socket.close()
            clients.remove(client_socket)
            print("發送異常了")
            break


# 迴圈監聽客戶端連接
while True:
    print('Waiting for client connections...')
    # 接受客戶端連接請求,新的客戶端請求時創建一個新的socket,用於處理客戶端的請求,原有的socket繼續監聽其他客戶端的連接請求;
    # 函數是一個阻塞函數,當沒有客戶端連接請求時,會一直等待,直到有客戶端連接請求到達
    client_socket, client_address = server_socket.accept()
    print('客戶socket', client_socket, 'ip和埠', client_address)
    clients.append(client_socket)
    # 創建一個新的線程處理客戶端連接
    client_thread = threading.Thread(target=handle_client, args=(client_socket, ))
    client_thread.start()

2)click

import socket
import threading
import time

# 伺服器IP地址和埠號
SERVER_IP = 'IP地址'
SERVER_PORT = 埠號
# 創建一個socket對象
client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 連接伺服器
client_socket.connect((SERVER_IP, SERVER_PORT))


def receive():
    while True:
        try:
            # 接收伺服器發送的消息
            message = client_socket.recv(1024).decode('utf-8')
            if message == 'quert':
                client_socket.close()
                break
            print('\n對方回答:', message)
        except:
            # 發生異常時關閉連接
            client_socket.close()
            break


def send():
    while True:
        # 獲取用戶輸入的消息
        message = input()
        if message == 'quert':
            client_socket.close()
            print('連接斷開,通信結束')
            break
        else:
            # 發送消息給伺服器
            client_socket.send(message.encode('utf-8'))


# 創建兩個線程,分別用於接收和發送消息
receive_thread = threading.Thread(target=receive)
send_thread = threading.Thread(target=send)
# 啟動線程
receive_thread.start()
send_thread.start()

這裡需要註意的是,既然是聊天程式,那麼肯定涉及兩個click,兩個click代碼一致即可,運行時先運行server,後運行click,之後就可以在Terminal下建立對話了。

以上是click在同一臺電腦上,若不在同一臺電腦的,建立對話需在同一個區域網下,ip地址在dos視窗下輸入ipconfig\all即可查看,埠號可隨便填個,只要不被占用就行。

三.scoket+chatgpt建立搜索對話框

1)獲取chatgpt搜索介面,這裡就直接貼代碼了

def api(params, tokened="token"):
    url = "https://api.aigcfun.com/api/v1/text?key=" + tokened
    data = {"messages": [{"role": "system", "content": "請以markdown的形式返回答案"},
                         {"role": "user", "content": params}], "tokensLength": 32, "model": "gpt-3.5-turbo"}
    response_data = requests.post(url=url, json=data)
    return response_data.json()["choices"][0]["text"]

params為用戶問的問題,return返回的是chatgpt回答的答案

2)獲取到chatgpt的回答內容後返回給服務端

def receive():
    while True:
        try:
            # 接收伺服器發送的消息
            message = client_socket.recv(1024).decode('utf-8')
            data_message = api(message)
            while data_message == None:  # 由於chatgpt有時響應的比較慢,這裡需添加個迴圈判斷
                continue
            # if data_message == '您今日使用次數已達上限,請明日再試!':
            #     data_message = api(params=message, tokened=token())

            client_socket.send(data_message.encode('utf-8'))  # 發送chatgpt回答的問題給服務端
        except:
            # 發生異常時關閉連接
            client_socket.close()
            break

3)目前chatpgt雖是免費的,但每天都有次數限制,所以可以請求獲取token的介面來獲取新的token。這裡就不做演示了(hhhhhh.........)

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 模型之間的關係(Relations Between Models) 上一章介紹了為包含基本欄位的模型創建自定義視圖。然而,在任何真實的業務場景中,我們都需要不止一個模型。此外,模型之間的鏈接是必要的。人們可以很容易地想象一個模型包含客戶,另一個模型則包含用戶列表。你可能需要參考任何現有業務模型上的客 ...
  • ChatGPT是一個基於GPT-3.5架構的自然語言處理工具,它具有文本生成、文本分類、對話生成等多種能力。作為一種強大的自然語言處理工具,ChatGPT可以應用於智能客服、智能問答、內容創作等多個領域。如果您對ChatGPT感興趣,可以通過關註本公眾號瞭解更多信息,並體驗基於ChatGPT的小程式... ...
  • Spring Spring為簡化開發而生,讓程式員只關心核心業務的實現,儘可能的不在關註非業務邏輯代碼(事務控制,安全日誌等)。 1,Spring八大模塊 這八大模塊組成了Spring 1.1 Spring Core模塊 這是Spring框架的最基礎的部分,它提供了依賴註入(DependencyIn ...
  • React Native 備忘清單 適合初學者的綜合 React Native 備忘清單,在開始 React Native 之前需要先掌握 react 庫入門,為開發人員分享快速參考備忘單。 React Native (簡稱RN)是Facebook於2015年4月開源的跨平臺移動應用開發框架,是Fa ...
  • React 備忘清單 IT寶庫整理適合初學者入門的React開發速查備忘清單,為開發人員分享快速參考備忘單。 React是用於構建用戶界面的JavaScript庫,起源於Facebook的內部項目,該公司對市場上所有 JavaScript MVC框架都不滿意,決定自行開發一套,用於架設Instagr ...
  • 一、延遲計算 RDD 代表的是分散式數據形態,因此,RDD 到 RDD 之間的轉換,本質上是數據形態上的轉換(Transformations) 在 RDD 的編程模型中,一共有兩種運算元,Transformations 類運算元和 Actions 類運算元。開發者需要使用 Transformations ...
  • 一 回顧trait使用 https://blog.csdn.net/bushuwei/article/details/103514174發現之前本人說明很模糊,自己居然不知道為什麼其實這裡的$c,就是class B再次回顧邏輯 二 分析 self和static區別說的沒毛病 Trait基類use t ...
  • 一、函數的定義 可以分為以下兩種: 1、函數聲明和函數定義分離 這種方法將函數聲明和函數定義分開,通常在頭文件中先聲明函數原型,然後在源文件中實現函數定義。 例如,頭文件 example.h 中聲明瞭一個函數 add: #ifndef EXAMPLE_H #define EXAMPLE_H int ...
一周排行
    -Advertisement-
    Play Games
  • ## 引言 最近發現自己喜歡用的 Todo 軟體總是差點意思,畢竟每個人的習慣和工作流不太一樣,我就想著自己寫一個小的[Todo 項目]( https://github.com/circler3/TodoTrack ),核心的功能是自動記錄 Todo 執行過程中消耗的時間(尤其面向程式員),按照自己 ...
  • ### 前言 當我們編寫 C# 代碼時,經常需要處理大量的數據集合。在傳統的方式中,我們往往需要先將整個數據集合載入到記憶體中,然後再進行操作。但是如果數據集合非常大,這種方式就會導致記憶體占用過高,甚至可能導致程式崩潰。 C# 中的`yield return`機制可以幫助我們解決這個問題。通過使用`y ...
  • 1. ADO.NET的前世今生 ADO.NET的名稱起源於ADO(ActiveX Data Objects),是一個COM組件庫,用於在以往的Microsoft技術中訪問數據。之所以使用ADO.NET名稱,是因為Microsoft希望表明,這是在NET編程環境中優先使用的數據訪問介面。 ADO.NE ...
  • 1. 為什麼需要單元測試 在我們之前,測試某些功能是否能夠正常運行時,我們都將代碼寫到Main方法中,當我們測試第二個功能時,我們只能選擇將之前的代碼清掉,重新編寫。此時,如果你還想重新測試你之前的功能時,這時你就顯得有些難為情了,因為代碼都被你清掉了。當然你完全可以把代碼寫到一個記事本中進行記錄, ...
  • 1. 透過現象看本質 反射被譽為是 c#中的黑科技 ,在很多領域中都有反射的身影,例如,我們經常使用的ORM框架,ABP框架 等。 反射指程式可以訪問、檢測和修改它本身狀態或行為的一種能力。. 程式集包含模塊,而模塊包含類型,類型又包含成員。. 反射則提供了封裝程式集、模塊和類型的對象。. 您可以使 ...
  • # Rust Web 全棧開發之 Web Service 中的錯誤處理 ## Web Service 中的統一錯誤處理 ### Actix Web Service 自定義錯誤類型 -> 自定義錯誤轉為 HTTP Response - 資料庫 - 資料庫錯誤 - 串列化 - serde 錯誤 - I/ ...
  • 在前面的幾篇文章中,詳細地給大家介紹了Java里的集合。但在介紹集合時,我們涉及到了泛型的概念卻並沒有詳細學習,所以今天我們要花點時間給大家專門講解什麼是泛型、泛型的作用、用法、特點等內容 ...
  • ###BIO:同步阻塞 主線程發起io請求後,需要等待當前io操作完成,才能繼續執行。 ###NIO:同步非阻塞 引入selector、channel、等概念,當主線程發起io請求後,輪詢的查看系統是否準備好執行io操作,沒有準備好則主線程不會阻塞會繼續執行,準備好主線程會阻塞等待io操作完成。 # ...
  • 摘要:在讀多寫少的環境中,有沒有一種比ReadWriteLock更快的鎖呢?有,那就是JDK1.8中新增的StampedLock! 本文分享自華為雲社區《【高併發】高併發場景下一種比讀寫鎖更快的鎖》,作者: 冰 河。 什麼是StampedLock? ReadWriteLock鎖允許多個線程同時讀取共 ...
  • ## 併發與並行😣 ### 併發與並行的概念和區別 並行:同一個時間段內多個任務同時在不同的CPU核心上執行。強調同一時刻多個任務之間的”**同時執行**“。 併發:同一個時間段內多個任務都在進展。強調多個任務間的”**交替執行**“。 ![](https://img2023.cnblogs.co ...