arrays - Swift - Any - 无法将类型 MyType<Int> 的值转换为预期的元素类型 MyType<protocol<>>

标签 arrays swift generics compiler-errors any

我在下面列出了一个相当人为的例子,来 self 正在研究的更大的东西。我遇到了我认为是关于使用 Any 类型的 Swift 编译器错误。

struct Labelable <T> {
  let t: T
  let label: String
}

func allLabels(labelables: [Labelable<Any>]) -> [String] {
  return labelables.map { $0.label }
}

let labeledInt = Labelable(t: 22, label: "Steve's Age")
let labeledString = Labelable(t: "Johnson", label: "Stacy's last name")

let labels = allLabels([labeledInt, labeledString])

print(labels)

问题错误:

Cannot convert value of type 'Labelable<Int>' to expected element type 'Labelable<protocol<>>'

有没有人对此问题有任何见解和解决方法?或者,也许有更好的方法来完成我想做的事情?

谢谢

编辑:

这就是我最终所做的。啊。

protocol LabelableProtocol {
  var label: String { get }
}

struct Labelable <T>: LabelableProtocol {
  let t: T
  let label: String
}

func allLabels(labelables: [LabelableProtocol]) -> [String] {
  return labelables.map { $0.label }
}

let labeledInt = Labelable(t: 22, label: "Steve's Age")
let labeledString = Labelable(t: "Johnson", label: "Stacy's last name")

let labels = allLabels([labeledInt, labeledString])

print(labels)

最佳答案

这里的问题是 Any 不是类型,而是协议(protocol)。您可以通过尝试通过另一个协议(protocol)扩展它来看到这一点(不允许通过其他协议(protocol)扩展协议(protocol)):

protocol MyDummyProtocol {}

extension Int : MyDummyProtocol {} // OK

extension Any : MyDummyProtocol {} 
    //Error: Non-nominal type Any (aka protocol<>) cannot be extended.

因此,当调用 allLabels 时,您不能将 IntString 类型转换为 Any 协议(protocol)。

但是,您可以在结构中专门使用 Any 作为通用 T:

let labelAny = Labelable<Any>(t: 22, label: "Steve's Age")
let labelAnyAgain = Labelable<Any>(t: "Johnson", label: "Stacy's last name")

let labels = allLabels([labelAny, labelAnyAgain]) // OK

然而,这里更有趣的是研究结构中的通用属性 t。现在应该是 Any 了吧? Any 的好处是我们可以将它向下转换为基本的 swift 类型。例如,考虑:

struct Labelable <T> {
    let t: T
    let label: String
}

func allGenerics(labelables: [Labelable<Any>]) -> [Any] {
    return labelables.map { $0.t }
}

let labelAny = Labelable<Any>(t: 22, label: "Steve's Age")
let labelAnyAgain = Labelable<Any>(t: "Johnson", label: "Stacy's last name")

let labelsGeneric = allGenerics([labelAny, labelAnyAgain])

for label in labelsGeneric {
    switch label {
    case is Int: print("is int: " + "\(label)")
    case is String: print("is string: " + "\(label)")
    default: print("Default...")
    }
}
// Prints:
// is int: 22
// is string: Johnson

在这里,我们使用 Any 作为泛型(即使它具体是一个 Any),然后简单地让向下转换 is成为我们的通用行为。

参见 Language Guide - Type Casting - Type Casting for Any and AnyObject了解详情。

Any can represent an instance of any type at all, including function types.

..

Use Any and AnyObject only when you explicitly need the behavior and capabilities they provide. It is always better to be specific about the types you expect to work with in your code.

关于arrays - Swift - Any - 无法将类型 MyType<Int> 的值转换为预期的元素类型 MyType<protocol<>>,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34444546/

相关文章:

objective-c - 如何在 Objective-C 中对 NSArray 进行切片?

javascript - D3 Stacked Bar Chart 中引用 JS 数组

ios - 第二次启动应用程序只显示黑色 View

java - 反射(reflect)具有泛型类型参数的方法时出现 NoSuchMethodException

java - 什么时候实现 Comparable<super class of X> 而不是 Comparable<X>?

Java:在泛型类中使用类型化变量

java - 我在调用 Java 中的静态递归方法时遇到问题

php - 在php/mysql中插入数组数据

ios - 在 iOS 中分析文本到语音的字符串

ios - 如何在 Swift5 中关闭应用程序时更改应用程序状态?