ios琐碎笔记

  • 抛出异常&异常处理
  • NSInvocation执行多参数方法
  • 强制消除Xcode警告
  • UI控件对齐方式属性
  • UINavigationItem,UIBarButtonItem,UITabBarItem,UINavigationBar,UITabBar,UITabBarButton属性总结
  • UIButton相关设置
  • 切换控制器
  • 查看Class所有的成员变量
  • runtime
  • iOS修改项目名称,Swift命名空间
  • 字符串转emoji
  • CADisplayLink定时器

  • 抛出异常&异常处理
    swift3.0异常处理
    guard let jsonPath = Bundle.main.path(forResource: "Contents.json", ofType: nil) else {
    return
    }
    let jsonUrl = URL(fileURLWithPath: jsonPath)

          //方式一:try方式 需要在do{}内
          do {
              let jsonData =  try Data.init(contentsOf: jsonUrl, options: .mappedRead)
              let anyObject = try JSONSerialization.jsonObject(with: jsonData, options: .mutableContainers)
              guard let dictArray = anyObject as? [String : AnyObject] else {
                  return
              }
              for dict in dictArray {
                  print(dict)
              }
          } catch {
              //抛出异常
              return
          }
          //方式二:try? 如果该方法出现了异常,则该方法返回nil.如果没有异常,则返回对应的对象
          guard let jsonData =  try? Data.init(contentsOf: jsonUrl, options: .mappedRead) else {
              return
          }
          guard let anyObject = try? JSONSerialization.jsonObject(with: jsonData, options: .mutableContainers) else {
              return
          }
          guard let dictArray = anyObject as? [String : AnyObject] else {
              return
          }
          for dict in dictArray {
              print(dict)
          }
          //方式三:try! 直接告诉系统,该方法没有异常,如果该方法出现了异常,那么程序会报错(崩溃)
          let jsonData3 = try! Data.init(contentsOf: jsonUrl, options: .mappedRead)
          let anyObject3 = try! JSONSerialization.jsonObject(with: jsonData3, options: .mutableContainers)
          guard let dictArray3 = anyObject3 as? [String : AnyObject] else {
              return
          }
          for dict in dictArray3 {
              print(dict)
          }
    
    
    
      1. @throw [NSException exceptionWithName:@"错误" reason:@"方法找不到" userInfo:nil];
      2. [NSException raise:@"错误" format:@"%@方法找不到", NSStringFromSelector(selector)];
      3. 
    void handleException(NSException *exception)
     {  
     }
    -(BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
          // 设置捕捉异常的回调
          NSSetUncaughtExceptionHandler(handleException);
    
             return YES;
     }
      4.   
       @try {
       }
      @catch (NSException *exception) {
       }
       @finally {
       }
    

  • NSInvocation执行多参数方法

      - (id)performSelector:(SEL)selector withObjects:(NSArray *)objects
      {
          // 方法签名(方法的描述)
          NSMethodSignature *signature = [[self class] instanceMethodSignatureForSelector:selector];
          if (signature == nil) {
              [NSException raise:@"错误" format:@"%@方法找不到", NSStringFromSelector(selector)];
          }
          
          // NSInvocation : 利用一个NSInvocation对象包装一次方法调用(方法调用者、方法名、方法参数、方法返回值)
          NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature];
          invocation.target = self;
          invocation.selector = selector;
          
          // 设置参数
          NSInteger paramsCount = signature.numberOfArguments - 2; // 除self、_cmd以外的参数个数
          paramsCount = MIN(paramsCount, objects.count);
          for (NSInteger i = 0; i < paramsCount; i++) {
              id object = objects[i];
              if ([object isKindOfClass:[NSNull class]]) continue;
              [invocation setArgument:&object atIndex:i + 2];
          }
          
          // 调用方法
          [invocation invoke];
          
          // 获取返回值
          id returnValue = nil;
          if (signature.methodReturnLength) { // 有返回值类型,才去获得返回值
              [invocation getReturnValue:&returnValue];
          }
          
          return returnValue;
      }
    

  • 强制消除Xcode警告
    #pragma clang diagnostic push //开始
    #pragma clang diagnostic ignored "-Warc-performSelector-leaks" //错误类型
    #pragma clang diagnostic pop //结束

  • UI控件对齐方式属性

      四个容易混淆的属性:
      一. textAligment : 文字的水平方向的对齐方式
      1> 取值
      NSTextAlignmentLeft      = 0,    // 左对齐
      NSTextAlignmentCenter    = 1,    // 居中对齐
      NSTextAlignmentRight    = 2,    // 右对齐
      
      2> 哪些控件有这个属性 : 一般能够显示文字的控件都有这个属性
      * UITextField
      * UILabel
      * UITextView
      
      二. contentVerticalAlignment : 内容的垂直方向的对齐方式
      1> 取值
      UIControlContentVerticalAlignmentCenter  = 0, // 居中对齐
      UIControlContentVerticalAlignmentTop     = 1, // 顶部对齐
      UIControlContentVerticalAlignmentBottom  = 2, // 底部对齐
      UIControlContentVerticalAlignmentFill   = 3  //填充
      
      2> 哪些控件有这个属性 : 继承自UIControl的控件或者UIControl本身
      * UIControl
      * UIButton
      * UITextField
      * ...
      
      三. contentHorizontalAlignment : 内容的水平方向的对齐方式
      1> 取值
      UIControlContentHorizontalAlignmentCenter = 0, // 居中对齐
      UIControlContentHorizontalAlignmentLeft   = 1, // 左对齐
      UIControlContentHorizontalAlignmentRight  = 2, // 右对齐
      UIControlContentHorizontalAlignmentFill   = 3  //填充
      
      2> 哪些控件有这个属性 : 继承自UIControl的控件或者UIControl本身
      * UIControl
      * UIButton
      * UITextField
      * ...
      
     四. contentMode : 内容模式(控制内容的对齐方式), 一般对UIImageView很有用
      1> 取值
      /**
       规律:
       1.Scale : 图片会拉伸
       2.Aspect : 图片会保持原来的宽高比
       */
      // 前3个情况, 图片都会拉伸
      // (默认)拉伸图片至填充整个UIImageView(图片的显示尺寸会跟UIImageView的尺寸一样)
      UIViewContentModeScaleToFill,
      // 按照图片原来的宽高比进行伸缩, 伸缩至适应整个UIImageView(图片的内容不能超出UIImageView的尺寸范围)
      UIViewContentModeScaleAspectFit,
      // 按照图片原来的宽高比进行伸缩, 伸缩至 图片的宽度和UIImageView的宽度一样 或者 图片的高度和UIImageView的高度一样
      UIViewContentModeScaleAspectFill,
      
      // 后面的所有情况, 都会按照图片的原来尺寸显示, 不会进行拉伸
      UIViewContentModeRedraw,  // 当控件的尺寸改变了, 就会重绘一次(重新调用setNeedsDisplay, 调用drawRect:)
      UIViewContentModeCenter,
      UIViewContentModeTop,
      UIViewContentModeBottom,
      UIViewContentModeLeft,
      UIViewContentModeRight,
      UIViewContentModeTopLeft,
      UIViewContentModeTopRight,
      UIViewContentModeBottomLeft,
      UIViewContentModeBottomRight,
      
      2> 哪些控件有这个属性 : 所有UI控件都有
      
      五. 如果有多个属性的作用冲突了, 只有1个属性有效(就近原则)
    

  • UINavigationItem属性总结

      一、UINavigationItem
      1> 获得方式
      self.navigationItem // self是指控制器
      
      2> 作用
      可以用来设置当前控制器顶部导航栏的内容
      // 设置导航栏中间的内容
      self.navigationItem.title
      self.navigationItem.titleView
      
      二、UIBarButtonItem
      1> 用在什么地方
      // 设置导航栏左上角的内容
      self.navigationItem.leftBarButtonItem
      // 设置导航栏右上角的内容
      self.navigationItem.rightBarButtonItem
      
      2> 作用
      相当于一个按钮
      
      三、UITabBarItem
      1> 获得方式
      self.tabBarItem // self是指控制器
      
      2> 作用
      可以用来设置当前控制器对应的选项卡标签的内容
      // 标签的标题
      self.tabBarItem.title
      // 标签的图标
      self.tabBarItem.image
      // 标签的选中图标
      self.tabBarItem.selectdImage
      
      四、UINavigationBar
      1. 导航控制器顶部的栏(UI控件)
      2. UINavigationBar上面显示什么内容, 取决于当前控制器的navigationItem属性
      3. UINavigationBar是view, navigationItem是model
      4. 由navigationItem给UINavigationBar提供显示的数据
      
      UINavigationBar *barGlob = [UINavigationBar appearance];//全局设置bar
      UINavigationBar *bar = [UINavigationBar appearanceWhenContainedInInstancesOfClasses:[self class]];//设置self导航条的bar
      [bar setBackgroundImage:[UIImage imageNamed:@""] forBarMetrics:UIBarMetricsDefault];
      //forBarMetrics有点类似于按钮的for state状态,即什么状态下显示
      //UIBarMetricsDefault-竖屏横屏都有,横屏导航条变宽,则自动repeat图片
      //UIBarMetricsCompact-竖屏没有,横屏有,相当于之前老iOS版本里地UIBarMetricsLandscapePhone
    
      
      五、UITabBar
      1. UITabBarController底部的选项卡条
      
      六、UITabBarButton
      1. UITabBar底部的每一个标签
      2. 每一个UITabBarButton里面显示什么内容,取决于当前控制器的tabBarItem属性
      3. UITabBarButton是view, tabBarItem是model
      4. 由tabBarItem给UITabBarButton提供显示的数据
    

  • UIButton相关设置
    // 设置按钮的尺寸为背景图片的尺寸
    button.size = button.currentBackgroundImage.size;
    //取消点击效果
    btn.adjustsImageWhenHighlighted = false
    / 默认按钮的尺寸跟背景图片一样大
    // sizeToFit:默认会根据按钮的背景图片或者image和文字计算出按钮的最合适的尺寸
    [btn sizeToFit];

  • 切换控制器

      UIViewController *vc = [UIViewController new];
      //push
      [self.navigationController pushViewController:vc animated:YES];
      //modal
      [self presentViewController:vc animated:YES completion:nil];
      //添加到win
      UIViewController *rootVc = [UIApplication sharedApplication].keyWindow.rootViewController;
      rootVc = vc;
    

  • 查看Class所有的成员变量

      + (void)initialize{
          unsigned int count = 0;
          //拷贝所有的成员变量
          Ivar *ivars = class_copyIvarList([UIViewController class], &count);
          for (int i = 0; i < count; i++) {
              //取出成员变量
              Ivar ivar = *(ivars + i);
              NSLog(@"%s", ivar_getName(ivar));
          }
          //释放
          free(ivars);
      }
    

  • runtime
    method1与method2替换
    Method method1 = class_getInstanceMethod(self, @selector(method1));
    Method method2 = class_getInstanceMethod(self, @selector(method2));
    method_exchangeImplementations(method1, method2);

  • iOS修改项目名称
    build Settings -> product name
    Swift命名空间:let ns = NSBundle.mainBundle().infoDictionary!["CFBundleExecutable"] as! String

  • 字符串转emoji
    override func emoji() {

      // emoji表情对应的十六进制
      let code = "0x2600"
      
      // 1.从字符串中取出十六进制的数
      // 创建一个扫描器, 扫描器可以从字符串中提取我们想要的数据
      let scanner = NSScanner(string: code)
    
      // 2.将十六进制转换为字符串
      var result:UInt32 = 0
      scanner.scanHexInt(&result)
    
      // 3.将十六进制转换为emoji字符串
      let emojiStr = Character(UnicodeScalar(result))
      
      // 3.显示
      print(emojiStr)
    }
    

  • CADisplayLink定时器
    //NSTimer很少用于绘图,因为调度优先级比较低,并不会准时调用
    //CADisplayLink:每次屏幕刷新的时候就会调用,屏幕一般一秒刷新60次
    //[self setNeedsDisplay] 注意:这个方法并不会马上调用drawRect,其实这个方法只是给当前控件添加刷新的标记,等下一次屏幕刷新的时候才会调用drawRect
    CADisplayLink *link = [CADisplayLink displayLinkWithTarget:self selector:@selector(timeChange)];
    // 添加主运行循环
    [link addToRunLoop:[NSRunLoop mainRunLoop] forMode:NSDefaultRunLoopMode];
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,457评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,837评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,696评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,183评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,057评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,105评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,520评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,211评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,482评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,574评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,353评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,897评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,489评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,683评论 2 335

推荐阅读更多精彩内容