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
  • 1、預覽地址:http://139.155.137.144:9012 2、qq群:801913255 一、前言 隨著網路的發展,企業對於信息系統數據的保密工作愈發重視,不同身份、角色對於數據的訪問許可權都應該大相徑庭。 列如 1、不同登錄人員對一個數據列表的可見度是不一樣的,如數據列、數據行、數據按鈕 ...
  • 前言 上一篇文章寫瞭如何使用RabbitMQ做個簡單的發送郵件項目,然後評論也是比較多,也是準備去學習一下如何確保RabbitMQ的消息可靠性,但是由於時間原因,先來說說設計模式中的簡單工廠模式吧! 在瞭解簡單工廠模式之前,我們要知道C#是一款面向對象的高級程式語言。它有3大特性,封裝、繼承、多態。 ...
  • Nodify學習 一:介紹與使用 - 可樂_加冰 - 博客園 (cnblogs.com) Nodify學習 二:添加節點 - 可樂_加冰 - 博客園 (cnblogs.com) 介紹 Nodify是一個WPF基於節點的編輯器控制項,其中包含一系列節點、連接和連接器組件,旨在簡化構建基於節點的工具的過程 ...
  • 創建一個webapi項目做測試使用。 創建新控制器,搭建一個基礎框架,包括獲取當天日期、wiki的請求地址等 創建一個Http請求幫助類以及方法,用於獲取指定URL的信息 使用http請求訪問指定url,先運行一下,看看返回的內容。內容如圖右邊所示,實際上是一個Json數據。我們主要解析 大事記 部 ...
  • 最近在不少自媒體上看到有關.NET與C#的資訊與評價,感覺大家對.NET與C#還是不太瞭解,尤其是對2016年6月發佈的跨平臺.NET Core 1.0,更是知之甚少。在考慮一番之後,還是決定寫點東西總結一下,也回顧一下.NET的發展歷史。 首先,你沒看錯,.NET是跨平臺的,可以在Windows、 ...
  • Nodify學習 一:介紹與使用 - 可樂_加冰 - 博客園 (cnblogs.com) Nodify學習 二:添加節點 - 可樂_加冰 - 博客園 (cnblogs.com) 添加節點(nodes) 通過上一篇我們已經創建好了編輯器實例現在我們為編輯器添加一個節點 添加model和viewmode ...
  • 前言 資料庫併發,數據審計和軟刪除一直是數據持久化方面的經典問題。早些時候,這些工作需要手寫複雜的SQL或者通過存儲過程和觸發器實現。手寫複雜SQL對軟體可維護性構成了相當大的挑戰,隨著SQL字數的變多,用到的嵌套和複雜語法增加,可讀性和可維護性的難度是幾何級暴漲。因此如何在實現功能的同時控制這些S ...
  • 類型檢查和轉換:當你需要檢查對象是否為特定類型,並且希望在同一時間內將其轉換為那個類型時,模式匹配提供了一種更簡潔的方式來完成這一任務,避免了使用傳統的as和is操作符後還需要進行額外的null檢查。 複雜條件邏輯:在處理複雜的條件邏輯時,特別是涉及到多個條件和類型的情況下,使用模式匹配可以使代碼更 ...
  • 在日常開發中,我們經常需要和文件打交道,特別是桌面開發,有時候就會需要載入大批量的文件,而且可能還會存在部分文件缺失的情況,那麼如何才能快速的判斷文件是否存在呢?如果處理不當的,且文件數量比較多的時候,可能會造成卡頓等情況,進而影響程式的使用體驗。今天就以一個簡單的小例子,簡述兩種不同的判斷文件是否... ...
  • 前言 資料庫併發,數據審計和軟刪除一直是數據持久化方面的經典問題。早些時候,這些工作需要手寫複雜的SQL或者通過存儲過程和觸發器實現。手寫複雜SQL對軟體可維護性構成了相當大的挑戰,隨著SQL字數的變多,用到的嵌套和複雜語法增加,可讀性和可維護性的難度是幾何級暴漲。因此如何在實現功能的同時控制這些S ...