解決 iOS 14.5 UDP 廣播 Sendto 返回 -1
本文轉(zhuǎn)載自微信公眾號「網(wǎng)羅開發(fā)」,作者展菲。轉(zhuǎn)載本文請聯(lián)系網(wǎng)羅開發(fā)公眾號。
1. 問題背景
- 手機(jī)系統(tǒng)升級到 iOS 14.5 之后,UDP 廣播發(fā)送失敗
- 項目中老版本使用到 socket
- 項目中新版本使用 CocoaAsyncSocket
- 兩種 UDP 發(fā)包方式都會報錯 No route to host
報錯具體內(nèi)容如下:
- sendto: -1
- client: sendto fail, but just ignore it
- : No route to host
2. 問題分析
2.1 sendto 返回 -1 問題排查
我們知道發(fā)送廣播 sendto 返回 -1,正常情況sendto 返回值大于 0 。
首先判斷 socket 連接是否建立
- self._sck_fd4 = socket(AF_INET,SOCK_DGRAM,0);
- if (DEBUG_ON) {
- NSLog(@"client init() _sck_fd4=%d",self._sck_fd4);
- }
self._sck_fd4 打印:
- server init(): _sck_fd4=12
socket 連接正常,接下來判斷數(shù)據(jù)發(fā)包
- sendto(self._sck_fd4, bytes, dataLen, 0, (struct sockaddr*)&target_addr, addr_len) = -1
數(shù)據(jù)發(fā)送失敗
2.2 增加 NSLocalNetworkUsageDescription 權(quán)限
- Info.plist 添加 NSLocalNetworkUsageDescription
- 發(fā)送一次UDP廣播,觸發(fā)權(quán)限彈框,讓用戶點擊好,允許訪問本地網(wǎng)絡(luò)。
發(fā)現(xiàn)問題依舊存在
2.3 發(fā)送單播排查
由于項目中發(fā)送廣播設(shè)置的 hostName 為 255.255.255.255,為了排查決定先發(fā)送單播看是否能成功。
將單播地址改為 192.168.0.101 之后發(fā)現(xiàn)是可以發(fā)送成功的,然后在新版本 CocoaAsyncSocket 庫中發(fā)送單播也是可以成功的。
UDP 廣播推薦使用 192.168.0.255 ,將廣播地址改了之后,問題解決了,設(shè)備可以收到 UDP 廣播數(shù)據(jù)。
3. 問題解決
由于 192.168.0.255 廣播地址只是當(dāng)前本地地址,App 中需要動態(tài)改變前三段 192.168.0 本地地址,解決方法如下:
- NSString *localInetAddr4 = [ESP_NetUtil getLocalIPv4];
- NSArray *arr = [localInetAddr4 componentsSeparatedByString:@"."];
- NSString *deviceAddress4 = [NSString stringWithFormat:@"%@.%@.%@.255",arr[0], arr[1], arr[2]];
發(fā)包過濾,只需要過濾地址最后一段是否為 255
- bool isBroadcast = [targetHostName hasSuffix:@"255"];