将默认值或 nil 值设置为函数的泛型类型参数

时间:2021-06-29 17:21:53

标签: ios swift generics protocols

假设我有一个协议

protocol TestProtocol {
}

另外,有一个结构体并从协议继承它。

struct StructOne: TestProtocol {
}

现在,我有一个视图控制器类并创建了一个通用函数来接受一组 TestProtocol 类型的对象(这是一个通用参数)。这是用于 SDK API 调用的传递参数。

但是在一些API调用中,我不需要传递这个参数数组。所以,我只想在函数定义中设置 nil 值或默认空数组。

这是课堂

class TestViewController: UIViewController {
//  func genericCall<T: TestProtocol>(param: [T] = []) { // Not work
//  func genericCall<T: TestProtocol>(param: [T]? = nil) { // Not work
  func genericCall<T: TestProtocol>(param: [T]?) {
    if param?.isEmpty == true {
      print("Empty Param Calling")
    } else {
      print("With Param Calling")
    }
  }
   
  override func viewDidLoad() {
    super.viewDidLoad()
    let param = [StructOne(), StructOne()]
    self.genericCall(param: param) // This one work
    self.genericCall(param: [] as [StructOne]) // This one also work. But want default value in function
    self.genericCall(param: nil) // Not work : Error - Generic parameter 'T' could not be inferred
//    self.genericCall() // Not work with default empty value : Error - Generic parameter 'T' could not be inferred
  }
}

我收到此编译时错误:无法推断通用参数“T”

我可以在函数调用期间设置一个空数组。这是提及here

我也检查了这个 link,如果只有 T 类型,它允许设置 nil 值,但这里是 T ([T]) 的数组。

有什么方法可以设置默认的 nil 值或任何其他方法来设置默认的空数组?这样我们就可以避免向每个函数调用传递一个空数组。

更新:

我不能这样使用它。由于 SDK 函数调用不允许我传递参数值。

func genericCall(param: [TestProtocol] = []) {
    // param: Not allowed me to pass to the sdk call function.
    if param.isEmpty == true {
        print("Empty Param Calling")
    } else {
        print("With Param Calling")
    }
}

注意:这是一个演示代码。实际上,我使用的是其中一个 SDK,因此我无法在协议中进行更多更改。

2 个答案:

答案 0 :(得分:3)

nil 在这种情况下过于宽泛,因为编译器无法推断它应该将 nil 应用于哪个 [T]?

您需要在此处明确指定 Optional 泛型参数:

self.genericCall(param: [StructOne]?.none)

答案 1 :(得分:1)

您可以创建一个泛型方法,将泛型类型限制为 RangeReplaceableCollection,将其 Element 限制为 TestProtocol。它会起作用的原因是 RangeReplaceableCollection 要求符合它的协议提供一个空的初始化程序:


protocol TestProtocol { }

struct StructOne: TestProtocol { }

class TestViewController: UIViewController {

    func genericCall<T: RangeReplaceableCollection>(param: T = .init()) -> T where T.Element:  TestProtocol {
        if param.isEmpty {
          print("Empty Param Calling")
        } else {
          print("With Param Calling")
        }
        return param
    }
    override func viewDidLoad() {
        super.viewDidLoad()
        let result: [StructOne] = genericCall() // This one work
        print(result)
    }
}

相关问题