-1.改变一个label上不同的颜色
NSRange range = [self.vip_label.text rangeOfString:@"/"];
NSMutableAttributedString *attString = [[NSMutableAttributedString alloc] initWithString:self.vip_label.text];
[attString addAttributes:@{NSForegroundColorAttributeName:[UIColor colorWithHexString:@"#03a2fd"]} range:NSMakeRange(0, range.location)];
1.获取WebView 的高度
- (void)webViewDidFinishLoad:(UIWebView *)webView {
CGFloat height = [[webView stringByEvaluatingJavaScriptFromString:@"document.body.offsetHeight"] floatValue];
CGRect frame = webView.frame;
webView.frame = CGRectMake(frame.origin.x, frame.origin.y, frame.size.width, height);
}
2.View设置图片.有两种方法可以实现该功能;
第一种方法是把图片装换成背景颜色,然后在加给图片的背景颜色就好了
UIColor *bgColor = [UIColor colorWithPatternImage: [UIImage imageNamed:@"bgImg.png"];
UIView *myView = [[UIView alloc] initWithFrame:CGRectMake(0,0,320,480)];
[myView setBackGroundColor:bgColor];
第二种方法如下
UIImage *image = [UIImage imageNamed:@"yourPicName@2x.png"];
yourView.layer.contents = (__bridge id)image.CGImage;
// 设置显示的图片范围
yourView.layer.contentsCenter = CGRectMake(0.25,0.25,0.5,0.5);//四个值在0-1之间,对应的为x,y,width,height。
3.搜索调cancel修改标题
- (BOOL)searchBarShouldBeginEditing:(UISearchBar *)searchBar
{
searchController.searchBar.showsCancelButton = YES;
UIButton *canceLBtn = [searchController.searchBar valueForKey:@"cancelButton"];
[canceLBtn setTitle:@"取消" forState:UIControlStateNormal];
[canceLBtn setTitleColor:[UIColor colorWithRed:14.0/255.0 green:180.0/255.0 blue:0.0/255.0 alpha:1.00] forState:UIControlStateNormal];
searchBar.showsCancelButton = YES;
return YES;
}
4.滑动时隐藏navgationBar
navigationController.hidesBarsOnSwipe = Yes;
5.禁止运行时,自动锁屏的问题
[[UIApplication sharedApplication] setIdleTimerDisabled:YES];
6.修改占位符颜色和大小
textField.placeholder = @"请输入用户名";
[textField setValue:[UIColor redColor] forKeyPath:@"_placeholderLabel.textColor"];
[textField setValue:[UIFont boldSystemFontOfSize:16] forKeyPath:@"_placeholderLabel.font"];
7.禁止复制黏贴
- (BOOL)canPerformAction:(SEL)action withSender:(id)sender
{
if ([UIMenuController sharedMenuController]) {
[UIMenuController sharedMenuController].menuVisible = NO;
}
return NO;
}
8.简易实用实现导航栏透明方法
// 导航栏变为透明
[self.navigationController.navigationBar setBackgroundImage:[UIImage new] forBarMetrics:0];
// 让黑线消失的方法
self.navigationController.navigationBar.shadowImage=[UIImage new];
9.简单实现忽略静音开关
播放视频时,实际使用中,我们是否开启了静音开关,都会有外放的声音,两行代码简单实现。
viewdidload中加入下面代码就能简单解决
//忽略静音开关
AVAudioSession *audioSession = [AVAudioSession sharedInstance];
[audioSession setCategory:AVAudioSessionCategoryPlayback error:nil];
9.iOS 的沙盒目录结构是怎样的? App Bundle 里面都有什么?
1.沙盒结构
Application:存放程序源文件,上架前经过数字签名,上架后不可修改
Documents:常用目录,iCloud备份目录,存放数据,这里不能存缓存文件,否则上架不被通过
Library
Caches:存放体积大又不需要备份的数据,SDWebImage缓存路径就是这个
Preference:设置目录,iCloud会备份设置信息
tmp:存放临时文件,不会被备份,而且这个文件下的数据有可能随时被清除的可能
10.iOS 7的多任务添加了哪两个新的 API? 各自的使用场景是什么?
1.后台获取(Background Fetch):后台获取使用场景是用户打开应用之前就使app有机会执行代码来获取数据,刷新UI。这样在用户打开应用的时候,最新的内容将已然呈现在用户眼前,而省去了所有的加载过程。
2.推送唤醒(Remote Notifications):使用场景是使设备在接收到远端推送后让系统唤醒设备和我们的后台应用,并先执行一段代码来准备数据和UI,然后再提示用户有推送。这时用户如果解锁设备进入应用后将不会再有任何加载过程,新的内容将直接得到呈现。
11.Push Notification 是如何工作的?
推送通知分为两种,一个是本地推送,一个是远程推送
本地推送:不需要联网也可以推送,是开发人员在APP内设定特定的时间来提醒用户干什么
远程推送:需要联网,用户的设备会于苹果APNS服务器形成一个长连接,用户设备会发送uuid和Bundle idenidentifier给苹果服务器,苹果服务器会加密生成一个deviceToken给用户设备,然后设备会将deviceToken发送给APP的服务器,服务器会将deviceToken存进他们的数据库,这时候如果有人发送消息给我,服务器端就会去查询我的deviceToken,然后将deviceToken和要发送的信息发送给苹果服务器,苹果服务器通过deviceToken找到我的设备并将消息推送到我的设备上,这里还有个情况是如果APP在线,那么APP服务器会于APP产生一个长连接,这时候APPF服务器会直接通过deviceToken将消息推送到设备上
12.为什么scrollview滚动的时候会导致NSTimer失灵
定时器里面有个runoop mode,一般T定时器是运行在defaultmode上但是如果滑动了这个页面,主线程runloop会转到UITrackingRunLoopMode中,这时候就不能处理定时器了,造成定时器失效,原因就是runroop mode选错了,解决办法有2个,一个是更改mode为NSRunLoopCommonModes(无论runloop运行在哪个mode,都能运行),还有种办法是切换到主线程来更新UI界面的刷新
13.你会如何存储用户的一些敏感信息,如登录的 token
使用keychain来存储,也就是钥匙串,使用keychain需要导入Security框架
自定义一个keychain的类
#import@implementationYCKKeyChain+ (NSMutableDictionary*)getKeychainQuery:(NSString*)service {return[NSMutableDictionarydictionaryWithObjectsAndKeys: (__bridge_transferid)kSecClassGenericPassword,(__bridge_transferid)kSecClass, service, (__bridge_transferid)kSecAttrService, service, (__bridge_transferid)kSecAttrAccount, (__bridge_transferid)kSecAttrAccessibleAfterFirstUnlock,(__bridge_transferid)kSecAttrAccessible,nil];}+ (void)save:(NSString*)service data:(id)data {// 获得搜索字典NSMutableDictionary*keychainQuery = [selfgetKeychainQuery:service];// 添加新的删除旧的SecItemDelete((__bridge_retainedCFDictionaryRef)keychainQuery);// 添加新的对象到字符串[keychainQuery setObject:[NSKeyedArchiverarchivedDataWithRootObject:data] forKey:(__bridge_transferid)kSecValueData];// 查询钥匙串SecItemAdd((__bridge_retainedCFDictionaryRef)keychainQuery,NULL);}+ (id)load:(NSString*)service {idret =nil;NSMutableDictionary*keychainQuery = [selfgetKeychainQuery:service];// 配置搜索设置[keychainQuery setObject:(id)kCFBooleanTrueforKey:(__bridge_transferid)kSecReturnData]; [keychainQuery setObject:(__bridge_transferid)kSecMatchLimitOne forKey:(__bridge_transferid)kSecMatchLimit];CFDataRefkeyData =NULL;if(SecItemCopyMatching((__bridge_retainedCFDictionaryRef)keychainQuery, (CFTypeRef*)&keyData) == noErr) {@try{ ret = [NSKeyedUnarchiverunarchiveObjectWithData:(__bridge_transferNSData*)keyData]; }@catch(NSException*e) {NSLog(@"Unarchive of %@ failed: %@", service, e); }@finally{ } }returnret;}+ (void)delete:(NSString*)service {NSMutableDictionary*keychainQuery = [selfgetKeychainQuery:service]; SecItemDelete((__bridge_retainedCFDictionaryRef)keychainQuery);}
在别的类实现存储,加载,删除敏感信息方法
// 用来标识这个钥匙串staticNSString*constKEY_IN_KEYCHAIN =@"com.yck.app.allinfo";// 用来标识密码staticNSString*constKEY_PASSWORD =@"com.yck.app.password";+ (void)savePassWord:(NSString*)password{NSMutableDictionary*passwordDict = [NSMutableDictionarydictionary]; [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];}
14.关于block引用的问题
1.如果要在block中直接使用外部强指针会发生错误,使用以下代码在block外部实现可以解决
__weaktypeof(self) weakSelf =self;
2.但是如果在block内部使用延时操作还使用弱指针的话会取不到该弱指针,需要在block内部再将弱指针强引用一下
__strongtypeof(self) strongSelf = weakSelf;
15.使用 NSUserDefaults 时,如何处理布尔的默认值?(比如返回 NO,不知道是真的 NO 还是没有设置过)
if([[NSUserDefaults standardUserDefaults]objectForKey:ID] == nil){NSLog(@"没有设置"); }
16.开发常用的工具有哪些?
通过这个问题面试官也可以大概知道你是什么水平的,如果你只能讲什么xcode,cocoapods或者xcode自带的一些工具的话面试官也许就会把你定位于初级
参考答案:
1.友盟统计
2.青花瓷:这个软件还是蛮不错的,可以用来过滤网络请求,模拟低速网路,还可以修改网络请求内容这些
3.Reveal:调试页面不错,还有用来学习别人的demo时候可以拿来看UI层次结构,还可以用来标记可以设为透明的控件,用来优化性能
17.单例的缺点也就是会一直占着这块内存,不会被释放
18.问题[◇◇◇]:谈谈NSNotification 和 KVO 的使用场景?
两者都是观察者模式,不同的是,KVO是被观察者直接发送消息给观察者,是对象间的交互,而通知则是观察者和被观察者通过通知中心对象之间进行交互,即消息由被观察者发送到通知中心对象,再由中心对象发给观察者,两者之间并不进行直接的交互
19.
问题[※※※※]:如何用GCD同步若干个异步调用?(如根据若干个url异步加载多张图片,然后在都下载完成后合成一张整图)
使用Dispatch Group追加block到Global Group Queue,这些block如果全部执行完毕,就会执行Main Dispatch Queue中的结束处理的block
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
dispatch_group_t group = dispatch_group_create();
dispatch_group_async(group, queue, ^{ /*加载图片1 */ });
dispatch_group_async(group, queue, ^{ /*加载图片2 */ });
dispatch_group_async(group, queue, ^{ /*加载图片3 */ });
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
// 合并图片
});
20.问题[○○○○]:UITableViewCell上有个UILabel,显示NSTimer实现的秒表时间,手指滚动cell过程中,label是否刷新,为什么?
这是否刷新取决于timer加入到Run Loop中的Mode是什么。Mode主要是用来指定事件在运行循环中的优先级的,分为:
NSDefaultRunLoopMode(kCFRunLoopDefaultMode):默认,空闲状态
UITrackingRunLoopMode:ScrollView滑动时会切换到该Mode
UIInitializationRunLoopMode:run loop启动时,会切换到该Mode
NSRunLoopCommonModes(kCFRunLoopCommonModes):
苹果公开提供的Mode有两个:
NSDefaultRunLoopMode(kCFRunLoopDefaultMode)
NSRunLoopCommonModes(kCFRunLoopCommonModes)
在编程中:如果我们把一个NSTimer对象以NSDefaultRunLoopMode添加到主运行循环中的时候, ScrollView滚动过程中会因为Mode的切换,而导致NSTimer将不再被调度。当我们滚动的时候,也希望不调度,那就应该使用默认模式。但是,如果希望在滚动时,定时器也要回调,那就应该使用NSRunLoopCommonModes
21.去掉丝滑效果
//- (void)viewDidAppear:(BOOL)animated{
// [super viewDidAppear:animated];
// if ([self.navigationController respondsToSelector:@selector(interactivePopGestureRecognizer)]) {
// self.navigationController.interactivePopGestureRecognizer.enabled = NO;
// }
//}
//- (void)viewWillDisappear:(BOOL)animated{
// [super viewWillDisappear:animated];
// if ([self.navigationController respondsToSelector:@selector(interactivePopGestureRecognizer)]) {
// self.navigationController.interactivePopGestureRecognizer.enabled = NO;
// }
//}
22.拨打电话
UIAlertAction *sure1 = [UIAlertAction actionWithTitle:@"拨打" style:(UIAlertActionStyleDefault) handler:^(UIAlertAction * _Nonnull action) {
[[UIApplication sharedApplication] openURL:[NSURL URLWithString:@"tel://400-832-5566"]];
}];
23.一行文字的颜色不同
self.vip_label.text = [NSString stringWithFormat:@"%@/30000",[dic objectForKey:@"growth"]];
NSRange range = [self.vip_label.text rangeOfString:@"/"];
NSMutableAttributedString *attString = [[NSMutableAttributedString alloc] initWithString:self.vip_label.text];
[attString addAttributes:@{NSForegroundColorAttributeName:[UIColor blueColor]} range:NSMakeRange(0, range.location)];
self.vip_label.attributedText = attString;
24.提示框
UIAlertController *alert = [UIAlertController alertControllerWithTitle:@"电话号码" message:@"400-832-5566"preferredStyle:(UIAlertControllerStyleAlert)];
UIAlertAction *sure = [UIAlertAction actionWithTitle:@"关闭" style:(UIAlertActionStyleDefault) handler:^(UIAlertAction * _Nonnull action) {
}];
UIAlertAction *sure1 = [UIAlertAction actionWithTitle:@"拨打" style:(UIAlertActionStyleDefault) handler:^(UIAlertAction * _Nonnull action) {
[[UIApplication sharedApplication] openURL:[NSURL URLWithString:@"tel://400-832-5566"]];
}];
[alert addAction:sure];
[alert addAction:sure1];
[self presentViewController:alert animated:YES completion:nil];