iPhone開發(fā)關(guān)于UDID和UUID的一些理解
一.UDID(Unique Device Identifier)
UDID是Unique Device Identifier的縮寫,中文意思是設(shè)備唯一標識.
在很多需要限制一臺設(shè)備一個賬號的應用中經(jīng)常會用到,在Symbian時代,我們是使用IMEI作為設(shè)備的唯一標識的,可惜的是Apple官方不允許開發(fā)者獲得設(shè)備的IMEI.
ios5 sdk中的獲取方法:
- [UIDevice currentDevice] uniqueIdentifier]
uniqueIdentifier在UIDevice.h中的定義如下:
- @property(nonatomic,readonly,retain) NSString *uniqueIdentifier __OSX_AVAILABLE_BUT_DEPRECATED(__MAC_NA,__MAC_NA,__IPHONE_2_0,__IPHONE_5_0); // a string unique to each device based on various hardware info.
意思是iOS2.0以上及iOS5.0以下的系統(tǒng)可用,但不建議使用.Apple有可能在ios5.0之后刪除該函數(shù).
經(jīng)過測試,未越獄的iPhone,系統(tǒng)版本為5.0.1,依然可以獲取UDID.
但是我們需要注意的一點是,對于已越獄了的設(shè)備,UDID并不是唯一的.使用Cydia插件UDIDFaker,可以為每一個應用分配不同的UDID.
所以UDID作為標識唯一設(shè)備的用途已經(jīng)不大了.
二.UUID(Universally Unique Identifier)
UUID是Universally Unique Identifier的縮寫,中文意思是通用唯一識別碼.
由網(wǎng)上資料顯示,UUID是一個軟件建構(gòu)的標準,也是被開源軟件基金會(Open Software Foundation,OSF)的組織在分布式計算環(huán)境(Distributed Computing Environment,DCE)領(lǐng)域的一部份.UUID的目的,是讓分布式系統(tǒng)中的所有元素,都能有唯一的辨識資訊,而不需要透過中央控制端來做辨識資訊的指定.
根據(jù)以上定義可知,同一設(shè)備上的不同應用的UUID是互斥的,即能在改設(shè)備上標識應用.但是并沒有明確指出能標識出裝有同一應用的不同設(shè)備,但是根據(jù)我推測,這個UUID應該是根據(jù)設(shè)備標識和應用標識生成唯一標識,再經(jīng)過加密而來的(純推測).
iOS中獲取UUID的代碼如下:
- 1 -(NSString*) uuid { 2 CFUUIDRef puuid = CFUUIDCreate( nil ); 3 CFStringRef uuidString = CFUUIDCreateString( nil, puuid ); 4 NSString * result = (NSString *)CFStringCreateCopy( NULL, uuidString); 5 CFRelease(puuid); 6 CFRelease(uuidString); 7 return [result autorelease]; 8 }
雖然UUID是官方提出的一種替代UDID的建議方案,但網(wǎng)上有資料說UUID不能保證在以后的系統(tǒng)升級后(IOS6,7)還能用.
經(jīng)過我測試目前,UUID在IOS4和IOS5下均可以使用,而且UUID每次生成的值都不一樣,需要開發(fā)者自行保存UUID.
如果使用UUID為標識保存用戶的資料在網(wǎng)絡上,當用戶重裝軟件后,UUID的值就可能會發(fā)生改變(基本上可說是百分百會發(fā)生改變),用戶則無法重新下載原來的網(wǎng)絡資料.
三.一個可行的解決方案
經(jīng)過上述的探討,我們不難發(fā)現(xiàn),無論是使用UDID或是UUID,我們的目的通常都是為了讓用戶可以自動注冊,而不需要賬號密碼.而使用UDID和UUID作為用戶的ID也并不是毫無缺陷.
現(xiàn)在網(wǎng)上有一現(xiàn)成的解決方案,使用設(shè)備的Mac地址,因為Mac地址也是唯一的.unix有系統(tǒng)調(diào)用可以獲取Mac地址.但有些事情需要注意:
1.iPhone可能有多個Mac地址,wifi的地址,以及SIM卡的地址.一般來講,我們?nèi)n0的地址,因為他是iPhone的wifi的地址,是肯定存在的.(例外情況依然有:市面上依然存在一部分聯(lián)通的閹割版無wifi的iPhone)
2.Mac地址涉及到隱私,不應該胡亂將用戶的Mac地址傳播!所以我們需要將Mac地址進行hash之后,才能作為DeviceId上傳.
關(guān)于第一個注意點的問題,經(jīng)過我測試,沒有Wifi功能的iPhone3GS一樣可以獲得Mac地址,所以這應該是目前標識設(shè)備唯一最好的一個解決方案.
解決方案github下載地址github。com/gekitz/UIDevice-with-UniqueIdentifier-for-iOS-5
背景:
大多數(shù)應用都會用到蘋果設(shè)備的UDID號,UDID通常有以下兩種用途:
1)用于一些統(tǒng)計與分析目的;【第三方統(tǒng)計工具如友盟,廣告商如ADMOB等】
2)將UDID作為用戶ID來唯一識別用戶,省去用戶名,密碼等注冊過程。
不過,2011年時,蘋果就宣布ios5.0以后的系統(tǒng)中將不再支持以下方法獲取用戶的UDID【蘋果設(shè)備的唯一識別碼】:
- [UIDevice currentDevice] uniqueIdentifier];
【注:對于已越獄了的設(shè)備,UDID并不是唯一的.使用Cydia插件UDIDFaker,可以為每一個應用分配不同的UDID】
同時,蘋果公司建議使用UUID【一種開放的軟件構(gòu)建標準】來替代:
- -(NSString*) uuid {
- CFUUIDRef puuid = CFUUIDCreate( nil );
- CFStringRef uuidString = CFUUIDCreateString( nil, puuid );
- NSString * result = (NSString *)CFStringCreateCopy( NULL, uuidString);
- CFRelease(puuid);
- CFRelease(uuidString); return [result autorelease];
- }
該方法每次都會獲取一個唯一的標識字符串,開發(fā)者可以在應用第一次啟動時候調(diào)用一次,然后將該串存儲起來,以便以后替代UDID來使用。
問題是如果用戶刪除該應用再次安裝時,又會生成新的字符串,所以不能保證唯一識別該設(shè)備。
而最近(2012年3月),有消息稱蘋果應用商店開始拒絕使用UDID的應用上架。
替代方案:
現(xiàn)在網(wǎng)上有一現(xiàn)成的解決方案,使用設(shè)備的Mac地址,因為Mac地址也是唯一的.unix有系統(tǒng)調(diào)用可以獲取Mac地址.但有些事情需要注意:
1.iPhone可能有多個Mac地址,wifi的地址,以及SIM卡的地址.一般來講,我們?nèi)n0的地址,因為他是iPhone的wifi的地址,是肯定存在的.
2.Mac地址涉及到隱私,不應該胡亂將用戶的Mac地址傳播!所以我們需要將Mac地址進行hash之后,才能作為DeviceId上傳.
該解決方案源碼地址:https://github.com/gekitz/UIDevice-with-UniqueIdentifier-for-iOS-5
該方案提供了兩個方法:
uniqueDeviceIdentifier (返回MAC和CFBundleIdentifier的MD5值)
uniqueGlobalDeviceIdentifier(返回MAC的MD5值)
使用方法:
- #import "UIDevice+IdentifierAddition.h" NSLog(@"%@",[[UIDevice currentDevice] uniqueDeviceIdentifier]);
- NSLog(@"%@",[[UIDevice currentDevice] uniqueGlobalDeviceIdentifier]);
測試結(jié)果:
WIFI下:
UDID:XXXX21f1f19edff198e2a2356bf4XXXX
新生成的:XXXX7dc3c577446a2bcbd77935bdXXXX
3G下:
UDID:XXXX21f1f19edff198e2a2356bf4XXXX
新生成的:XXXX7dc3c577446a2bcbd77935bdXXXX
GPRS下
UDID:XXXX21f1f19edff198e2a2356bf4XXXX
新生成的:XXXX7dc3c577446a2bcbd77935bdXXXX
飛行模式下:
UDID:XXXX21f1f19edff198e2a2356bf4XXXX
新生成的:XXXX7dc3c577446a2bcbd77935bdXXXX
刪除應用重裝后:
UDID:XXXX21f1f19edff198e2a2356bf4XXXX
新生成的:XXXX7dc3c577446a2bcbd77935bdXXXX