swift - Swift 中需要奇怪的转换

标签 swift

我在 WWDC2014 上观看了一些视频并尝试编写我喜欢的代码,但我注意到的一件奇怪的事情是 Swift 一直生我的气并希望我转换为不同的数字类型。这很容易,但在 WWDC 的视频中,他们不需要这样做。以下是“Interface Builder 的新功能”中的示例:

-M_PI/2 一直给我错误:“找不到接受提供的参数的‘/’的重载”

有没有人有解决这个问题的方法,它不只是涉及转换,因为显然还有另一种方法可以做到这一点?对于与此类似的问题,我还有更多示例。

if !ringLayer {
            ringLayer = CAShapeLayer()

            let innerRect = CGRectInset(bounds, lineWidth / 2.0, lineWidth / 2.0)
            let innerPath = UIBezierPath(ovalInRect: innerRect)

            ringLayer.path = innerPath.CGPath
            ringLayer.fillColor = nil
            ringLayer.lineWidth = lineWidth
            ringLayer.strokeColor = UIColor.blueColor().CGColor
            ringLayer.anchorPoint = CGPointMake(0.5, 0.5)
            ringLayer.transform = CATransform3DRotate
                 (ringLayer.transform, -M_PI/2, 0, 0, 1)

            layer.addSublayer(ringLayer)
        }
        ringLayer.frame = layer.bounds

最佳答案

编辑:注意:CGFloat 在 beta 4 中发生了变化,特别是为了更容易处理这种 32/64 位差异。阅读发行说明,现在不要将以下内容当作福音:它是为 beta 2 编写的。

根据 this answer 的线索我已经解决了:这取决于所选的项目架构。如果我将项目架构保留为默认的 (armv7, arm64),那么我会得到与您使用此代码时相同的错误:

 // Error with arm7 target:
 ringLayer.transform = CATransform3DRotate(ringLayer.transform, -M_PI/2, 0, 0, 1)

...并且需要转换为 Float(好吧,下面是 CGFloat,我敢肯定)以使其工作:

 // Works with explicit cast on arm7 target
 ringLayer.transform = CATransform3DRotate(ringLayer.transform, Float(-M_PI/2), 0, 0, 1)

但是,如果我将目标体系结构更改为仅 arm64,则代码​​将按照视频中 Apple 示例中所写的方式运行:

 // Works fine with arm64 target:
 ringLayer.transform = CATransform3DRotate(ringLayer.transform, -M_PI/2, 0, 0, 1)

所以回答你的问题,我相信这是因为 CGFloat 在 64 位架构上被定义为 double ,所以可以使用 M_PI(也是 double )派生的值作为 CGFloat 参数。但是,当 arm7 是目标时,CGFloat 是 float ,而不是 double ,因此当直接将 M_PI(仍然是 double )派生的表达式作为 CGFloat 参数传递时,您会失去精度。

请注意,默认情况下,Xcode 只会为调试构建的“事件”架构构建——我发现可以通过在 iPhone 4S 和 iPhone 5S 方案之间切换来切换此错误Xcode,因为它们有不同的架构。我猜想在演示视频中选择了 64 位架构目标,但在您的项目中选择了 32 位架构?

鉴于 CGFloat 在 64 位架构上是 double 的,处理这个特定问题的最简单方法是始终转换为 CGFloat。

但作为当您需要在不同架构上做不同事情时处理此类问题的演示,Swift 确实支持 conditional compilation :

    #if arch(x86_64) || arch(arm64)
        ringLayer.transform = CATransform3DRotate (ringLayer.transform, -M_PI / 2, 0, 0, 1)
    #else
        ringLayer.transform = CATransform3DRotate (ringLayer.transform, CGFloat(-M_PI / 2), 0, 0, 1)
    #endif

然而,这只是一个例子。你真的不想到处做这种事,所以我肯定会坚持简单地使用 CGFloat(<whatever POSIX double value you need>)根据目标架构获取 32 位或 64 位值。

Apple 在以后的编译器版本中添加了更多帮助来处理不同的 float ——例如,在早期的测试版中你甚至不能使用 floor()。单精度 float 很容易,而现在(当前 Xcode 6.1)有 floor() 的覆盖, ceil() , 等等,对于 float 和 double,所以你不需要摆弄条件编译。

关于swift - Swift 中需要奇怪的转换,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24109578/

相关文章:

ios - 如何将相机预览放入 UIImageView?

swift - 如何初始化图像按钮?

swift - 如何在 SpriteKit 中将一个 Action 应用于多个 Sprite?

ios - Swift中如何使用Visual Format Language设置约束?

swift - 如何在 uialertview swift 3 中添加 uipickerview

swift - 使用默认初始化程序与方便初始化程序进行初始化

ios - 核心数据崩溃: EXC_BAD_ACCESS while trying access data got using predicate

ios - 索引超出多个部分的范围

ios - 如何从 UINavigationBar 子类设置 barTintColor

ios - TableViewCell scroll 滚动时选择了错误的单元格