iOS開發(fā)-獲取設(shè)備號,鑰匙串keychain 記錄獲取

在之前開發(fā)游戲官方渠道sdk的時候仇哆,通常會有快速登錄的需求沦辙,這時候就需要獲取設(shè)備的唯一標(biāo)識 ?來當(dāng)做用戶的賬號。

在一些大公司讹剔,比如網(wǎng)易油讯、騰訊、游久等辟拷,他們都會有自己的一套獲取設(shè)備號的方法撞羽。


之前試過一些阐斜,比如mac地址衫冻,uuid ,udid等谒出,這些方式也都被蘋果禁用了隅俘。

當(dāng)時,uuid可以獲取到笤喳,但是如果用戶在設(shè)置中 ??

設(shè)置-隱私-廣告-跟蹤廣告为居,這個如果是打開狀態(tài)的話,會造成獲取不到設(shè)備號(每次設(shè)備號都會重新生成)杀狡。所以蒙畴,游戲玩家進(jìn)入游戲,下線再次登錄之后呜象,會發(fā)現(xiàn)之前的游戲角色不見了膳凝。(或者變成別人的賬號,發(fā)生串號問題)


那么恭陡,我們就要獲取一個永不會變的“標(biāo)識符”蹬音,存在設(shè)備的鑰匙串中,即使刪除應(yīng)用程序(游戲)休玩,這個“標(biāo)識符”依然存在著淆。


如下:項目中實際的解決方案。

+ (NSString*)UDID

{

NSString *udid = [SvUDIDTools getUDIDFromKeyChain];

if(!udid) {

NSString *sysVersion = [UIDevice currentDevice].systemVersion;

CGFloat version = [sysVersion floatValue];

if(version >=7.0) {

udid = [SvUDIDTools _UDID_iOS7];

}

elseif(version >=2.0) {

udid = [SvUDIDTools _UDID_iOS6];

}

[SvUDIDTools settUDIDToKeyChain:udid];

}

returnudid;

}

/*

* iOS 6.0

* use wifi's mac address

*/

+ (NSString*)_UDID_iOS6

{

return[SvUDIDTools getMacAddress];

}

/*

* iOS 7.0

* Starting from iOS 7, the system always returns the value 02:00:00:00:00:00

* when you ask for the MAC address on any device.

* use identifierForVendor + keyChain

* make sure UDID consistency atfer app delete and reinstall

*/

+ (NSString*)_UDID_iOS7

{

return[[UIDevice currentDevice].identifierForVendor UUIDString];

}

#pragma mark -

#pragma mark Helper Method for Get Mac Address

// fromhttp://stackoverflow.com/questions/677530/how-can-i-programmatically-get-the-mac-address-of-an-iphone

+ (NSString *)getMacAddress

{

intmgmtInfoBase[6];

char*msgBuffer =NULL;

size_tlength;

unsignedcharmacAddress[6];

structif_msghdr*interfaceMsgStruct;

structsockaddr_dl*socketStruct;

NSString*errorFlag =nil;

// Setup the management Information Base (mib)

mgmtInfoBase[0] = CTL_NET;// Request network subsystem

mgmtInfoBase[1] = AF_ROUTE;// Routing table info

mgmtInfoBase[2] =0;

mgmtInfoBase[3] = AF_LINK;// Request link layer information

mgmtInfoBase[4] = NET_RT_IFLIST;// Request all configured interfaces

// With all configured interfaces requested, get handle index

if((mgmtInfoBase[5] = if_nametoindex("en0")) ==0)

errorFlag =@"if_nametoindex failure";

else

{

// Get the size of the data available (store in len)

if(sysctl(mgmtInfoBase,6,NULL, &length,NULL,0) <0)

errorFlag =@"sysctl mgmtInfoBase failure";

else

{

// Alloc memory based on above call

if((msgBuffer = malloc(length)) ==NULL)

errorFlag =@"buffer allocation failure";

else

{

// Get system information, store in buffer

if(sysctl(mgmtInfoBase,6, msgBuffer, &length,NULL,0) <0)

errorFlag =@"sysctl msgBuffer failure";

}

}

}

// Befor going any further...

if(errorFlag !=NULL)

{

NSLog(@"Error: %@", errorFlag);

if(msgBuffer) {

free(msgBuffer);

}

returnerrorFlag;

}

// Map msgbuffer to interface message structure

interfaceMsgStruct = (structif_msghdr *) msgBuffer;

// Map to link-level socket structure

socketStruct = (structsockaddr_dl *) (interfaceMsgStruct +1);

// Copy link layer address data in socket structure to an array

memcpy(&macAddress, socketStruct->sdl_data + socketStruct->sdl_nlen,6);

// Read from char array into a string object, into traditional Mac address format

NSString *macAddressString = [NSString stringWithFormat:@"%02X:%02X:%02X:%02X:%02X:%02X",

macAddress[0], macAddress[1], macAddress[2],

macAddress[3], macAddress[4], macAddress[5]];

NSLog(@"Mac Address: %@", macAddressString);

// Release the buffer memory

free(msgBuffer);

returnmacAddressString;

}

#pragma mark -

#pragma mark Helper Method for make identityForVendor consistency

+ (NSString*)getUDIDFromKeyChain

{

NSMutableDictionary *dictForQuery = [[NSMutableDictionary alloc] init];

[dictForQuery setValue:(id)kSecClassGenericPassword forKey:(id)kSecClass];

// set Attr Description for query

[dictForQuery setValue:[NSString stringWithUTF8String:kKeychainUDIDItemIdentifier]

forKey:kSecAttrDescription];

// set Attr Identity for query

NSData *keychainItemID = [NSData dataWithBytes:kKeychainUDIDItemIdentifier

length:strlen(kKeychainUDIDItemIdentifier)];

[dictForQuery setObject:keychainItemID forKey:(id)kSecAttrGeneric];

// The keychain access group attribute determines if this item can be shared

// amongst multiple apps whose code signing entitlements contain the same keychain access group.

NSString *accessGroup = [NSString stringWithUTF8String:kKeyChainUDIDAccessGroup];

if(accessGroup !=nil)

{

#if TARGET_IPHONE_SIMULATOR

// Ignore the access group if running on the iPhone simulator.

//

// Apps that are built for the simulator aren't signed, so there's no keychain access group

// for the simulator to check. This means that all apps can see all keychain items when run

// on the simulator.

//

// If a SecItem contains an access group attribute, SecItemAdd and SecItemUpdate on the

// simulator will return -25243 (errSecNoAccessForItem).

#else

[dictForQuery setObject:accessGroup forKey:(id)kSecAttrAccessGroup];

#endif

}

[dictForQuery setValue:(id)kCFBooleanTrue forKey:(id)kSecMatchCaseInsensitive];

[dictForQuery setValue:(id)kSecMatchLimitOne forKey:(id)kSecMatchLimit];

[dictForQuery setValue:(id)kCFBooleanTrue forKey:(id)kSecReturnData];

OSStatus queryErr= noErr;

NSData*udidValue =nil;

NSString *udid=nil;

queryErr = SecItemCopyMatching((CFDictionaryRef)dictForQuery, (CFTypeRef*)&udidValue);

NSMutableDictionary *dict =nil;

[dictForQuery setValue:(id)kCFBooleanTrue forKey:(id)kSecReturnAttributes];

queryErr = SecItemCopyMatching((CFDictionaryRef)dictForQuery, (CFTypeRef*)&dict);

if(queryErr == errSecItemNotFound) {

NSLog(@"KeyChain Item: %@ not found!!!", [NSString stringWithUTF8String:kKeychainUDIDItemIdentifier]);

}

elseif(queryErr != errSecSuccess) {

NSLog(@"KeyChain Item query Error!!! Error code:%ld", queryErr);

}

if(queryErr == errSecSuccess) {

NSLog(@"KeyChain Item: %@", udidValue);

if(udidValue) {

udid = [NSString stringWithUTF8String:udidValue.bytes];

[udidValue release];

}

[dict release];

}

[dictForQuery release];

returnudid;

}

+ (BOOL)settUDIDToKeyChain:(NSString*)udid

{

NSMutableDictionary *dictForAdd = [[NSMutableDictionary alloc] init];

[dictForAdd setValue:(id)kSecClassGenericPassword forKey:(id)kSecClass];

[dictForAdd setValue:[NSString stringWithUTF8String:kKeychainUDIDItemIdentifier] forKey:kSecAttrDescription];

[dictForAdd setValue:@"UUID"forKey:(id)kSecAttrGeneric];

// Default attributes for keychain item.

[dictForAdd setObject:@""forKey:(id)kSecAttrAccount];

[dictForAdd setObject:@""forKey:(id)kSecAttrLabel];

// The keychain access group attribute determines if this item can be shared

// amongst multiple apps whose code signing entitlements contain the same keychain access group.

NSString *accessGroup = [NSString stringWithUTF8String:kKeyChainUDIDAccessGroup];

if(accessGroup !=nil)

{

#if TARGET_IPHONE_SIMULATOR

// Ignore the access group if running on the iPhone simulator.

//

// Apps that are built for the simulator aren't signed, so there's no keychain access group

// for the simulator to check. This means that all apps can see all keychain items when run

// on the simulator.

//

// If a SecItem contains an access group attribute, SecItemAdd and SecItemUpdate on the

// simulator will return -25243 (errSecNoAccessForItem).

#else

[dictForAdd setObject:accessGroup forKey:(id)kSecAttrAccessGroup];

#endif

}

constchar*udidStr = [udid UTF8String];

NSData *keyChainItemValue = [NSData dataWithBytes:udidStr length:strlen(udidStr)];

[dictForAdd setValue:keyChainItemValue forKey:(id)kSecValueData];

OSStatus writeErr = noErr;

if([SvUDIDTools getUDIDFromKeyChain]) {// there is item in keychain

[SvUDIDTools updateUDIDInKeyChain:udid];

[dictForAdd release];

returnYES;

}

else{// add item to keychain

writeErr = SecItemAdd((CFDictionaryRef)dictForAdd,NULL);

if(writeErr != errSecSuccess) {

NSLog(@"Add KeyChain Item Error!!! Error Code:%ld", writeErr);

[dictForAdd release];

returnNO;

}

else{

NSLog(@"Add KeyChain Item Success!!!");

[dictForAdd release];

returnYES;

}

}

[dictForAdd release];

returnNO;

}

+ (BOOL)removeUDIDFromKeyChain

{

NSMutableDictionary *dictToDelete = [[NSMutableDictionary alloc] init];

[dictToDelete setValue:(id)kSecClassGenericPassword forKey:(id)kSecClass];

NSData *keyChainItemID = [NSData dataWithBytes:kKeychainUDIDItemIdentifier length:strlen(kKeychainUDIDItemIdentifier)];

[dictToDelete setValue:keyChainItemID forKey:(id)kSecAttrGeneric];

OSStatus deleteErr = noErr;

deleteErr = SecItemDelete((CFDictionaryRef)dictToDelete);

if(deleteErr != errSecSuccess) {

NSLog(@"delete UUID from KeyChain Error!!! Error code:%ld", deleteErr);

[dictToDelete release];

returnNO;

}

else{

NSLog(@"delete success!!!");

}

[dictToDelete release];

returnYES;

}

+ (BOOL)updateUDIDInKeyChain:(NSString*)newUDID

{

NSMutableDictionary *dictForQuery = [[NSMutableDictionary alloc] init];

[dictForQuery setValue:(id)kSecClassGenericPassword forKey:(id)kSecClass];

NSData *keychainItemID = [NSData dataWithBytes:kKeychainUDIDItemIdentifier

length:strlen(kKeychainUDIDItemIdentifier)];

[dictForQuery setValue:keychainItemID forKey:(id)kSecAttrGeneric];

[dictForQuery setValue:(id)kCFBooleanTrue forKey:(id)kSecMatchCaseInsensitive];

[dictForQuery setValue:(id)kSecMatchLimitOne forKey:(id)kSecMatchLimit];

[dictForQuery setValue:(id)kCFBooleanTrue forKey:(id)kSecReturnAttributes];

NSDictionary *queryResult =nil;

SecItemCopyMatching((CFDictionaryRef)dictForQuery, (CFTypeRef*)&queryResult);

if(queryResult) {

NSMutableDictionary *dictForUpdate = [[NSMutableDictionary alloc] init];

[dictForUpdate setValue:[NSString stringWithUTF8String:kKeychainUDIDItemIdentifier] forKey:kSecAttrDescription];

[dictForUpdate setValue:keychainItemID forKey:(id)kSecAttrGeneric];

constchar*udidStr = [newUDID UTF8String];

NSData *keyChainItemValue = [NSData dataWithBytes:udidStr length:strlen(udidStr)];

[dictForUpdate setValue:keyChainItemValue forKey:(id)kSecValueData];

OSStatus updateErr = noErr;

// First we need the attributes from the Keychain.

NSMutableDictionary *updateItem = [NSMutableDictionary dictionaryWithDictionary:queryResult];

[queryResult release];

// Second we need to add the appropriate search key/values.

// set kSecClass is Very important

[updateItem setObject:(id)kSecClassGenericPassword forKey:(id)kSecClass];

updateErr = SecItemUpdate((CFDictionaryRef)updateItem, (CFDictionaryRef)dictForUpdate);

if(updateErr != errSecSuccess) {

NSLog(@"Update KeyChain Item Error!!! Error Code:%ld", updateErr);

[dictForQuery release];

[dictForUpdate release];

returnNO;

}

else{

NSLog(@"Update KeyChain Item Success!!!");

[dictForQuery release];

[dictForUpdate release];

returnYES;

}

}

[dictForQuery release];

returnNO;

}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末拴疤,一起剝皮案震驚了整個濱河市永部,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌呐矾,老刑警劉巖苔埋,帶你破解...
    沈念sama閱讀 219,110評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異凫佛,居然都是意外死亡讲坎,警方通過查閱死者的電腦和手機(jī)孕惜,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,443評論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來晨炕,“玉大人衫画,你說我怎么就攤上這事∥屠酰” “怎么了削罩?”我有些...
    開封第一講書人閱讀 165,474評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長费奸。 經(jīng)常有香客問我弥激,道長,這世上最難降的妖魔是什么愿阐? 我笑而不...
    開封第一講書人閱讀 58,881評論 1 295
  • 正文 為了忘掉前任微服,我火速辦了婚禮,結(jié)果婚禮上缨历,老公的妹妹穿的比我還像新娘以蕴。我一直安慰自己,他們只是感情好辛孵,可當(dāng)我...
    茶點故事閱讀 67,902評論 6 392
  • 文/花漫 我一把揭開白布丛肮。 她就那樣靜靜地躺著,像睡著了一般魄缚。 火紅的嫁衣襯著肌膚如雪宝与。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,698評論 1 305
  • 那天冶匹,我揣著相機(jī)與錄音习劫,去河邊找鬼。 笑死徙硅,一個胖子當(dāng)著我的面吹牛榜聂,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播嗓蘑,決...
    沈念sama閱讀 40,418評論 3 419
  • 文/蒼蘭香墨 我猛地睜開眼须肆,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了桩皿?” 一聲冷哼從身側(cè)響起豌汇,我...
    開封第一講書人閱讀 39,332評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎泄隔,沒想到半個月后拒贱,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,796評論 1 316
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,968評論 3 337
  • 正文 我和宋清朗相戀三年逻澳,在試婚紗的時候發(fā)現(xiàn)自己被綠了闸天。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,110評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡斜做,死狀恐怖苞氮,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情瓤逼,我是刑警寧澤笼吟,帶...
    沈念sama閱讀 35,792評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站霸旗,受9級特大地震影響贷帮,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜诱告,卻給世界環(huán)境...
    茶點故事閱讀 41,455評論 3 331
  • 文/蒙蒙 一撵枢、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧蔬啡,春花似錦诲侮、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,003評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽刮便。三九已至空猜,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間恨旱,已是汗流浹背辈毯。 一陣腳步聲響...
    開封第一講書人閱讀 33,130評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留搜贤,地道東北人谆沃。 一個月前我還...
    沈念sama閱讀 48,348評論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像仪芒,于是被迫代替她去往敵國和親唁影。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,047評論 2 355

推薦閱讀更多精彩內(nèi)容