iOS UI

iOS爬坑之路!

2019-08-06  本文已影响0人  小别致丶

记录一些常用和不常用的iOS知识点,防止遗忘丢失。(来源为收集自己项目中用到的或者整理看到博客中的知识点),如有错误,欢迎大家批评指正;如有好的知识点,也欢迎大家联系我,添加上去。谢谢!

一、调用代码使APP进入后台,达到点击Home键的效果。(私有API)

[[UIApplication sharedApplication] performSelector:@selector(suspend)];

二、获取UIWebView的高度 。

大概举个例子,类似下面的URL,里面直接含有中文,可能导致播放不了,那么我们要处理一个这个URL,因为他太操蛋了,居然用中文。

- (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);
} 

三、给UIView设置图片(UILabel一样适用)

第一种方法: 利用的UIView的设置背景颜色方法,用图片做图案颜色,然后传给背景颜色。

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。

四、去掉UITableView多余的分割线

yourTableView.tableFooterView = [UIView new];

五、调整cell分割线的位置,两个方法一起用,暴力解决,防脱发

-(void)viewDidLayoutSubviews {
    
    if ([self.mytableview respondsToSelector:@selector(setSeparatorInset:)]) {
        [self.mytableview setSeparatorInset:UIEdgeInsetsMake(0, 0, 0, 0)];
        
    }
    if ([self.mytableview respondsToSelector:@selector(setLayoutMargins:)])  {
        [self.mytableview setLayoutMargins:UIEdgeInsetsMake(0, 0, 0, 0)];
    }
    
}
 
#pragma mark - cell分割线
- (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath
{
    if ([cell respondsToSelector:@selector(setSeparatorInset:)]){
        [cell setSeparatorInset:UIEdgeInsetsMake(0, 0, 0, 0)];
    }
    if ([cell respondsToSelector:@selector(setLayoutMargins:)]) {
        [cell setLayoutMargins:UIEdgeInsetsMake(0, 0, 0, 0)];
    }
}

六、UILabel和UIImageView的交互userInteractionEabled默认为NO。那么如果你把这两个类做为父试图的话,里面的所有东东都不可以点击哦。userInteractionEabled设置为YES即可。
七、UISearchController和UISearchBar的Cancle按钮改title问题,简单粗暴

- (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;
}

八、UITableView收起键盘
一个属性搞定,效果好(UIScrollView同样可以使用)
以前是不是觉得[self.view endEditing:YES];很屌,这个下面的更屌。
yourTableView.keyboardDismissMode = UIScrollViewKeyboardDismissModeOnDrag;

另外一个枚举为UIScrollViewKeyboardDismissModeInteractive,表示在键盘内部滑动,键盘逐渐下去。

九、NSTimer

1、NSTimer计算的时间并不精确
2、NSTimer需要添加到runLoop运行才会执行,但是这个runLoop的线程必须是已经开启。
3、NSTimer会对它的tagert进行retain,我们必须对其重复性的使用intvailte停止。target如果是self(指UIViewController),那么VC的retainCount+1,如果你不释放NSTimer,那么你的VC就不会dealloc了,内存泄漏了。

十、UIViewController没有大小(frame)
只有UIView才能设置大小,VC是控制器,不能设置大小。

十一、获取今天是星期几

+ (NSString *) getweekDayStringWithDate:(NSDate *) date
{
    NSCalendar * calendar = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar]; // 指定日历的算法
    NSDateComponents *comps = [calendar components:NSWeekdayCalendarUnit fromDate:date];
 
    // 1 是周日,2是周一 3.以此类推
 
    NSNumber * weekNumber = @([comps weekday]);
    NSInteger weekInt = [weekNumber integerValue];
    NSString *weekDayString = @"(周一)";
    switch (weekInt) {
        case 1:
        {
            weekDayString = @"(周日)";
        }
            break;
 
        case 2:
        {
            weekDayString = @"(周一)";
        }
            break;
 
        case 3:
        {
            weekDayString = @"(周二)";
        }
            break;
 
        case 4:
        {
            weekDayString = @"(周三)";
        }
            break;
 
        case 5:
        {
            weekDayString = @"(周四)";
        }
            break;
 
        case 6:
        {
            weekDayString = @"(周五)";
        }
            break;
 
        case 7:
        {
            weekDayString = @"(周六)";
        }
            break;
 
        default:
            break;
    }
    return weekDayString;
 
}

十二、UIView的部分圆角

UIView *view2 = [[UIView alloc] initWithFrame:CGRectMake(120, 10, 80, 80)];
view2.backgroundColor = [UIColor redColor];
[self.view addSubview:view2];
 
UIBezierPath *maskPath = [UIBezierPath bezierPathWithRoundedRect:view2.bounds byRoundingCorners:UIRectCornerBottomLeft | UIRectCornerBottomRight cornerRadii:CGSizeMake(10, 10)];
CAShapeLayer *maskLayer = [[CAShapeLayer alloc] init];
maskLayer.frame = view2.bounds;
maskLayer.path = maskPath.CGPath;
view2.layer.mask = maskLayer;
//其中,
byRoundingCorners:UIRectCornerBottomLeft | UIRectCornerBottomRight
//指定了需要成为圆角的角。该参数是UIRectCorner类型的,可选的值有:
* UIRectCornerTopLeft
* UIRectCornerTopRight
* UIRectCornerBottomLeft
* UIRectCornerBottomRight
* UIRectCornerAllCorners

从名字很容易看出来代表的意思,使用“|”来组合就好了。

十三、设置滑动的时候隐藏navigationBar

navigationController.hidesBarsOnSwipe = Yes;

十四、iOS画虚线
记得先导入QuartzCore框架

#import <QuartzCore/QuartzCore.h>
 
CGContextRef context =UIGraphicsGetCurrentContext();  
CGContextBeginPath(context);  
CGContextSetLineWidth(context, 2.0);  
CGContextSetStrokeColorWithColor(context, [UIColor whiteColor].CGColor);  
CGFloat lengths[] = {10,10};  
CGContextSetLineDash(context, 0, lengths,2);  
CGContextMoveToPoint(context, 10.0, 20.0);  
CGContextAddLineToPoint(context, 310.0,20.0);  
CGContextStrokePath(context);  
CGContextClosePath(context);  

十五、 禁止程序运行时自动锁屏

[[UIApplication sharedApplication] setIdleTimerDisabled:YES];

十六、KVC相关运算,支持操作符
KVC同时还提供了很复杂的函数,主要有下面这些
简单集合运算符
简单集合运算符共有@avg, @count , @max , @min ,@sum5种,不支持自定义。

@interface Book : NSObject
@property (nonatomic,copy)  NSString* name;
@property (nonatomic,assign)  CGFloat price;
@end
@implementation Book
@end
 
 
Book *book1 = [Book new];
book1.name = @"The Great Gastby";
book1.price = 22;
Book *book2 = [Book new];
book2.name = @"Time History";
book2.price = 12;
Book *book3 = [Book new];
book3.name = @"Wrong Hole";
book3.price = 111;
 
Book *book4 = [Book new];
book4.name = @"Wrong Hole";
book4.price = 111;
 
NSArray* arrBooks = @[book1,book2,book3,book4];
NSNumber* sum = [arrBooks valueForKeyPath:@"@sum.price"];
NSLog(@"sum:%f",sum.floatValue);
NSNumber* avg = [arrBooks valueForKeyPath:@"@avg.price"];
NSLog(@"avg:%f",avg.floatValue);
NSNumber* count = [arrBooks valueForKeyPath:@"@count"];
NSLog(@"count:%f",count.floatValue);
NSNumber* min = [arrBooks valueForKeyPath:@"@min.price"];
NSLog(@"min:%f",min.floatValue);
NSNumber* max = [arrBooks valueForKeyPath:@"@max.price"];
NSLog(@"max:%f",max.floatValue);
 
打印结果
2019-07-20 16:45:54.696 KVCDemo[1484:127089] sum:256.000000
2019-07-20 16:45:54.697 KVCDemo[1484:127089] avg:64.000000
2019-07-20 16:45:54.697 KVCDemo[1484:127089] count:4.000000
2019-07-20 16:45:54.697 KVCDemo[1484:127089] min:12.000000

NSArray 快速求总和 最大值 最小值 和 平均值

NSArray *array = [NSArray arrayWithObjects:@"2.0", @"2.3", @"3.0", @"4.0", @"10", nil];
CGFloat sum = [[array valueForKeyPath:@"@sum.floatValue"] floatValue];
CGFloat avg = [[array valueForKeyPath:@"@avg.floatValue"] floatValue];
CGFloat max =[[array valueForKeyPath:@"@max.floatValue"] floatValue];
CGFloat min =[[array valueForKeyPath:@"@min.floatValue"] floatValue];
NSLog(@"%f\n%f\n%f\n%f",sum,avg,max,min);

十七、强制让App直接退出(非闪退,非崩溃)

- (void)exitApplication {
    AppDelegate *app = [UIApplication sharedApplication].delegate;
    UIWindow *window = app.window;
    [UIView animateWithDuration:1.0f animations:^{
        window.alpha = 0;
    } completion:^(BOOL finished) {
        exit(0);
    }];
}

十八、Label行间距

NSMutableAttributedString *attributedString =    
   [[NSMutableAttributedString alloc] initWithString:self.contentLabel.text];
    NSMutableParagraphStyle *paragraphStyle =  [[NSMutableParagraphStyle alloc] init];  
[paragraphStyle setLineSpacing:3];
 
//调整行间距       
[attributedString addAttribute:NSParagraphStyleAttributeName 
                         value:paragraphStyle 
                         range:NSMakeRange(0, [self.contentLabel.text length])];
self.contentLabel.attributedText = attributedString;

十九、CocoaPods pod install/pod update更新慢的问题

pod install –verbose –no-repo-update
pod update –verbose –no-repo-update
如果不加后面的参数,默认会升级CocoaPods的spec仓库,加一个参数可以省略这一步,然后速度就会提升不少。

二十、MRC和ARC混编设置方式
在XCode中targets的build phases选项下Compile Sources下选择 不需要arc编译的文件
双击输入 -fno-objc-arc 即可
MRC工程中也可以使用ARC的类,方法如下:
在XCode中targets的build phases选项下Compile Sources下选择要使用arc编译的文件
双击输入 -fobjc-arc 即可

二十一、把tableview里cell的小对勾的颜色改成别的颜色

_yourTableView.tintColor = [UIColor redColor];

二十二、禁止textField和textView的复制粘贴菜单

-(BOOL)canPerformAction:(SEL)action withSender:(id)sender
{
     if ([UIMenuController sharedMenuController]) {
       [UIMenuController sharedMenuController].menuVisible = NO;
     }
     return NO;
}

二十三、NSString处理技巧
使用场景举例:可以用在处理用户用户输入在UITextField的文本

//待处理的字符串
NSString *string = @" A B  CD   EFG\n MN\n";
 
//字符串替换,处理后的string1 = @"ABCDEF\nMN\n";
NSString *string1 = [string stringByReplacingOccurrencesOfString:@" " withString:@""];
 
//去除两端空格(注意是两端),处理后的string2 = @"A B  CD   EFG\n MN\n";
NSString *string2 = [string stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]];
 
//去除两端回车(注意是两端),处理后的string3 = @" A B  CD   EFG\n MN";
NSString *string3 = [string stringByTrimmingCharactersInSet:[NSCharacterSet newlineCharacterSet]];
 
//去除两端空格和回车(注意是两端),处理后的string4 = @"A B  CD   EFG\n MN";
NSString *string4 = [string stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];

二十四、主线程操作UI(对UI进行更新只能在主线程进行)
解释:所谓的在主线程更新UI、操作UI,大致的意思就是设置UILabel的text或者设置tabbar的badgeValue,设置UIImageView的image等等。
回到主线程方式1:

[self performSelectorOnMainThread:@selector(updateImage:) withObject:data waitUntilDone:YES];

performSelectorOnMainThread方法是NSObject的分类方法,每个NSObject对象都有此方法,
它调用的selector方法是当前调用控件的方法,例如使用UIImageView调用的时候selector就是UIImageView的方法
Object:代表调用方法的参数,不过只能传递一个参数(如果有多个参数请使用对象进行封装)
waitUntilDone:是否线程任务完成执行
回到主线程方式2:

dispatch_async(dispatch_get_main_queue(), ^{
    //更新UI的代码
});

GCD的方法,注意不要在主线程掉用。

二十五、两种方法删除NSUserDefaults所有记录

//方法一
NSString *appDomain = [[NSBundle mainBundle] bundleIdentifier];
[[NSUserDefaults standardUserDefaults] removePersistentDomainForName:appDomain];
 
//方法二
- (void)resetDefaults {
    NSUserDefaults * defs = [NSUserDefaults standardUserDefaults];
    NSDictionary * dict = [defs dictionaryRepresentation];
    for (id key in dict) {
        [defs removeObjectForKey:key];
    }
    [defs synchronize];
}

二十六、NSLog 输出格式集合

• %@     对象
• %d, %i    整数
• %u      无符整形
• %f       浮点/双字
• %x, %X   二进制整数
• %o      八进制整数
• %zu     size_t
• %p      指针
• %e      浮点/双字 (科学计算)
• %g      浮点/双字
• %s       C 字符串
• %.*s      Pascal字符串
• %c       字符
• %C       unichar
• %lld      64位长整数(long long)
• %llu      无符64位长整数
• %Lf       64位双字

二十七、常用GCD总结
为了方便地使用GCD,苹果提供了一些方法方便我们将block放在主线程 或 后台线程执行,或者延后执行。使用的例子如下:

    //  后台执行: 
     dispatch_async(dispatch_get_global_queue(0, 0), ^{ 
          // something 
     }); 
     // 主线程执行: 
     dispatch_async(dispatch_get_main_queue(), ^{ 
          // something 
     }); 
     // 一次性执行: 
     static dispatch_once_t onceToken; 
     dispatch_once(&onceToken, ^{ 
         // code to be executed once 
     }); 
     // 延迟2秒执行: 
     double delayInSeconds = 2.0; 
     dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, delayInSeconds * NSEC_PER_SEC); 
     dispatch_after(popTime, dispatch_get_main_queue(), ^(void){ 
         // code to be executed on the main queue after delay 
     }); 

dispatch_queue_t 也可以自己定义,如要要自定义queue,可以用dispatch_queue_create方法,示例如下:

dispatch_queue_t urls_queue = dispatch_queue_create("blog.devtang.com", NULL); 
    dispatch_async(urls_queue, ^{ 
         // your code 
}); 
dispatch_release(urls_queue); 

另外,GCD还有一些高级用法,例如让后台2个线程并行执行,然后等2个线程都结束后,再汇总执行结果。这个可以用dispatch_group, dispatch_group_async 和 dispatch_group_notify来实现,示例如下:

dispatch_group_t group = dispatch_group_create(); 
    dispatch_group_async(group, dispatch_get_global_queue(0,0), ^{ 
    // 并行执行的线程一 
}); 
dispatch_group_async(group, dispatch_get_global_queue(0,0), ^{ 
    // 并行执行的线程二 
}); 
dispatch_group_notify(group, dispatch_get_global_queue(0,0), ^{ 
    // 上面的线程走完成后,最后通知走次block,保证这部分代码最后执行 
}); 

二十八、 iOS中的随机数
生成0-x之间的随机正整数

int value =arc4random_uniform(x + 1);

生成随机正整数

int value = arc4random()

通过arc4random() 获取0到x-1之间的整数的代码如下:

int value = arc4random() % x; 

获取1到x之间的整数的代码如下:

int value = (arc4random() % x) + 1; 

最后如果想生成一个浮点数,可以在项目中定义如下宏:

#define ARC4RANDOM_MAX      0x100000000 

然后就可以使用arc4random() 来获取0到100之间浮点数了(精度是rand()的两倍),代码如下:

double val = floorf(((double)arc4random() / ARC4RANDOM_MAX) * 100.0f);

二十九、系统自带的UITableViewCell,其中cell.accessoryView可以自定义控件

if (indexPath.section == 2 && indexPath.row == 0) {
    cell.accessoryView = [[UISwitch alloc] init];
} else {
    cell.accessoryType = UITableViewCellAccessoryDisclosureIndicator;
}

三十、isKindOfClass, isMemberOfClass的用法区分

-(BOOL) isKindOfClass: classObj判断是否是这个类或者这个类的子类的实例
-(BOOL) isMemberOfClass: classObj 判断是否是这个类的实例

实例一:

   Person *person = [[Person alloc] init];      //父类
   Teacher *teacher = [[Teacher alloc] init];  //子类
 
   //YES   
   if ([teacher isMemberOfClass:[Teacher class]]) {  
        NSLog(@"teacher Teacher类的成员");  
   }  
   //NO   
   if ([teacher isMemberOfClass:[Person class]]) {  
       NSLog(@"teacher Person类的成员");  
   }  
   //NO   
   if ([teacher isMemberOfClass:[NSObject class]]) {  
       NSLog(@"teacher NSObject类的成员");  
   }  

实例二:

Person *person = [[Person alloc] init];  
Teacher *teacher = [[Teacher alloc] init];  
 
//YES   
if ([teacher isKindOfClass:[Teacher class]]) {  
    NSLog(@"teacher 是 Teacher类或Teacher的子类");  
}  
//YES   
if ([teacher isKindOfClass:[Person class]]) {  
    NSLog(@"teacher 是 Person类或Person的子类");  
}  
//YES   
if ([teacher isKindOfClass:[NSObject class]]) {  
    NSLog(@"teacher 是 NSObject类或NSObject的子类");  
}  

isMemberOfClass判断是否是属于这类的实例,是否跟父类有关系他不管,所以isMemberOfClass指到父类时才会为NO;

三十一、UIView的clipsTobounds属性

view2添加view1到中,如果view2大于view1,或者view2的坐标不全在view1的范围内,view2是盖着view1的,意思就是超出的部份也会画出来,UIView有一个属性,clipsTobounds 默认情况下是NO。如果,我们想要view2把超出的那部份现实出来,就得改变它的父视图也就view1的clipsTobounds属性值。view1.clipsTobounds = YES;
可以很好地解决覆盖的问题

三十二、UILabel显示HTML文本(IOS7以上)

NSString * htmlString = @"<html><body> Some html string \n <font size=\"13\" color=\"red\">This is some text!</font> </body></html>";
NSAttributedString * attrStr = [[NSAttributedString alloc] initWithData:[htmlString dataUsingEncoding:NSUnicodeStringEncoding] options:@{ NSDocumentTypeDocumentAttribute: NSHTMLTextDocumentType } documentAttributes:nil error:nil];
UILabel * myLabel = [[UILabel alloc] initWithFrame:self.view.bounds];
myLabel.attributedText = attrStr;
[self.view addSubview:myLabel];

三十三、添加pch文件的步聚

1:创建新文件 ios->other->PCH file,创建一个pch文件:“工程名-Prefix.pch”:
2:将building setting中的precompile header选项的路径添加“$(SRCROOT)/项目名称/pch文件名”(例如:$(SRCROOT)/LotteryFive/LotteryFive-Prefix.pch)
3:将Precompile Prefix Header为YES,预编译后的pch文件会被缓存起来,可以提高编译速度

三十四、Masonry

a:make.equalTo 或 make.greaterThanOrEqualTo (至多) 或 make.lessThanOrEqualTo(至少)
 
make.left.greaterThanOrEqualTo(label);
make.left.greaterThanOrEqualTo(label.mas_left);
 
//width >= 200 && width <= 400
make.width.greaterThanOrEqualTo(@200);
make.width.lessThanOrEqualTo(@400)
b:masequalTo 和 equalTo 区别:masequalTo 比equalTo多了类型转换操作,一般来说,大多数时候两个方法都是 通用的,但是对于数值元素使用mas_equalTo。对于对象或是多个属性的处理,使用equalTo。特别是多个属性时,必须使用equalTo
 
c:一些简便赋值
 
// make top = superview.top + 5, left = superview.left + 10,
// bottom = superview.bottom - 15, right = superview.right - 20
make.edges.equalTo(superview).insets(UIEdgeInsetsMake(5, 10, 15, 20))
 
// make width and height greater than or equal to titleLabel
make.size.greaterThanOrEqualTo(titleLabel)
 
// make width = superview.width + 100, height = superview.height - 50
make.size.equalTo(superview).sizeOffset(CGSizeMake(100, -50))
 
// make centerX = superview.centerX - 5, centerY = superview.centerY + 10
make.center.equalTo(superview).centerOffset(CGPointMake(-5, 10))
 
d:and关键字运用
 
make.left.right.and.bottom.equalTo(superview); 
make.top.equalTo(otherView);
e:优先;优先权(.priority,.priorityHigh,.priorityMedium,.priorityLow)
 
.priority允许您指定一个确切的优先级
.priorityHigh 等价于UILayoutPriorityDefaultHigh
.priorityMedium 介于高跟低之间
.priorityLow 等价于UILayoutPriorityDefaultLow
 
实例:
make.left.greaterThanOrEqualTo(label.mas_left).with.priorityLow();
make.top.equalTo(label.mas_top).with.priority(600);
g:使用mas_makeConstraints创建constraint后,你可以使用局部变量或属性来保存以便下次引用它;如果创建多个constraints,你可以采用数组来保存它们
 
// 局部或者全局
@property (nonatomic, strong) MASConstraint *topConstraint;
 
// 创建约束并赋值
[view1 mas_makeConstraints:^(MASConstraintMaker *make) {
    self.topConstraint = make.top.equalTo(superview.mas_top).with.offset(padding.top);
    make.left.equalTo(superview.mas_left).with.offset(padding.left);
}];
 
// 过后可以直接访问self.topConstraint
[self.topConstraint uninstall];
 
h:mas_updateConstraints更新约束,有时你需要更新constraint(例如,动画和调试)而不是创建固定constraint,可以使用mas_updateConstraints方法
 
 
- (void)updateConstraints {
    [self.growingButton mas_updateConstraints:^(MASConstraintMaker *make) {
        make.center.equalTo(self);
        make.width.equalTo(@(self.buttonSize.width)).priorityLow();
        make.height.equalTo(@(self.buttonSize.height)).priorityLow();
        make.width.lessThanOrEqualTo(self);
        make.height.lessThanOrEqualTo(self);
    }];
 
    //调用父updateConstraints
    [super updateConstraints];
}
 
i:mas_remakeConstraints更新约束,mas_remakeConstraints与mas_updateConstraints比较相似,都是更新constraint。不过,mas_remakeConstraints是删除之前constraint,然后再添加新的constraint(适用于移动动画);而mas_updateConstraints只是更新constraint的值。
 
 
- (void)changeButtonPosition {
    [self.button mas_remakeConstraints:^(MASConstraintMaker *make) {
        make.size.equalTo(self.buttonSize);
 
        if (topLeft) {
       make.top.and.left.offset(10);
        } else {
       make.bottom.and.right.offset(-10);
        }
    }];
}

三十五、iOS中的round/roundf/ceil/ceilf/floor/floorf

round:如果参数是小数,则求本身的四舍五入。
ceil:如果参数是小数,则求最小的整数但不小于本身(向上取,ceil的英文意思有天花板的意思)
floor:如果参数是小数,则求最大的整数但不大于本身(向下取,floor的英文意思有地板的意思)

Example:如果值是3.4的话,则
3.4 – round 3.000000
– ceil 4.000000
– floor 3.00000

三十六、汉字转为拼音

- (NSString *)Charactor:(NSString *)aString getFirstCharactor:(BOOL)isGetFirst
{
    //转成了可变字符串
    NSMutableString *str = [NSMutableString stringWithString:aString];
    //先转换为带声调的拼音
    CFStringTransform((CFMutableStringRef)str,NULL, kCFStringTransformMandarinLatin,NO);
    //再转换为不带声调的拼音
    CFStringTransform((CFMutableStringRef)str,NULL, kCFStringTransformMandarinLatin,NO);
    CFStringTransform((CFMutableStringRef)str, NULL, kCFStringTransformStripDiacritics, NO);
    NSString *pinYin = [str capitalizedString];
    //转化为大写拼音
    if(isGetFirst)
    {
        //获取并返回首字母
        return [pinYin substringToIndex:1];
    }
    else
    {
        return pinYin;
    }
}

三十七、去除编译器警告

a:方法弃用告警

#pragma clang diagnostic push  
 
#pragma clang diagnostic ignored "-Wdeprecated-declarations"      
//会报警告的方法,比如SEL 
[TestFlight setDeviceIdentifier:[[UIDevice currentDevice] uniqueIdentifier]];  
 
#pragma clang diagnostic pop  

b:未使用变量
#pragma clang diagnostic push   
#pragma clang diagnostic ignored "-Wunused-variable"   
int a;   
#pragma clang diagnostic pop 

三十八、数组逆序遍历

1、枚举法

NSArray *array = @[@"1",@"2",@"3",@"5",@"6"];
[array enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
    NSLog(@"%@",obj);
}];

2、for循环

NSArray*array=@[@"1",@"2",@"3",@"5",@"6"];
for (NSInteger index = array.count-1; index>=0; index--) {
    NSLog(@"%@",array[index]);
}

三十九、后台float类型数值的精度问题

比如后台给我一个4.003621,不是String类型的,是double类型的,然后我放到lable上显示

myLabel.text = [NSString stringWithFormat:@"%f",model.price];

结果就显示错了,4.004,后来想好办,直接formatter成字符串,就可以了,结果formatter后的字符串也不是我想的4.003621这样。一般这样的精确数字后台给多少,前端显示多少,不敢有毫厘差距,特别是金融和银行,不敢有半毛钱的差距。解决办法有两个。
第一,用%g去显示,这样可以保证后台给的啥样的数据,显示出来还是啥样的,不会做任何四舍五入或者切割
–%g: 自动选择 %e 或者 %f 各式;

myLabel.text = [NSString stringWithFormat:@"%g",model.price];

这样就能保证后台给的多少就显示多少,充分展示原数据。
第二、用科学计数法,自行百度

四十、字符串中既有汉字,又具有数字,想获取第一组数字

    NSString *s3 = @"2.07我的7哈哈";
    CGFloat F3 = s3.floatValue;

这样是可以直接取到2.07,但是2.07之前必须没有汉字或者英文字符,换句话说,必须数字开头。(PS:好像还是有计算机的那个毛病,获得的数据不是2.07,而是2.069999999999),哈哈,解决方案第三十九条。

四十一、UIImage 有一个属性叫size 。

当为对象Image赋值一张图片以后,Image会自动生成一个size。 那么也就是说你可以用image.size.x 来设置其父视图的size。即:

UIImageView *imageView = [[UIImageView alloc]initWithImage:@"text.png"];  

这样不用给定frame 系统也可以显示出带图片大小的ImageView。

四十二、OC语法细节:不允许直接修改OC对象的结构体属性的成员。

四十三、图片缩放不止可以用手势,也可以用ScrollView

self.imageView = [[UIImageView alloc] initWithImage:[UIImage imageNamed:@"1.png"]];
[self.scrollView addSubview:self.imageView];
self.scrollView.contentSize = self.imageView.image.size;
self.scrollView.delegate = self;
//设置缩放比例
self.scrollView.maximumZoomScale = 2.0;
self.scrollView.minimumZoomScale = 0.1;

/**
 *这个方法的返回值决定了要缩放的内容
 *返回值 只能是scrollerView的子控件或者孙子控件
 */
- (UIView *)viewForZoomingInScrollView:(UIScrollView *)scrollView
{
    return self.imageView;
}
 
- (void)scrollViewDidZoom:(UIScrollView *)scrollView
{
    //可以查看缩放比例
    NSLog(@"scrollView.zoomScale->%f",scrollView.zoomScale);
}

viewForZoomingInScrollView不仅仅可以缩放图片,只要是继承自UIIView的控件都可以进行缩放。

四十四、LayoutIfNeeded小理解

这个方法和另一个方法配对的,setNeedLayout和layoutIfNeed,还有一个关联的方法是layoutSubviews,在我们没有任何干预的情况下,一个view的fram或bounds发生变化时,系统会设置一个flag给这个view,当下一个渲染时机到来时系统会重新按新的布局来渲染视图。setNeedLayout就是我们主动为这个视图设置一个flag,告诉系统这个视图再下一个时机到来时要重新渲染,而layoutIfNeed则是告诉系统,如果设置了flag那么不用等待时机到来了,直接渲染吧。而layoutSubviews这个方法是系统调用的,我们不需要主动调用,我们只需要调用layoutIfNeed就可以了,让系统判断是否在当前时机下立即渲染。

另外,当想知道tableView reloadData后的contentSize的话可以在reloadData后用这两个方法,然后就可以直接提取contentSize了。

四十五、通知监听APP生命周期

UIApplicationDidEnterBackgroundNotification 应用程序进入后台
UIApplicationWillEnterForegroundNotification 应用程序将要进入前台
UIApplicationDidFinishLaunchingNotification 应用程序完成启动
UIApplicationDidFinishLaunchingNotification 应用程序由挂起变的活跃
UIApplicationWillResignActiveNotification 应用程序挂起(有电话进来或者锁屏)
UIApplicationDidReceiveMemoryWarningNotification 应用程序收到内存警告
UIApplicationDidReceiveMemoryWarningNotification 应用程序终止(后台杀死、手机关机等)
UIApplicationSignificantTimeChangeNotification 当有重大时间改变(凌晨0点,设备时间被修改,时区改变等)
UIApplicationWillChangeStatusBarOrientationNotification 设备方向将要改变
UIApplicationDidChangeStatusBarOrientationNotification 设备方向改变
UIApplicationWillChangeStatusBarFrameNotification 设备状态栏frame将要改变
UIApplicationDidChangeStatusBarFrameNotification 设备状态栏frame改变
UIApplicationBackgroundRefreshStatusDidChangeNotification 应用程序在后台下载内容的状态发生变化
UIApplicationProtectedDataWillBecomeUnavailable 本地受保护的文件被锁定,无法访问
UIApplicationProtectedDataWillBecomeUnavailable 本地受保护的文件可用了

作者简介: 就职于甜橙金融o2o事业部,负责iOS前端开发工作。我们学习新知识的同时,基础知识牢固掌握必不可少,有了扎实的基础知识,才能更容易像更高的点出发。

如需转载,请注明出处,谢谢~~~

上一篇下一篇

猜你喜欢

热点阅读