微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

ios – Xcode强制Swift可选解包两次(!!)

我正在分类UIStoryboardSegue,每次尝试使用两个UIView之一时,Xcode使我添加两个可选的解包(!!),如:
let sourceView = self.sourceViewController.view
sourceView!!.frame = CGRect(x: 0,y: 0,width: screenWidth,height: screenHeight

要么

let sourceView = self.sourceViewController.view!
sourceView!.frame = CGRect(x: 0,height: screenHeight

要么

self.sourceViewController.view!!.frame = CGRect(x: 0,height: screenHeight

我想知道有人可以解释为什么会这样.

解决方法

UIStoryboardSegue的sourceViewController属性键入为AnyObject,并且作为Objective-C兼容性功能,一旦导入Foundation,事情会变得与AnyObject有点怪异.

而不是在AnyObject类型上查找方法,Swift会查找Objective-C选择器,就像Objective-C与id类型一样.任何类的任何选择器都是公平的游戏:如果需要,您可以尝试在对象上调用activeProcessorCount,即使是nsprocessInfo选择器,编译器也可以让它执行. (由于明显的原因,它在运行时会失败).这被称为动态调度,与静态调度(Swift中的正常调用机制)相反.

尽管如此,动态调度有一点是它总是添加一层隐含的包装.如果您有一个返回String的Objective-C属性,动态调度将返回一个String!

当多个类声明具有相同名称但不同返回类型(或不同参数,但我们对这种情况不感兴趣)的选择器时,事情会变得毛茸茸.我不知道编译器如何选择它知道的许多相同命名的选择器.无论哪种方式,它都会选择一个,除非您将对象转换为更精确的类型,否则Swatch编译器将只允许您使用静态调度.

使用swiftc的-dump-ast参数,我们可以看到一个像编译器解析表达式的类似lisp的表示:

(pattern_binding_decl
  (pattern_named type='UIView?!' 'sourceView')
  (dynamic_member_ref_expr type='UIView?!' location=MySegue.swift:15:46 range=[MySegue.swift:15:25 - line:15:46] decl=UIKit.(file).UIGestureRecognizer.view
    (member_ref_expr type='AnyObject' location=MySegue.swift:15:25 range=[MySegue.swift:15:25 - line:15:25] decl=UIKit.(file).UIStoryboardSegue.sourceViewController
      (derived_to_base_expr implicit type='UIStoryboardSegue' location=MySegue.swift:15:20 range=[MySegue.swift:15:20 - line:15:20]
        (declref_expr type='Segue' location=MySegue.swift:15:20 range=[MySegue.swift:15:20 - line:15:20] decl=xxx.(file).Segue.func decl.self@MySegue.swift:14:7 specialized=no)))))

有很多的问题,但是您可以看到它生成一个dynamic_member_ref_expr而不是一个member_ref_expr.如果您滚动到右侧的“attribute”,您将看到它正在使用UIGestureRecognizer的视图属性(declared as UIView?),以便该表达式返回一个UIView?!

相反,UIViewController的视图属性被声明为UIView!如果编译器选择了这个选择器,那么你最终会使用UIView !!,而只需要一个级别的显式展开.

您可以(并且必须在这样的实例中)明确地展开隐式展开的值.用sourceView作为UIView?!,第一个!解开UIView?进入UIView?,第二个解开UIView?成为终于可用的UIView.这就是为什么你需要两个惊叹号.

声明用于动态调度的选择器的类是不相关的,只要目标对象实现它,接受兼容参数,并返回兼容类型. UIView的?和UIView!在二进制级别是兼容的,所以最终你的程序仍然运行.然而,如果你不知何故期待一个String或另一个无关的类型,你可能会惊喜.在我看来,你应该尽可能避免动态调度.

tl; dr:如果将sourceViewController转换为UIViewController,您将获得正确的视图属性定义,并且根本不需要解开它.

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


当我们远离最新的 iOS 16 更新版本时,我们听到了困扰 Apple 最新软件的错误和性能问题。
欧版/美版 特别说一下,美版选错了 可能会永久丧失4G,不过只有5%的概率会遇到选择运营商界面且部分必须连接到iTunes才可以激活
一般在接外包的时候, 通常第三方需要安装你的app进行测试(这时候你的app肯定是还没传到app store之前)。
前言为了让更多的人永远记住12月13日,各大厂都在这一天将应用变灰了。那么接下来我们看一下Flutter是如何实现的。Flutter中实现整个App变为灰色在Flutter中实现整个App变为灰色是非常简单的,只需要在最外层的控件上包裹ColorFiltered,用法如下:ColorFiltered(颜色过滤器)看名字就知道是增加颜色滤镜效果的,ColorFiltered( colorFilter:ColorFilter.mode(Colors.grey, BlendMode.
flutter升级/版本切换
(1)在C++11标准时,open函数的文件路径可以传char指针也可以传string指针,而在C++98标准,open函数的文件路径只能传char指针;(2)open函数的第二个参数是打开文件的模式,从函数定义可以看出,如果调用open函数时省略mode模式参数,则默认按照可读可写(ios_base:in | ios_base::out)的方式打开;(3)打开文件时的mode的模式是从内存的角度来定义的,比如:in表示可读,就是从文件读数据往内存读写;out表示可写,就是把内存数据写到文件中;
文章目录方法一:分别将图片和文字置灰UIImage转成灰度图UIColor转成灰度颜色方法二:给App整体添加灰色滤镜参考App页面置灰,本质是将彩色图像转换为灰度图像,本文提供两种方法实现,一种是App整体置灰,一种是单个页面置灰,可结合具体的业务场景使用。方法一:分别将图片和文字置灰一般情况下,App页面的颜色深度是24bit,也就是RGB各8bit;如果算上Alpha通道的话就是32bit,RGBA(或者ARGB)各8bit。灰度图像的颜色深度是8bit,这8bit表示的颜色不是彩色,而是256
领导让调研下黑(灰)白化实现方案,自己调研了两天,根据网上资料,做下记录只是学习过程中的记录,还是写作者牛逼