我正在实现一个名为ofType的函数,它过滤掉给定类型的所有元素.

这是我的代码:

class Animal {}
class Mammal: Animal {}
class Monkey: Mammal {}
class Pig: Mammal {}
class Human: Mammal {}

extension Array {
    func ofType<T>(_ Metatype: T.Type) -> [T] {
        return flatMap { type(of: $0) == Metatype ? $0 as? T : nil }
//      return flatMap { $0 as? T } // This is not working as the T is always the static type of the parameter,which is Animal in this example.
//      return flatMap { $0 as? Metatype } // This is not working either because of the grammar restriction.
    }
}

let animals = [Monkey(),Pig(),Human(),Mammal(),Animal()]
func animalType() -> Animal.Type {
    return Mammal.self
}
animals.ofType(animalType()).count // returns 1,expect to be 4.

在Objc中,我可以使用isKindOf()来检查对象是类的实例还是子类.在swift和as中有类似的操作,但是它们之后的类型应该是静态类型,而不是动态类型值(例如我可以写的是Mammal,但不是Mammal.self).

我不能使用类型参数T,因为在这个例子中,T等于Animal,这不是我想要的.

你对如何实现这个功能有任何想法吗?

就个人而言,我认为 @JeremyP’s suggestion使用Mirror是最好的;虽然我会做一些调整:
/// Conditionally cast `x` to a given dynamic Metatype value,taking into consideration
/// class inheritance hierarchies.
func conditionallyCast<T,U>(_ x: T,to destType: U.Type) -> U? {

  if type(of: x) is AnyClass && destType is AnyClass { // class-to-class

    let isCastable = sequence(
      first: Mirror(reflecting: x),next: { $0.superclassMirror }
    )
    .contains { $0.subjectType == destType }

    return isCastable ? (x as! U) : nil
  }

  // otherwise fall back to as?
  return x as? U
}

在这里,我们使用sequence(first:next:)从动态类型的x创建一个元类型序列,通过它可能具有的任何超类元类型(可能是第一次使用我看过的看起来不太糟糕的函数:P).另外,我们又回到了做什么?当我们知道我们没有进行类到类的转换时,它会使该函数也能使用协议元类型.

然后你可以简单地说:

extension Sequence {
  func ofType<T>(_ Metatype: T.Type) -> [T] {
    return flatMap { conditionallyCast($0,to: Metatype) }
  }
}

protocol P {}
class Animal {}
class Mammal: Animal {}
class Monkey: Mammal,P {}
class Pig: Mammal {}
class Human: Mammal,P {}

let animals = [Monkey(),Animal()]

let animalType: Animal.Type = Mammal.self
print(animals.ofType(animalType)) // [Monkey,Pig,Human,Mammal]

print(animals.ofType(P.self)) // [Monkey,Human]

假设您在Apple平台上(即可以访问Objective-C运行时),另一个选择是使用Objective-C元类方法isSubclass(of:)来检查给定的元类型是否相等,或者是否为子类另一个:

import Foundation

/// Conditionally cast `x` to a given dynamic Metatype value,to destType: U.Type) -> U? {

  let sourceType = type(of: x)

  if let sourceType = sourceType as? AnyClass,let destType = destType as? AnyClass { // class-to-class

    return sourceType.isSubclass(of: destType) ? (x as! U) : nil
  }

  // otherwise fall back to as?
  return x as? U
}

这是有效的,因为在Apple平台上,Swift类构建在Obj-C类之上 – 因此Swift类的元类型是Obj-C元类对象.

如何在swift中检查一个对象是否是一种动态类类型?的更多相关文章

  1. 可选链

    是不是很苦恼呀,因此可选链就应运而生了将上面的访问链中的的强制解析的感叹号(!可选链可以处理各类之间的属性、方法、下标等的,在这里我只举了属性,其他的就不一一举例说明了,作为一名优秀的程序员还是要慢慢锻炼有触类旁通的能力。

  2. 如何在swift中检查一个对象是否是一种动态类类型?

    我正在实现一个名为ofType的函数,它过滤掉给定类型的所有元素.这是我的代码:在Objc中,我可以使用isKindOf()来检查对象是类的实例还是子类.在swift和as中有类似的操作,但是它们之后的类型应该是静态类型,而不是动态类型值.我不能使用类型参数T,因为在这个例子中,T等于Animal,这不是我想要的.你对如何实现这个功能有任何想法吗?

  3. ios – Swift中的isMemberOfClass

    关键字is等同于isKindOfClass.但我无法在swift中找到与isMemberOfClass相同的东西.注意:我的问题不是关于isKindOfClass或isMemberofclass之间的区别,而是问题是关于什么是Swift中isMemberofClass的等价物有人请澄清一下解决方法您正在寻找类型(:).例:

  4. 在C#中,如何创建具有不同类型对象的IEnumerable类&gt;

    在C#中,如何使用不同类型的对象创建IEnumerable类例如:我想做一些事情:解决方法另一种方法,如果你想要一个命名集合:执行:在这里,你延伸了Collection的基础,它实现了IEnumerable.

  5. string – 使用JRE库替换StrSubstitutor

    目前我正在使用org.apache.commons.lang.text.StrSubstitutor来做:鉴于我想从我的项目中删除commons-lang依赖项,使用标准JRE库的StrSubstitutor的工作和简约实现是什么?注意:StrSubstitutor的工作方式如下:屈服于resolveString=“快速的棕色狐狸跳过懒狗.”解决方法如果性能不是优先级,则可以使用appendRep

  6. Java对象赋值是什么意思?

    我有以下两个班级:当我运行Cat时,我得到了以下结果:我能理解1,2,3和5,但为什么#4不是:“Cat:static–4”?

随机推荐

  1. Swift UITextField,UITextView,UISegmentedControl,UISwitch

    下面我们通过一个demo来简单的实现下这些控件的功能.首先,我们拖将这几个控件拖到storyboard,并关联上相应的属性和动作.如图:关联上属性和动作后,看看实现的代码:

  2. swift UISlider,UIStepper

    我们用两个label来显示slider和stepper的值.再用张图片来显示改变stepper值的效果.首先,这三个控件需要全局变量声明如下然后,我们对所有的控件做个简单的布局:最后,当slider的值改变时,我们用一个label来显示值的变化,同样,用另一个label来显示stepper值的变化,并改变图片的大小:实现效果如下:

  3. preferredFontForTextStyle字体设置之更改

    即:

  4. Swift没有异常处理,遇到功能性错误怎么办?

    本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  5. 字典实战和UIKit初探

    ios中数组和字典的应用Applicationschedule类别子项类别名称优先级数据包contactsentertainment接触UIKit学习用Swift调用CocoaTouchimportUIKitletcolors=[]varbackView=UIView(frame:CGRectMake(0.0,0.0,320.0,CGFloat(colors.count*50)))backView

  6. swift语言IOS8开发战记21 Core Data2

    上一话中我们简单地介绍了一些coredata的基本知识,这一话我们通过编程来实现coredata的使用。还记得我们在coredata中定义的那个Model么,上面这段代码会加载这个Model。定义完方法之后,我们对coredata的准备都已经完成了。最后强调一点,coredata并不是数据库,它只是一个框架,协助我们进行数据库操作,它并不关心我们把数据存到哪里。

  7. swift语言IOS8开发战记22 Core Data3

    上一话我们定义了与coredata有关的变量和方法,做足了准备工作,这一话我们来试试能不能成功。首先打开上一话中生成的Info类,在其中引用头文件的地方添加一个@objc,不然后面会报错,我也不知道为什么。

  8. swift实战小程序1天气预报

    在有一定swift基础的情况下,让我们来做一些小程序练练手,今天来试试做一个简单地天气预报。然后在btnpressed方法中依旧增加loadWeather方法.在loadWeather方法中加上信息的显示语句:运行一下看看效果,如图:虽然显示出来了,但是我们的text是可编辑状态的,在storyboard中勾选Editable,再次运行:大功告成,而且现在每次单击按钮,就会重新请求天气情况,大家也来试试吧。

  9. 【iOS学习01】swift ? and !  的学习

    如果不初始化就会报错。

  10. swift语言IOS8开发战记23 Core Data4

    接着我们需要把我们的Rest类变成一个被coredata管理的类,点开Rest类,作如下修改:关键字@NSManaged的作用是与实体中对应的属性通信,BinaryData对应的类型是NSData,CoreData没有布尔属性,只能用0和1来区分。进行如下操作,输入类名:建立好之后因为我们之前写的代码有些地方并不适用于coredata,所以编译器会报错,现在来一一解决。

返回
顶部