У меня есть массив, состоящий из AnyObject
. Я хочу перебрать его и найти все элементы, которые являются экземплярами массива.
Как проверить, является ли объект данного типа в Swift?
У меня есть массив, состоящий из AnyObject
. Я хочу перебрать его и найти все элементы, которые являются экземплярами массива.
Как проверить, является ли объект данного типа в Swift?
Если вы хотите проверить конкретный тип, вы можете сделать следующее:
if let stringArray = obj as? [String] {
// obj is a string array. Do something with stringArray
}
else {
// obj is not a string array
}
Вы можете использовать "как!" и это вызовет ошибку времени выполнения, если obj
не относится к типу [String]
let stringArray = obj as! [String]
Вы также можете проверять один элемент за раз:
let items : [Any] = ["Hello", "World"]
for obj in items {
if let str = obj as? String {
// obj is a String. Do something with str
}
else {
// obj is not a String
}
}
В Swift 2.2 - 5 теперь вы можете делать:
if object is String
{
}
Затем для фильтрации вашего массива:
let filteredArray = originalArray.filter({ $0 is Array })
Если у вас есть несколько типов для проверки:
switch object
{
case is String:
...
case is OtherClass:
...
default:
...
}
Если вы хотите узнать только, является ли объект подтипом данного типа, тогда существует более простой подход:
class Shape {}
class Circle : Shape {}
class Rectangle : Shape {}
func area (shape: Shape) -> Double {
if shape is Circle { ... }
else if shape is Rectangle { ... }
}
"Используйте оператор проверки типа (is), чтобы проверить, является ли экземпляр определенного тип подкласса. Оператор проверки типа возвращает true, если экземпляр имеет этот тип подкласса и false, если это не так ". Excerpt From: Apple Inc." Язык быстрого программирования ". iBooks.
В приведенном выше выражении важна фраза "определенного типа подкласса". Использование is Circle
и is Rectangle
принимается компилятором, потому что это значение shape
объявлено как shape
(суперкласс из Circle
и Rectangle
).
Если вы используете примитивные типы, суперкласс будет Any
. Вот пример:
21> func test (obj:Any) -> String {
22. if obj is Int { return "Int" }
23. else if obj is String { return "String" }
24. else { return "Any" }
25. }
...
30> test (1)
$R16: String = "Int"
31> test ("abc")
$R17: String = "String"
32> test (nil)
$R18: String = "Any"
У меня есть два способа сделать это:
if let thisShape = aShape as? Square
Или:
aShape.isKindOfClass(Square)
Вот подробный пример:
class Shape { }
class Square: Shape { }
class Circle: Shape { }
var aShape = Shape()
aShape = Square()
if let thisShape = aShape as? Square {
println("Its a square")
} else {
println("Its not a square")
}
if aShape.isKindOfClass(Square) {
println("Its a square")
} else {
println("Its not a square")
}
Изменить: 3 сейчас:
let myShape = Shape()
if myShape is Shape {
print("yes it is")
}
для swift4:
if obj is MyClass{
// then object type is MyClass Type
}
Предположим, что drawTriangle является экземпляром UIView. Чтобы проверить, имеет ли drawTriangle тип UITableView:
В Swift 3,
if drawTriangle is UITableView{
// in deed drawTriangle is UIView
// do something here...
} else{
// do something here...
}
Это также может использоваться для классов, определенных вами. Вы можете использовать это, чтобы проверить subviews представления.
Почему бы не использовать встроенные функции, созданные специально для этой задачи?
let myArray: [Any] = ["easy", "as", "that"]
let type = type(of: myArray)
Result: "Array<Any>"
Будьте предупреждены об этом:
var string = "Hello" as NSString
var obj1:AnyObject = string
var obj2:NSObject = string
print(obj1 is NSString)
print(obj2 is NSString)
print(obj1 is String)
print(obj2 is String)
Все четыре последние строки возвращают true, потому что, если вы введете
var r1:CGRect = CGRect()
print(r1 is String)
... конечно, он печатает "ложь", но предупреждение указывает, что сбрасывается Cast из CGRect в String. Таким образом, некоторый тип связан с мостом, а ключевое слово 'is' вызывает неявное преобразование.
Вам следует использовать один из следующих вариантов:
myObject.isKind(of: MyClass.self))
myObject.isMember(of: MyClass.self))
Если вы просто хотите проверить класс, не получая предупреждение из-за неиспользуемого определенного значения (пусть someVariable...), вы можете просто заменить материал let булевым:
if (yourObject as? ClassToCompareWith) != nil {
// do what you have to do
}
else {
// do something else
}
Xcode предложил это, когда я использовал путь let и не использовал определенное значение.
Почему бы не использовать что-то вроде этого
fileprivate enum types {
case typeString
case typeInt
case typeDouble
case typeUnknown
}
fileprivate func typeOfAny(variable: Any) -> types {
if variable is String {return types.typeString}
if variable is Int {return types.typeInt}
if variable is Double {return types.typeDouble}
return types.typeUnknown
}
в Swift 3.
myObject as? String
возвращает nil
, если myObject
не является String
. В противном случае он возвращает String?
, поэтому вы можете получить доступ к самой строке с помощью myObject!
или добавить ее с помощью myObject! as String
безопасно.
Swift 3:
class Shape {}
class Circle : Shape {}
class Rectangle : Shape {}
if aShape.isKind(of: Circle.self) {
}
Если у вас есть ответ вроде этого:
{
"registeration_method": "email",
"is_stucked": true,
"individual": {
"id": 24099,
"first_name": "ahmad",
"last_name": "zozoz",
"email": null,
"mobile_number": null,
"confirmed": false,
"avatar": "http://abc-abc-xyz.amazonaws.com/images/placeholder-profile.png",
"doctor_request_status": 0
},
"max_number_of_confirmation_trials": 4,
"max_number_of_invalid_confirmation_trials": 12
}
и вы хотите проверить значение is_stucked
, которое будет считаться AnyObject, все, что вам нужно сделать, это
if let isStucked = response["is_stucked"] as? Bool{
if isStucked{
print("is Stucked")
}
else{
print("Not Stucked")
}
}
Если вы не знаете, что в ответе от сервера вы получите массив словарей или отдельный словарь, вам нужно проверить, содержит ли результат массив или нет.
В моем случае всегда получаю массив словарей, кроме одного раза. Итак, чтобы справиться с этим, я использовал приведенный ниже код для Swift 3.
if let str = strDict["item"] as? Array<Any>
Вот как? Array проверяет, является ли полученное значение массивом (элементов словаря). В противном случае вы можете обработать, если это единственный элемент словаря, который не хранится внутри массива.
Swift 4.2, в моем случае использую функцию isKind.
isKind (of :) Возвращает логическое значение, которое указывает, является ли получатель экземпляром данного класса или экземпляром какого-либо класса, который наследуется от этого класса.
let items : [AnyObject] = ["A", "B" , ... ]
for obj in items {
if(obj.isKind(of: NSString.self)){
print("String")
}
}
Подробнее https://developer.apple.com/documentation/objectivec/nsobjectprotocol/1418511-iskind
Просто ради полноты на основе принятого ответа и некоторых других:
let items : [Any] = ["Hello", "World", 1]
for obj in items where obj is String {
// obj is a String. Do something with str
}
Но вы также можете (compactMap
также "отображает" значения, которых нет у filter
):
items.compactMap { $0 as? String }.forEach{ /* do something with $0 */ ) }
И версия, использующая switch
:
for obj in items {
switch (obj) {
case is Int:
// it an integer
case let stringObj as String:
// you can do something with stringObj which is a String
default:
print("\(type(of: obj))") // get the type
}
}
Но придерживаясь вопроса, чтобы проверить, является ли он массивом (т.е. [String]
):
let items : [Any] = ["Hello", "World", 1, ["Hello", "World", "of", "Arrays"]]
for obj in items {
if let stringArray = obj as? [String] {
print("\(stringArray)")
}
}
Или в более общем плане (см. Ответ на этот другой вопрос):
for obj in items {
if obj is [Any] {
print("is [Any]")
}
if obj is [AnyObject] {
print("is [AnyObject]")
}
if obj is NSArray {
print("is NSArray")
}
}