假设我有一个协议
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,因此我无法在协议中进行更多更改。
答案 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)
}
}