Tengo una función que hace una matriz plana de una matriz anidada. Funciona perfectamente para int array.
let array: [Any] = [1, 2, [3]]
func makeFlatArray(_ array: [Any]) -> [Int] {
var flatArray = [Int]()
for item in array {
if let item = item as? Int {
flatArray.append(item)
} else if let item = item as? [Any] {
let result = makeFlatArray(item)
flatArray += result
}
}
return flatArray
}
Pero quiero hacerlo genérico y tengo error al intentarlo.
func makeFlatArrayGeneric<T>(_ array: [Any]) -> [T] {
var flatArray = [T]()
for item in array {
if let item = item as? T {
flatArray.append(item)
} else if let item = item as? [Any] {
let result = makeFlatArrayGeneric(item)
flatArray += result
}
}
return flatArray
}
Error:
Playground execution failed:
error: FlatNestedArraysExample.playground:22:26: error: generic parameter "T" could not be inferred
let result = makeFlatArrayGeneric(item)
^
FlatNestedArraysExample.playground:16:6: note: in call to function "makeFlatArrayGeneric"
func makeFlatArrayGeneric<T>(_ array: [Any]) -> [T] {
Respuestas
6 para la respuesta № 1Tienes que decirle al compilador el tipo de T.
func makeFlatArrayGeneric<T>(_ array: [Any]) -> [T] {
var flatArray = [T]()
for item in array {
if let item = item as? T {
flatArray.append(item)
} else if let item = item as? [Any] {
let result: [T] = makeFlatArrayGeneric(item)
flatArray += result
}
}
return flatArray
}
let array: [Any] = [1, 2, [3], [4, [5]]]
let items: [Int] = makeFlatArrayGeneric(array) //[1, 2, 3, 4, 5]
Un enfoque más funcional para resolver tu problema.
extension Array {
func flatten() -> Array<Element> {
return self.flatMap({ element -> [Element] in
if let array = element as? Array {
return array.flatten()
} else {
return [element]
}
})
}
}
let items: [Int] = array.flatten() //[1, 2, 3, 4, 5]
0 para la respuesta № 2
Tienes que hacer algo para darte cuenta de lo que es T
Una forma de hacerlo es asignar el valor de retorno a una variable cuyo tipo es conocido:
let result: [T] = makeFlatArrayGeneric(item)
Otra forma es pasar el tipo como parámetro:
func makeFlatArrayGeneric<T>(_ array: [Any], type: T.Type) -> [T] {
var flatArray = [T]()
for item in array {
if let item = item as? T {
flatArray.append(item)
} else if let item = item as? [Any] {
let result = makeFlatArrayGeneric(item, type: type)
flatArray += result
}
}
return flatArray
}
Y llámalo así:
makeFlatArrayGeneric(yourArray, type: Int.self)