国产成人精品亚洲777人妖,欧美日韩精品一区视频,最新亚洲国产,国产乱码精品一区二区亚洲

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

iOS開(kāi)發(fā)-你能用到的面試題

瀏覽:91日期:2022-09-17 13:46:20
Push Notification 是如何工作的?推送通知分為兩種,一個(gè)是本地推送,一個(gè)是遠(yuǎn)程推送本地推送:不需要聯(lián)網(wǎng)也可以推送,是開(kāi)發(fā)人員在APP內(nèi)設(shè)定特定的時(shí)間來(lái)提醒用戶干什么遠(yuǎn)程推送:需要聯(lián)網(wǎng),用戶的設(shè)備會(huì)于蘋(píng)果APNS服務(wù)器形成一個(gè)長(zhǎng)連接,用戶設(shè)備會(huì)發(fā)送uuid和Bundle idenidentifier給蘋(píng)果服務(wù)器,蘋(píng)果服務(wù)器會(huì)加密生成一個(gè)deviceToken給用戶設(shè)備,然后設(shè)備會(huì)將deviceToken發(fā)送給APP的服務(wù)器,服務(wù)器會(huì)將deviceToken存進(jìn)他們的數(shù)據(jù)庫(kù),這時(shí)候如果有人發(fā)送消息給我,服務(wù)器端就會(huì)去查詢我的deviceToken,然后將deviceToken和要發(fā)送的信息發(fā)送給蘋(píng)果服務(wù)器,蘋(píng)果服務(wù)器通過(guò)deviceToken找到我的設(shè)備并將消息推送到我的設(shè)備上,這里還有個(gè)情況是如果APP在線,那么APP服務(wù)器會(huì)于APP產(chǎn)生一個(gè)長(zhǎng)連接,這時(shí)候APPF服務(wù)器會(huì)直接通過(guò)deviceToken將消息推送到設(shè)備上什么是 Runloop?

是一個(gè)與線程相關(guān)的機(jī)制,可以理解為一個(gè)循環(huán),在這個(gè)循環(huán)里面等待事件然后處理事件.而這個(gè)循環(huán)是基于線程的,在Cocoa中每個(gè)線程都有它的runroop,通過(guò)他這樣的機(jī)制,線程可以在沒(méi)有事件要處理的時(shí)候休息,有事件運(yùn)行,減輕CPU壓力,這題可以衍生出為什么在滑動(dòng)時(shí)會(huì)導(dǎo)致定時(shí)器失敗,在下面有解答

Toll-Free Bridging 是什么?什么情況下會(huì)使用?

Toll-Free Bridging用于在Foundation對(duì)象與Core Foundation對(duì)象之間交換數(shù)據(jù),俗稱橋接

在ARC環(huán)境下,Foundation對(duì)象轉(zhuǎn)成 Core Foundation對(duì)象使用__bridge橋接以后ARC會(huì)自動(dòng)2個(gè)對(duì)象使用__bridge_retained橋接需要手動(dòng)釋放Core Foundation對(duì)象在ARC環(huán)境下, Core Foundation對(duì)象轉(zhuǎn)成 Foundation對(duì)象使用__bridge橋接,如果Core Foundation對(duì)象被釋放,Foundation對(duì)象也同時(shí)不能使用了,需要手動(dòng)管理Core Foundation對(duì)象使用__bridge_transfer橋接,系統(tǒng)會(huì)自動(dòng)管理2個(gè)對(duì)象當(dāng)系統(tǒng)出現(xiàn)內(nèi)存警告時(shí)會(huì)發(fā)生什么?會(huì)將不在當(dāng)前窗口上的view暫時(shí)移除釋放掉沙盒里temp里的緩存文件如果用戶放任內(nèi)存警告,最終會(huì)導(dǎo)致軟件強(qiáng)制被系統(tǒng)關(guān)閉什么是 Protocol,Delegate 一般是怎么用的?協(xié)議是一個(gè)方法簽名的列表,在其中可以定義若干個(gè)方法,遵守該協(xié)議的類可以實(shí)現(xiàn)協(xié)議里的方法,在協(xié)議中使用@property只會(huì)生成setter和getter方法的聲明delegate用法:成為一個(gè)類的代理,可以去實(shí)現(xiàn)協(xié)議里的方法autorelease 對(duì)象在什么情況下會(huì)被釋放?分兩種情況:手動(dòng)干預(yù)釋放和系統(tǒng)自動(dòng)釋放手動(dòng)干預(yù)釋放就是指定autoreleasepool,當(dāng)前作用域大括號(hào)結(jié)束就立即釋放系統(tǒng)自動(dòng)去釋放:不手動(dòng)指定autoreleasepool,Autorelease對(duì)象會(huì)在當(dāng)前的 runloop 迭代結(jié)束時(shí)釋放kCFRunLoopEntry(1):第一次進(jìn)入會(huì)自動(dòng)創(chuàng)建一個(gè)autoreleasekCFRunLoopBeforeWaiting(32):進(jìn)入休眠狀態(tài)前會(huì)自動(dòng)銷毀一個(gè)autorelease,然后重新創(chuàng)建一個(gè)新的autoreleasekCFRunLoopExit(128):退出runloop時(shí)會(huì)自動(dòng)銷毀最后一個(gè)創(chuàng)建的autorelease為什么 NotificationCenter 要 removeObserver? 如何實(shí)現(xiàn)自動(dòng) remove?如果不移除的話,萬(wàn)一注冊(cè)通知的類被銷毀以后又發(fā)了通知,程序會(huì)崩潰.因?yàn)橄蛞爸羔槹l(fā)送了消息實(shí)現(xiàn)自動(dòng)remove:通過(guò)自釋放機(jī)制,通過(guò)動(dòng)態(tài)屬性將remove轉(zhuǎn)移給第三者,解除耦合,達(dá)到自動(dòng)實(shí)現(xiàn)remove當(dāng) TableView 的 Cell 改變時(shí),如何讓這些改變以動(dòng)畫(huà)的形式呈現(xiàn)?

這里舉個(gè)例子,點(diǎn)擊cell以后以動(dòng)畫(huà)形式改變cell高度

@interface ViewController () @property (nonatomic, strong) NSIndexPath *index; @end @implementation ViewController static NSString *ID = @'cell';- (void)viewDidLoad { [super viewDidLoad];}- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{ UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID]; cell.textLabel.text = [NSString stringWithFormat:@'%ld',(long)indexPath.row]; return cell;}- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section{ return 20;}- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath{ if(self.index == indexPath){ return 120; } return 60;}- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath{ self.index = indexPath; [tableView deselectRowAtIndexPath:indexPath animated:TRUE]; // 重點(diǎn)是這2句代碼實(shí)現(xiàn)的功能 [tableView beginUpdates]; [tableView endUpdates];}為什么 UIScrollView 的滾動(dòng)會(huì)導(dǎo)致 NSTimer 失效?

定時(shí)器里面有個(gè)runoop mode,一般定時(shí)器是運(yùn)行在defaultmode上但是如果滑動(dòng)了這個(gè)頁(yè)面,主線程runloop會(huì)轉(zhuǎn)到UITrackingRunLoopMode中,這時(shí)候就不能處理定時(shí)器了,造成定時(shí)器失效,原因就是runroop mode選錯(cuò)了,解決辦法有2個(gè),一個(gè)是更改mode為NSRunLoopCommonModes(無(wú)論runloop運(yùn)行在哪個(gè)mode,都能運(yùn)行),還有種辦法是切換到主線程來(lái)更新UI界面的刷新

為什么當(dāng) Core Animation 完成時(shí),layer 又會(huì)恢復(fù)到原先的狀態(tài)?

因?yàn)檫@些產(chǎn)生的動(dòng)畫(huà)只是假象,并沒(méi)有對(duì)layer進(jìn)行改變

你會(huì)如何存儲(chǔ)用戶的一些敏感信息,如登錄的 token使用keychain來(lái)存儲(chǔ),也就是鑰匙串,使用keychain需要導(dǎo)入Security框架

自定義一個(gè)keychain的類

#import <Security/Security.h> @implementation YCKKeyChain + (NSMutableDictionary *)getKeychainQuery:(NSString *)service { return [NSMutableDictionary dictionaryWithObjectsAndKeys: (__bridge_transfer id)kSecClassGenericPassword,(__bridge_transfer id)kSecClass, service, (__bridge_transfer id)kSecAttrService, service, (__bridge_transfer id)kSecAttrAccount, (__bridge_transfer id)kSecAttrAccessibleAfterFirstUnlock,(__bridge_transfer id)kSecAttrAccessible, nil];}+ (void)save:(NSString *)service data:(id)data { // 獲得搜索字典 NSMutableDictionary *keychainQuery = [self getKeychainQuery:service]; // 添加新的刪除舊的 SecItemDelete((__bridge_retained CFDictionaryRef)keychainQuery); // 添加新的對(duì)象到字符串 [keychainQuery setObject:[NSKeyedArchiver archivedDataWithRootObject:data] forKey:(__bridge_transfer id)kSecValueData]; // 查詢鑰匙串 SecItemAdd((__bridge_retained CFDictionaryRef)keychainQuery, NULL);}+ (id)load:(NSString *)service { id ret = nil; NSMutableDictionary *keychainQuery = [self getKeychainQuery:service]; // 配置搜索設(shè)置 [keychainQuery setObject:(id)kCFBooleanTrue forKey:(__bridge_transfer id)kSecReturnData]; [keychainQuery setObject:(__bridge_transfer id)kSecMatchLimitOne forKey:(__bridge_transfer id)kSecMatchLimit]; CFDataRef keyData = NULL; if (SecItemCopyMatching((__bridge_retained CFDictionaryRef)keychainQuery, (CFTypeRef *)&keyData) == noErr) { @try { ret = [NSKeyedUnarchiver unarchiveObjectWithData:(__bridge_transfer NSData *)keyData];} @catch (NSException *e) { NSLog(@'Unarchive of %@ failed: %@', service, e);} @finally {} } return ret;}+ (void)delete:(NSString *)service { NSMutableDictionary *keychainQuery = [self getKeychainQuery:service]; SecItemDelete((__bridge_retained CFDictionaryRef)keychainQuery);}

在別的類實(shí)現(xiàn)存儲(chǔ),加載,刪除敏感信息方法

// 用來(lái)標(biāo)識(shí)這個(gè)鑰匙串 static NSString * const KEY_IN_KEYCHAIN = @'com.yck.app.allinfo'; // 用來(lái)標(biāo)識(shí)密碼 static NSString * const KEY_PASSWORD = @'com.yck.app.password';+ (void)savePassWord:(NSString *)password{ NSMutableDictionary *passwordDict = [NSMutableDictionary dictionary]; [passwordDict setObject:password forKey:KEY_PASSWORD]; [YCKKeyChain save:KEY_IN_KEYCHAIN data:passwordDict];}+ (id)readPassWord{ NSMutableDictionary *passwordDict = (NSMutableDictionary *)[YCKKeyChain load:KEY_IN_KEYCHAIN]; return [passwordDict objectForKey:KEY_PASSWORD];}+ (void)deletePassWord{ [YCKKeyChain delete:KEY_IN_KEYCHAIN];}有用過(guò)一些開(kāi)源組件吧,能簡(jiǎn)單說(shuō)幾個(gè)么,大概說(shuō)說(shuō)它們的使用場(chǎng)景實(shí)現(xiàn)。AFN:網(wǎng)絡(luò)請(qǐng)求FMDB:使用數(shù)據(jù)庫(kù)MJExtension: JSON與Model互轉(zhuǎn)SVProgressHUD:提示HUDMasonry:自動(dòng)布局MJRefresh:下拉和上拉刷新什么時(shí)候會(huì)發(fā)生 EXC BAD ACCESS 異常?訪問(wèn)一個(gè)僵尸對(duì)象,訪問(wèn)僵尸對(duì)象的成員變量或者向其發(fā)消息死循環(huán)NSNotification 和 KVO 的使用場(chǎng)景?KVO使用場(chǎng)景:當(dāng)一個(gè)對(duì)象的特定屬性改變的時(shí)候,需要被通知一個(gè)或者多個(gè)對(duì)象的時(shí)候NSNotification使用場(chǎng)景:跨層級(jí)傳遞值,多個(gè)對(duì)象通知多個(gè)對(duì)象使用 Block 時(shí)需要注意哪些問(wèn)題?在block內(nèi)部使用外部指針,需要在用__weak修飾外部指針__weak typeof(self) weakSelf = self;在block內(nèi)部如果調(diào)用了延時(shí)函數(shù)還使用弱指針會(huì)取不到該指針,因?yàn)橐呀?jīng)被銷毀了,需要在block內(nèi)部再將弱指針重新強(qiáng)引用一下__strong typeof(self) strongSelf = weakSelf;如果需要在block內(nèi)部改變外部變量的話,需要在用__block修飾外部變量筆者也寫(xiě)過(guò)一篇block博客performSelector:withObject:afterDelay: 內(nèi)部大概是怎么實(shí)現(xiàn)的,有什么注意事項(xiàng)么?創(chuàng)建一個(gè)定時(shí)器,時(shí)間結(jié)束后系統(tǒng)會(huì)使用runtime通過(guò)方法名稱(Selector本質(zhì)就是方法名稱)去方法列表中找到對(duì)應(yīng)的方法實(shí)現(xiàn)并調(diào)用方法注意事項(xiàng)調(diào)用performSelector:withObject:afterDelay:方法時(shí),先判斷希望調(diào)用的方法是否存在respondsToSelector:這個(gè)方法是異步方法,必須在主線程調(diào)用,在子線程調(diào)用永遠(yuǎn)不會(huì)調(diào)用到想調(diào)用的方法使用 NSUserDefaults 時(shí),如何處理布爾的默認(rèn)值?(比如返回 NO,不知道是真的 NO 還是沒(méi)有設(shè)置過(guò))

if([[NSUserDefaults standardUserDefaults] objectForKey:ID] == nil){ NSLog(@'沒(méi)有設(shè)置'); }哪些途徑可以讓 ViewController 瘦下來(lái)?把 Data Source 和其他 Protocols 分離出來(lái)(將UITableView或者UICollectionView的代碼提取出來(lái)放在其他類中)將業(yè)務(wù)邏輯移到 Model 中(和模型有關(guān)的邏輯全部在model中寫(xiě))把網(wǎng)絡(luò)請(qǐng)求邏輯移到 Model 層(網(wǎng)絡(luò)請(qǐng)求依靠模型)把 View 代碼移到 View 層(自定義View)有哪些常見(jiàn)的 Crash 場(chǎng)景?訪問(wèn)了僵尸對(duì)象訪問(wèn)了不存在的方法數(shù)組越界在定時(shí)器下一次回調(diào)前將定時(shí)器釋放,會(huì)Crash

標(biāo)簽: IOS
相關(guān)文章:
主站蜘蛛池模板: 高青县| 昌平区| 隆尧县| 西昌市| 新余市| 镇安县| 平南县| 边坝县| 云龙县| 太仓市| 信丰县| 眉山市| 保德县| 绥宁县| 甘泉县| 南川市| 项城市| 绍兴市| 福清市| 于都县| 焦作市| 新晃| 石景山区| 邹城市| 东台市| 白山市| 高陵县| 兴隆县| 乡城县| 紫云| 大理市| 阜新| 焦作市| 长乐市| 新建县| 长海县| 昭平县| 儋州市| 沧州市| 东源县| 凉山|