這里只介紹一些獲取設(shè)備的一些基本信息情妖,像IMEI毡证、MAC地址等一些被蘋果禁掉獲取的信息就不作介紹了居扒。雖然這些可以通過某種方法可以獲取到腻异,但是我自己在網(wǎng)上都沒查到靠譜的方法诈茧,作為一個(gè)菜鳥就不深究了,簡(jiǎn)單說一些從網(wǎng)上查到的信息。
8種iOS獲取設(shè)備唯一標(biāo)識(shí)的方法捂掰,希望對(duì)大家有用敢会。
UDID
UDID(Unique Device Identifier)曾沈,iOS 設(shè)備的唯一識(shí)別碼,是一個(gè)40位十六進(jìn)制序列(越獄的設(shè)備通過某些工具可以改變?cè)O(shè)備的 UDID)鸥昏,移動(dòng)網(wǎng)絡(luò)可以利用 UDID 來識(shí)別移動(dòng)設(shè)備塞俱。
許多開發(fā)者把 UDID 跟用戶的真實(shí)姓名、密碼吏垮、住址障涯、其它數(shù)據(jù)關(guān)聯(lián)起來,網(wǎng)絡(luò)窺探者會(huì)從多個(gè)應(yīng)用收集這些數(shù)據(jù)膳汪,然后順藤摸瓜得到這個(gè)人的許多隱私數(shù)據(jù)唯蝶,同時(shí)大部分應(yīng)用確實(shí)在頻繁傳輸 UDID 和私人信息。 為了避免集體訴訟遗嗽,蘋果最終決定在 iOS 5 的時(shí)候粘我,將這一慣例廢除。
現(xiàn)在應(yīng)用試圖獲取 UDID 已被禁止且不允許上架
MAC 地址
MAC(Medium / Media Access Control)地址痹换,用來表示互聯(lián)網(wǎng)上每一個(gè)站點(diǎn)的標(biāo)示符征字,是一個(gè)六個(gè)字節(jié)(48位)的十六進(jìn)制序列。前三個(gè)字節(jié)是由 IEEE 的注冊(cè)管理機(jī)構(gòu) RA 負(fù)責(zé)給不同廠家分配的”編制上唯一的標(biāo)示符(Organizationally Unique Identifier)”娇豫,后三個(gè)字節(jié)由各廠家自行指派給生產(chǎn)的適配器接口匙姜,稱為擴(kuò)展標(biāo)示符。
MAC 地址在網(wǎng)絡(luò)上用來區(qū)分設(shè)備的唯一性冯痢,接入網(wǎng)絡(luò)的設(shè)備都有一個(gè)MAC地址氮昧,他們肯定都是唯一的。一部 iPhone 上可能有多個(gè) MAC 地址浦楣,包括 WIFI 的袖肥、SIM 的等,但是 iTouch 和 iPad 上就有一個(gè) WIFI 的椒振,因此只需獲取 WIFI 的 MAC 地址就好了昭伸。一般會(huì)采取 MD5(MAC 地址 + bundleID)獲取唯一標(biāo)識(shí)。
但是 MAC 地址和 UDID 一樣澎迎,存在隱私問題庐杨, iOS 7 之后,所有設(shè)備請(qǐng)求 MAC 地址會(huì)返回一個(gè)固定值夹供,這個(gè)方法也不攻自破了灵份。
OpenUDID
UDID 被棄用后,廣大開發(fā)者需要尋找一個(gè)可以替代的 UDID哮洽,并且不受蘋果控制的方案填渠,由此,OpenUDID 成為了當(dāng)時(shí)使用最廣泛的開源 UDID 代替方案。OpenUDID 利用一個(gè)非常巧妙的方法在不同程序間存儲(chǔ)標(biāo)示符:在粘貼板中用了一個(gè)特殊的名稱來存儲(chǔ)標(biāo)示符氛什,通過這種方法莺葫,其他應(yīng)用程序也可以獲取。
蘋果在 iOS 7 之后對(duì)粘貼板做了限制枪眉,導(dǎo)致同一個(gè)設(shè)備上的應(yīng)用間捺檬,無法再共享一個(gè) OpenUDID。
UUID + 自己存儲(chǔ)
UUID(Universally Unique IDentifier)贸铜,通用唯一標(biāo)示符堡纬,是一個(gè)32位的十六進(jìn)制序列,使用小橫線來連接:8-4-4-4-12蒿秦,通過 NSUUID(iOS 6 之后)[NSUUID UUID].UUIDString 或者 CFUUID(iOS 2 之后) CFBridgingRelease(CFUUIDCreateString(kCFAllocatorDefault, CFUUIDCreate(kCFAllocatorDefault))) 來獲取烤镐,但是每次獲取的值都不一樣,需要自己存儲(chǔ)棍鳖。
推送 token + bundleID
推送 token 保證設(shè)備唯一炮叶,但是必須有網(wǎng)絡(luò)情況下才能工作,該方法不依賴于設(shè)備本身鹊杖,但依賴于 apple push悴灵,而 apple push 有時(shí)候會(huì)抽風(fēng)的扛芽。
IDFA
IDFA-identifierForIdentifier(廣告標(biāo)示符)骂蓖,在同一個(gè)設(shè)備上的所有 APP 都會(huì)取到相同的值,是蘋果專門給各廣告提供商用來追蹤用戶而設(shè)定的川尖。雖然 iPhone 默認(rèn)是允許追蹤的登下,而且一般用戶都不知道有這么個(gè)設(shè)置,但是用戶可以在 設(shè)置 - 隱私 - 廣告追蹤 里重置此 ID 的值叮喳,或者限制此 ID 的使用被芳,所以有可能會(huì)取不到值。
IDFV
IDFV-identifierForVendor(Vendor 標(biāo)示符)馍悟,通過 [UIDevice currentDevice].identifierForVendor.UUIDString 來獲取畔濒。是通過 bundleID 的反轉(zhuǎn)的前兩部分進(jìn)行匹配,如果相同是同一個(gè) Vendor 锣咒,例如對(duì)于 com.mayan.app_1 和 com.mayan.app_2 這兩個(gè) bundleID 來說侵状,就屬于同一個(gè) Vendor ,共享同一個(gè) IDFV毅整,和 IDFA 不同的是趣兄,IDFV 的值一定能取到的,所以非常適合于作為內(nèi)部用戶行為分析的主 ID 來識(shí)別用戶悼嫉。但是用戶刪除了該 APP 艇潭,則 IDFV 值會(huì)被重置,再次安裝此 APP ,IDFV 的值和之前的不同蹋凝。
IDFV + keychain
通過以上幾種儲(chǔ)存唯一標(biāo)識(shí)的方法的分析鲁纠,總結(jié)一下各有優(yōu)劣。很多方法被蘋果禁止或者漏洞太多鳍寂,越來越不被開發(fā)者使用房交,現(xiàn)在蘋果主推 IDFA 和 IDFV 這兩種方法,分別對(duì)外和對(duì)內(nèi)伐割,但是 IDFV 在 APP 重新安裝時(shí)會(huì)更改候味,所以我的方法是通過第一次生成的 IDFV 存儲(chǔ)到 keychain 中,以后每次獲取標(biāo)識(shí)符都從[ keychain 中獲取]隔心。
NSString *iPhoneName = [UIDevice currentDevice].name;
NSLog(@"iPhone名稱-->%@", iPhoneName);
NSString *appVerion = [[[NSBundle mainBundle] infoDictionary] objectForKey:@"CFBundleShortVersionString"];
NSLog(@"app版本號(hào)-->%@", appVerion);
CGFloat batteryLevel = [[UIDevice currentDevice] batteryLevel];
NSLog(@"電池電量-->%f", batteryLevel);
NSString *localizedModel = [UIDevice currentDevice].localizedModel;
NSLog(@"localizedModel-->%@", localizedModel);
NSString *systemName = [UIDevice currentDevice].systemName;
NSLog(@"當(dāng)前系統(tǒng)名稱-->%@", systemName);
NSString *systemVersion = [UIDevice currentDevice].systemVersion;
NSLog(@"當(dāng)前系統(tǒng)版本號(hào)-->%@", systemVersion);
struct utsname systemInfo; uname(&systemInfo);
NSString *device_model = [NSString stringWithCString:systemInfo.machine encoding:NSUTF8StringEncoding];
NSLog(@"device_model-->%@", device_model);
NSString *idfa = [[[ASIdentifierManager sharedManager] advertisingIdentifier] UUIDString];
NSLog(@"廣告位標(biāo)識(shí)符idfa-->%@", idfa);
NSString *uuid = [[[UIDevice currentDevice] identifierForVendor] UUIDString];
NSLog(@"唯一識(shí)別碼uuid-->%@", uuid);
// 這個(gè)方法后面會(huì)單獨(dú)列出
NSString *macAddress = [self getMacAddress];
NSLog(@"macAddress-->%@", macAddress);
MAC地址:(網(wǎng)上找的白群,有問題,下面會(huì)繼續(xù)研究)
- (NSString *)getMacAddress {
int mib[6];
size_t len;
char *buf;
unsigned char *ptr;
struct if_msghdr *ifm;
struct sockaddr_dl *sdl;
mib[0] = CTL_NET;
mib[1] = AF_ROUTE;
mib[2] = 0;
mib[3] = AF_LINK;
mib[4] = NET_RT_IFLIST;
if ((mib[5] = if_nametoindex("en0")) == 0)
{ printf("Error: if_nametoindex error/n");
return NULL;
}
if (sysctl(mib, 6, NULL, &len, NULL, 0) < 0)
{ printf("Error: sysctl, take 1/n");
return NULL;
}
if ((buf = malloc(len)) == NULL)
{ printf("Could not allocate memory. error!/n");
return NULL;
}
if (sysctl(mib, 6, buf, &len, NULL, 0) < 0)
{ printf("Error: sysctl, take 2");
return NULL;
}
ifm = (struct if_msghdr *)buf;
sdl = (struct sockaddr_dl *)(ifm + 1);
ptr = (unsigned char *)LLADDR(sdl);
NSString *outstring = [NSString stringWithFormat:@"%02x%02x%02x%02x%02x%02x", *ptr, *(ptr+1), *(ptr+2), *(ptr+3), *(ptr+4), *(ptr+5)];
free(buf);
return [outstring uppercaseString];
}