iOS App之間的通信 -local socket

来源:http://www.cnblogs.com/JJDev/archive/2016/09/15/5873233.html
-Advertisement-
Play Games

之前看到一篇文章介紹到App之間的五種通信方式,它分別有URL Scheme,Keychain,UIPastedboard,UIDocumentInteractionController以及利用socket進行本地通信。前面4種都有用到過,也相對比較簡單,幾行代碼的事。對於最後一種之前一直沒用到過( ...


之前看到一篇文章介紹到App之間的五種通信方式,它分別有URL Scheme,Keychain,UIPastedboard,UIDocumentInteractionController以及利用socket進行本地通信。前面4種都有用到過,也相對比較簡單,幾行代碼的事。對於最後一種之前一直沒用到過(原諒我還是個小白),所以今天試著寫了下,這兒記錄在這裡和大家分享。

好了,廢話不多說,開始:

首先,說下它的原理,其實很簡單,一個App在本地的埠進行TCP的bind和listen,另外一個App在本地同一個埠進行connect,這樣就建立了一個正常的TCP連接,可以想傳什麼數據就傳什麼數據。下麵開始先創建服務端:

1,首先用socket()函數創建一個套接字

/*
 * socket返回一個int值,-1為創建失敗
 * 第一個參數指明瞭協議族/域 ,通常有AF_INET(IPV4)、AF_INET6(IPV6)、AF_LOCAL
 * 第二個參數指定一個套介面類型:SOCK_STREAM,SOCK_DGRAM、SOCK_SEQPACKET等
 * 第三個參數指定相應的傳輸協議,諸如TCP/UDP等,一般設置為0來使用這個預設的值
 */
int sock = socket(AF_INET, SOCK_STREAM, 0); 
if(sock == -1){
  close(sock);
  NSLog(@"socket error : %d",sock);
return; }

 2,綁定本機地址和埠號

// 地址結構體數據,記錄ip和埠號
struct sockaddr_in sockAddr;
// 聲明使用的協議
sockAddr.sin_family = AF_INET;
// 獲取本機的ip,轉換成char類型的 
const char *ip = [[self getIPAddress] cStringUsingEncoding:NSASCIIStringEncoding];
// 將ip賦值給結構體,inet_addr()函數是將一個點分十進位的IP轉換成一個長整數型數
sockAddr.sin_addr.s_addr = inet_addr(ip);
// 設置埠號,htons()是將整型變數從主機位元組順序轉變成網路位元組順序
sockAddr.sin_port = htons(12345);
/*
 * bind函數用於將套接字關聯一個地址,返回一個int值,-1為失敗
 * 第一個參數指定套接字,就是前面socket函數調用返回額套接字
 * 第二個參數為指定的地址
 * 第三個參數為地址數據的大小
 */
int bd = bind(sock,(struct sockaddr *) &sockAddr, sizeof(sockAddr));
if(bd == -1){
  close(sock);
  NSLog(@"bind error : %d",bd);
  return;
}

 3,監聽綁定的地址

/*
 * listen函數使用主動連接套接介面變為被連接介面,使得可以接受其他進程的請求,返回一個int值,-1為失敗
 * 第一個參數是之前socket函數返回的套接字
 * 第二個參數可以理解為連接的最大限制
 */
int ls = listen(sock,20);
if(ls == -1){
  close(sock);
  NSLog(@"listen error : %d",ls);
  return;
}

 4,下麵就是等待客戶端的連接,使用accept()(由於accept函數會阻塞線程,在等待連接的過程中會一直卡著,所以建議將其放在子線程裡面)

// 1,開啟一個子線程
NSTread *recvThread = [[NSThread alloc] initwithTarget:self selector:@selector(recvData) object: nil];
[recvThread start];

- (void)recvData{

// 2,等待客戶端連接
// 聲明一個地址結構體,用於後面接收客戶端返回的地址  
  struct sockaddr_in recvAddr;
// 地址大小
  socklen_t recv_size = sizeof(struct sockaddr_in);
/*
 * accept()函數在連接成功後會返回一個新的套接字(self.newSock),用於之後和這個客戶端之前收發數據
 * 第一個參數為之前監聽的套接字,之前是局部變數,現在需要改為全局的
 * 第二個參數是一個結果參數,它用來接收一個返回值,這個返回值指定客戶端的地址
 * 第三個參數也是一個結果參數,它用來接收recvAddr結構體的代銷,指明其所占的位元組數
 */
self.newSock = accept(self.sock,(struct sockaddr *) &recvAddr, &recv_size);
// 3,來到這裡就代表已經連接到一個新的客戶端,下麵就可以進行收發數據了,主要用到了send()和recv()函數
  ssize_t bytesRecv = -1; // 返回數據位元組大小
  char recvData[128] = ""; // 返回數據緩存區
// 如果一端斷開連接,recv就會馬上返回,bytesrecv等於0,然後while迴圈就會一直執行,所以判斷等於0是跳出去
  while(1){
  bytesRecv = recv(self.newSocket,recvData,128,0); // recvData為收到的數據
  if(bytesRecv == 0){
    break;     
  }
  }
}

 5,發送數據

- (void)sendMessage{
    
    char sendData[32] = "hello client";
    ssize_t size_t = send(self.newSocket, sendData, strlen(sendData), 0); 
 
}

 客戶端那邊就主要分為:創建套接字,根據ip和埠號獲取服務端的主機地址,然後再連接,連接成功過後就能夠向服務端收發數據了,下麵我們看代碼。

1,和服務端一樣用socket函數創建套接字

int sock = socket(AF_INET, SOCK_STREAM,0);
if(sock == -1){

  NSLog(@"socket error : %d",sock);
  return;
}

 2,獲取主機的地址

NSString *host = [self getIPAddress]; // 獲取本機ip地址
// 返回對應於給定主機名的包含主機名字和地址信息的hostent結構指針
struct hostent *remoteHostEnt = gethostbyname([host UTF8String]);
if(remoteHostEnt == NULL){

  close(sock);
  NSLog(@"無法解析伺服器主機名");
  return;
}
// 配置套接字將要連接主機的ip地址和埠號,用於connect()函數 struct in_addr *remoteInAddr = (struct in_addr *)remoteHost->h_addr_list[0]; struct sockaddr_in socktPram; socketPram.sin_family = AF_INT; socketPram.sin_addr = *remoteInAddr; socketPram.sin_port = htons([port intValue]);

 3,使用connect()函數連接主機

/*
 * connect函數通常用於客戶端簡歷tcp連接,連接指定地址的主機,函數返回一個int值,-1為失敗
 * 第一個參數為socket函數創建的套接字,代表這個套接字要連接指定主機
 * 第二個參數為套接字sock想要連接的主機地址和埠號
 * 第三個參數為主機地址大小
 */
int con = connect(sock, (struct sockaddr *) &socketPram, sizeof(socketPram));
if(con == -1){
  close(sock);
  NSLog(@"連接失敗");
  return;
}
NSLog("連接成功"); // 來到這代表連接成功;

4,連接成功之後就可以收發數據了

- (IBAction)senddata:(id)sender {
    // 發送數據
    char sendData[32] = "hello service";
    ssize_t size_t = send(self.sock, sendData, strlen(sendData), 0);
    NSLog(@"%zd",size_t);
}

- (void)recvData{
    // 接受數據,放在子線程
    ssize_t bytesRecv = -1;
    char recvData[32] = "";
    while (1) {
    
        bytesRecv = recv(self.sock, recvData, 32, 0);
        NSLog(@"%zd %s",bytesRecv,recvData);
        if (bytesRecv == 0) {
            break;
        }
    }
}

 好了,利用socket在本地進行兩個App的通訊就這樣就行了。第一次寫博文,一是記錄下自己的心得,二是和大家一起分享,文中有不對的地方希望大家可以指出。最後附上Demo的地址,兩個項目,有興趣的大家可以下下來試下。https://pan.baidu.com/s/1nvcvC8p


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

-Advertisement-
Play Games
更多相關文章
  • 簡單說明 這篇文章是我學習react一個多月來的總結,從基礎開始(包括編輯器設置,構建工具搭建),一步一步走向react開發。相信我,看完這篇文章,跟著文章的步驟走,保證讓你入門react並愛上react,掌握react-router,redux。本文遵循由淺入深的原則。 一、編輯器開始 這裡介紹的 ...
  • 回調函數 在javascript中,當一個函數A作為另外一個函數B的其中一個參數時,則稱A函數為回調函數,即A可以在函數B的運行周期內執行(開始,中間,結束)。 舉例來說,有一個函數用於生成node. 又有一個findNodes函數聲明用於查找所有節點,然後通過callback回調進行執行代碼。 關 ...
  • 1、DIV+CSS定義及優勢 Div+css 是什麼? Div+css 是一種目前比較流行的網頁佈局技術 Div 來存放需要顯示的數據(文字,圖表..) , css 就是用來指定怎樣顯示, 從而做到數據和顯示相互的效果 Div+css 優勢 採用DIV+CSS模式的網站具有以下優勢: 表現和內容相分 ...
  • 很久前的某一天,一位大神問我,你知道ES6相對於ES5有什麼改進嗎? 我一臉懵逼的反問,那個啥,啥是ES5、ES6啊。 不得不承認與大神之間的差距,回來深思了這個問題,結合以前的知識,算是有了點眉目。 JavaScript一種直譯式腳本語言,是一種動態類型、弱類型、基於原型的語言,我們知道javas ...
  • 可見上邊的樣式可實現半透明的遮罩層,我們只需將一個div標簽放置在我們的body中,如下: 在需要遮罩的時候,我們只需將mask變成可見的即可: ...
  • 2、日期類型數值轉換 3、漂亮的隨機代碼 4、合併數組 5、用0補全位數 6、交換值 7、刪除數組元素 8、獲取最大值和最小值 9、拷貝數組 10、日常中true、false判斷 11、獲取今天的0點:0分:0秒和23點:59分:59秒 ...
  • 看CSS3時發現了一個nth-of-type選擇器,發現平時基本沒見過用,就研究了一下,w3c是這樣說明的: :nth-of-type(n) 選擇器匹配屬於父元素的特定類型的第 N 個子元素的每個元素. 看起來和nth-child很像 :nth-child(n) 選擇器匹配屬於其父元素的第 N 個子 ...
  • 原作:Getting started with jQuery Mobile —— Matthew David 翻譯:filod 譯文:http://blog.filod.net/jquerymobile-2/295.html 轉載聲明:請註明原作、翻譯以及譯文鏈接。 你每天都會對著它講話,和它玩游戲 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...