30

假设我有一个名为 SwiftKit 的框架,它有一个名为 someClassMethod 的 UIView 扩展类方法和一个名为 someProperty 的属性:

// SwiftKit
public extension UIView {
    class func someClassMethod() {
        print("someClassMethod from Swift Kit")
    }

    var someProperty: Double {
        print("someProperty from Swift Kit")
        return 0
    }
}

而且我还有一个叫 SwiftFoundation 的框架,它里面还有一个 UIView 的扩展类方法叫 someClassMethod 和一个叫 someProperty 的属性:

// SwiftFoundation
public extension UIView {
    class func someClassMethod() {
        print("someClassMethod from Swift Foundation")
    }

    var someProperty: Double {
        print("someProperty from Swift Foundation")
        return 0
    }
}

然后我创建了一个项目来介绍这些框架,事情是,如果我将它们都导入同一个 swift 文件并访问这些扩展,我得到一个“模糊使用 someProperty/someClassMethod()”错误,即使我指定了调用SwiftKit.UIView.someClassMethod() 的形式:

import UIKit
import SwiftKit
import SwiftFoundation

class ViewController: UIViewController {
    override func viewDidLoad() {
        super.viewDidLoad()
        // Do any additional setup after loading the view, typically from a nib.

        self.view.someProperty              // error: Ambiguous use of 'somProperty'
        SwiftKit.UIView.someClassMethod()   // error: Ambiguous use of 'someClassMethod()'
    }
}

如果我只导入其中一个,模棱两可的错误就会消失,但会发生更奇怪的事情:

import UIKit
import SwiftKit
//import SwiftFoundation

class ViewController: UIViewController {
    override func viewDidLoad() {
        super.viewDidLoad()
        // Do any additional setup after loading the view, typically from a nib.

        self.view.someProperty
        SwiftKit.UIView.someClassMethod()
    }
}

控制台打印出来:

来自 Swift 基金会的 someProperty

来自 Swift Foundation 的 someClassMethod

我的问题是:我怎样才能毫不含糊地调用这些扩展(类/实例方法、属性)?如果我不能,这是否意味着我们应该像通常使用 Objective-C 一样为扩展名添加前缀?

4

2 回答 2

7

细节

  • 斯威夫特 3,Xcode 8.1
  • 斯威夫特 4,Xcode 9.1
  • 斯威夫特 5.1,Xcode 11.2.1

问题

框架 SwiftFoundation 和 SwiftKit 具有相同名称的属性和函数

决定

方式1

使用不同的属性和函数名称

// SwiftFoundation
public extension UIView {
    public class func swiftFoundationSomeClassMethod() {
        print("someClassMethod from Swift Foundation")
    }

    public var swiftFoundationSomeProperty: Double {
        print("someProperty from Swift Foundation")
        return 0
    }
}

方式2

对属性和函数进行分组

// SwiftKit
public extension UIView {
    public class SwiftKit {
        public class func someClassMethod() {
            print("someClassMethod from Swift Kit")
        }

        public var someProperty: Double {
            print("someProperty from Swift Kit")
            return 0
        }
    }

    var SwiftKit: SwiftKit { return SwiftKit() }
}

结果

import UIKit
import SwiftKit
import SwiftFoundation

class ViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()

        _ = view.SwiftKit.someProperty
        UIView.SwiftKit.someClassMethod()

        _ = view.swiftFoundationSomeProperty
        UIView.swiftFoundationSomeClassMethod()
    }
}

项目

在此处输入图像描述 在此处输入图像描述

你的方法

SwiftFoundation.UIView.swiftFoundationSomeClassMethod()

您使用命名空间的变体不正确,因为这两个框架的所有 UIView 扩展都包含在您的 UIView 类中。查看下图,您可以在 SwiftFoundation 中看到 SwiftKit 类和 swiftFoundationSomeClassMethod()。这可能会使其他开发人员感到困惑。

在此处输入图像描述

于 2016-12-01T22:18:12.683 回答
1

如果它是 ObjCNSObject对象的扩展,例如UIView,那么是的,扩展方法需要前缀。

但是,那些觉得下划线难看的人可以尝试使用 Swift 协议来替换UIColor.red.my_toImage()这里的替代方法UIColor.red.my.toImage()Better way to manage swift extensions in your project

于 2019-06-10T18:01:35.210 回答