基於 GCDAsyncSocket,簡單實現類似《你猜我畫》的 socket 數據傳輸

来源:http://www.cnblogs.com/howie-ch/archive/2017/02/16/6404543.html
-Advertisement-
Play Games

基於 GCDAsyncSocket,簡單實現類似《你猜我畫》的 socket 數據傳輸 ...


一、前言

  • Socket
    • Socket 是對 TCP/IP 協議的封裝,其中IP協議對應為網路層,TCP 協議對應為傳輸層,而我們常用的HTTP協議,是位於應用層,在七層模型中HTTP協議是基於 TCP/IP 的,我們想要使用 TCP/IP 協議,則要通過 Socket
  • Socket 編程用途(其他待補充)
    • 長連接
    • 端到端的即時通訊
  • Socket 和 Http(來源網路)
    • socket 一般用於比較即時的通信和實時性較高的情況,比如推送,聊天,保持心跳長連接等,http 一般用於實時性要求不那麼高的情況,比如信息反饋,圖片上傳,獲取新聞信息等。

二、類似《你猜我畫》簡易效果說明

  • 效果(分別是模擬器和手機截圖)

  • 工作中碰到類似需求,但沒找到類似的成熟的第三方框架,只有先看看原理性的東西了。其實也就基於 socket 即時傳輸圖片數據、筆畫數據,還有聊天文字,也可以拓展做其他的指令控制
  • 沒有做註冊登錄,沒有做用戶管理,只是簡單原理性的探討
  • 基於 GCDAsyncSocket 框架進行,關於 GCDAsyncSocket 的介紹可自行瞭解

三、服務端部分代碼

  • 直接用 mac 程式作為服務端
    • Server 類
/*!
 @method  開啟服務
 @abstract 開啟伺服器 TCP 連接服務
 */
- (void)startServer {

    self.serverSocket = [[GCDAsyncSocket alloc]initWithDelegate:self
                                                    delegateQueue:dispatch_get_main_queue()];
                                                    NSError *error = nil;
    [self.serverSocket acceptOnPort:5555
                              error:&error];
    if (error) {
        NSLog(@"服務開啟失敗");
    } else {
        NSLog(@"服務開啟成功");
    }

}
#pragma mark - GCDAsyncSocketDelegate
/*!
 @method  收到socket端連接回調
 @abstract 伺服器收到socket端連接回調
 */
- (void)socket:(GCDAsyncSocket *)sock didAcceptNewSocket:(GCDAsyncSocket *)newSocket {
    [self.clientSocketArray addObject:newSocket];
    [newSocket readDataWithTimeout:-1
                               tag:self.clientSocketArray.count];
}
/*!
 @method  收到socket端數據的回調
 @abstract 伺服器收到socket端數據的回調
 */
- (void)socket:(GCDAsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag {
    // 直接進行轉發數據
    for (GCDAsyncSocket *clientSocket in self.clientSocketArray) {
        if (sock != clientSocket) {

            [clientSocket writeData:data
                withTimeout:-1
                        tag:0];
        }
    }
    [sock readDataWithTimeout:-1
                          tag:0];

}
  • main 中

    int main(int argc, const char * argv[]) {
    @autoreleasepool {
    
        Server *chatServer = [[Server alloc]init];
        [chatServer startServer];
        // 開啟主運行迴圈
        [[NSRunLoop mainRunLoop] run];
    }
    return 0;
    }

四、移動端部分代碼

  • 基於 GCDAsyncSocket 的接受數據代理方法及發送數據方法
  • 圖片數據的發送
// 回調 發送圖片
    __weak typeof(self) weakSelf = self;
    bgImgView.block = ^(UIImage *img) {

        weakSelf.drawView.drawImg = img;
        // image
        NSData *imgData = UIImageJPEGRepresentation(weakSelf.drawView.drawImg, 0.2);
        NSMutableData *dat = [NSMutableData data];
        [dat appendData:imgData];
        // 拼接二進位數據流的結束符
        NSData *endData = [@"$" dataUsingEncoding:NSUTF8StringEncoding];
        [dat appendData:endData];
        // 發送數據
        [weakSelf.clientSocket writeData:dat
                         withTimeout:-1
                                 tag:111111];

    };
  • 圖片二進位數據的傳輸是基於流的,一段一段的,避免斷包缺包等問題,需要拼接結束符,圖片數據結束
  • 圖片數據的接受接受
        // 拼接數據 轉成圖片

       [self.socketReadData appendData:data];

       NSData *endData = [data subdataWithRange:NSMakeRange(data.length -1, 1)];

       NSString *end= [[NSString alloc] initWithData:endData
                                            encoding:NSUTF8StringEncoding];

       if ([end isEqualToString:@"$"]) {

           UIImage *tmpImg = [UIImage imageWithData:self.socketReadData];

           self.drawView.drawImg = tmpImg;

           [self.drawView setNeedsDisplay];

           [self.clientSocket readDataWithTimeout:-1
                                              tag:111111];
           // 拼完圖片 恢復預設
           self.socketReadData = nil;

       }
  • 畫布筆畫數據的傳輸
    • 因為傳輸的是二進位數據,所以採取將貝塞爾曲線轉換成 CGPoint 坐標數組,再加上線寬和線的顏色,最後組成一個字典,轉換為二進位進行傳輸
    • 考慮到坐標點在不同屏幕上需要適配,因此需要把當前手機端的屏幕高寬一起傳輸
/*!
 @method  發送路徑
 @abstract 通過socket 發送路徑信息
 */
- (void)sendPath {
    // path 坐標點及 轉換
    NSArray *points = [(UIBezierPath *)self.dataModel.path points];
    NSMutableArray *tmp = [NSMutableArray array];
    for (id value in points) {
        CGPoint point = [value CGPointValue];
        NSDictionary *dic = @{@"x" : @(point.x), @"y": @(point.y)};
        [tmp addObject:dic];
    }

    // 顏色類別
    NSInteger colorNum = 0;

    if (CGColorEqualToColor(self.drawView.color.CGColor, [UIColor redColor].CGColor)) {
        colorNum = 1;
    }
    else  if (CGColorEqualToColor(self.drawView.color.CGColor, [UIColor blueColor].CGColor)  ){

        colorNum = 2;
    } else if (CGColorEqualToColor(self.drawView.color.CGColor, [UIColor greenColor].CGColor)  ) {
        colorNum = 3;
    }


    // 傳遞數據格式
    NSDictionary *pathDataDict = @{
                                   @"path" : tmp,
                                   @"width" : @(self.drawView.width),
                                   @"color" : @(colorNum),
                                   @"screenW": @([UIScreen mainScreen].bounds.size.width),
                                   @"screenH": @([UIScreen mainScreen].bounds.size.height)
                                   };

    NSData *pathData = [NSJSONSerialization
                        dataWithJSONObject:pathDataDict
                        options:NSJSONWritingPrettyPrinted
                        error:nil];


    [self.clientSocket writeData:pathData
                     withTimeout:-1
                             tag:111111];
}
  • 筆畫數據的接受
    • 需要轉換坐標,解析自定義傳輸的數據格式
       // 1、接受坐標點
       NSInteger w = [tmpDict[@"screenW"] integerValue];
       NSInteger h = [tmpDict[@"screenH"] integerValue];
       CGFloat scaleW = [UIScreen mainScreen].bounds.size.width / w;
       CGFloat scaleH = [UIScreen mainScreen].bounds.size.height / h;
       // 處理點
       NSArray *pointDict = tmpDict[@"path"];
       DIYBezierPath *path = [[DIYBezierPath alloc]init];
       for (NSDictionary *tmpDict in pointDict) {
           CGPoint point = CGPointMake([tmpDict[@"x"] floatValue] * scaleW, [tmpDict[@"y"] floatValue] * scaleH);
           NSInteger index = [pointDict indexOfObject:tmpDict];
           if (index == 0) {
               [path moveToPoint:point];
           } else {
               [path addLineToPoint:point];
           }

       }
       switch ([tmpDict[@"color"] integerValue]) {
           case 0:
               self.drawView.color = [UIColor blackColor];
               break;
           case 1:
               self.drawView.color = [UIColor redColor];
               break;
           case 2:
               self.drawView.color = [UIColor blueColor];
               break;
           case 3:
               self.drawView.color = [UIColor greenColor];
               break;

           default:
               break;
       }
       self.drawView.width = [tmpDict[@"width"] floatValue];
       self.drawView.currentPath = path;
       self.drawView.currentPath.pathColor = self.drawView.color;
       self.drawView.currentPath.lineWidth = self.drawView.width;
       [self.drawView.pathArray addObject:path];
       [self.drawView setNeedsDisplay];

五、小demo地址

https://github.com/HOWIE-CH/-You-guess-I-painted-_socket.git

六、問題

  • 定義了圖片文件二進位數據、筆畫路徑二進位數據、聊天字元串二進位數據,三種格式的二進位數據,在 GCDAsyncSocket 接受數據的代理方法,需要判斷接受的二進位文件的類型再進行解析,如果有更好的方式可留言。
  • 只是簡單的功能的嘗試,有時存在畫的一條線過長就傳輸不過去的情況,存在圖片偶爾傳輸不完整的情況
  • 不清楚是否有相關成熟的框架,如果有,請留言。
    

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

-Advertisement-
Play Games
更多相關文章
  • 紅包互換軟體開發 互換紅包軟體定製開發,聯繫傑森團隊微電188-2624-7572. 1、精彩互換系統模式開發付出關鍵:統統的會員介入精彩互換系統模式付出的金額由平臺在後盾設定,能夠設定一個值,也能夠設定多個值。比如說付出5元的,平臺給你返的紅包額度在1-100元之間,付出20元的,平臺給你返的紅包 ...
  • 之前上傳圖片都是直接將圖片轉化為io流傳給伺服器,沒有用框架傳圖片。 最近做項目,打算換個方法上傳圖片。 Android發展到現在,Okhttp顯得越來越重要,所以,這次我選擇用Okhttp上傳圖片。 Okhttp目前已經更新到Okhttp3版本了,用法跟之前相比,也有一些差別。在網上找了很多資料, ...
  • 今天在項目中碰到一個問題,在一個頁面的頂部的標題欄顯示公司的名字,但由於公司名稱較長,顯示不開,影響美觀。故在網上查閱資料,在此做個小的總結。 TextView中有個ellipsize屬性,作用是當文字過長時,該控制項該如何顯示,解釋如下: 1.Android:ellipsize=”start”—–省 ...
  • 由OpenDigg 出品的iOS開源項目周報第八期來啦。我們的iOS開源周報集合了OpenDigg一周來新收錄的優質的iOS開源項目,方便iOS開發人員便捷的找到自己需要的項目工具等。 ...
  • UIDocumentInteractionController UIActivityViewController Shared Keychain Access Custom URL Scheme Web Service iCloud API UIPasteboard 參考 http://enharm ...
  • iOS UISearchController 的使用方法 UISearchController 讓用戶在 UISearchBar 上輸入搜索關鍵詞,展示搜索結果或者進行其他操作。UISearchController 把兩個控制器(UIViewController)連在一起。父控制器放置 UISear ...
  • 二碼公益軟體開發,二碼公益app開發,二碼公益開發聯繫微電188-2624-7572. 隨著社會的發展、經濟的進步,人口老齡化問題不可避免地日益凸顯,“尊老、敬老、扶老”,成為了每個中國人越發不能忽視的社會責任。 在這樣的大環境下,“二碼公益”應運而生,它將取代傳統的商業模式,無論是作為商家還是消費 ...
  • android中一個對象已經不需要了,但是其他對象還持有他的引用,導致他不能回收,導致這個對象暫存在記憶體中,這樣記憶體泄漏就出現了。 記憶體泄漏出現多了,會是應用占用過多的沒存,當占用的記憶體超過了系統分配的記憶體容量,就會出現記憶體溢出了導致應用Crash. 瞭解了記憶體泄漏的原因及影響後,我們需要做的就是掌 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...