- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
更具体地说,imageNamed 不会暴露给 NSImage 的 Swift 子类,即使继承了所有其他便利 init。
根据 Apple 的文档,Objective-C 工厂方法“被映射为 Swift 中的便利初始化程序。”
因此是 NSImage 工厂方法
+ (NSImage *)imageNamed:(NSString *)名称
被映射到 Swift 的
NSImage(命名为:“任何东西”)
另外,在Swift Programming Language一书中我们看到子类初始化继承的规则是:
“规则 1如果您的子类没有定义任何指定的初始化器,它会自动继承其父类(super class)的所有指定初始化器。
规则 2如果您的子类提供了其所有父类(super class)指定初始化器的实现——通过按照规则 1 继承它们,或者通过提供自定义实现作为其定义的一部分——那么它会自动继承所有父类(super class)便利初始化器。”
因此子类
T 类:NSImage {
}
将继承所有父类(super class)的初始化器,包括指定的和方便的,但以下将不起作用
让我= T(命名为:“任何东西”)
我错过了什么?
最佳答案
原因似乎是 named:
初始化器的返回类型为 NSImage
class NSImage : NSObject, NSCopying, NSCoding, NSSecureCoding, NSPasteboardReading, NSObjectProtocol, NSPasteboardWriting {
/*All instance variables are private*/
init?(named name: String) -> NSImage /* If this finds & creates the image, only name is saved when archived */
而其他(不是所有)初始化器没有返回类型
init(size aSize: NSSize)
init?(data: NSData) /* When archived, saves contents */
init?(contentsOfFile fileName: String) /* When archived, saves contents */
init?(contentsOfURL url: NSURL) /* When archived, saves contents */
init?(byReferencingFile fileName: String) /* When archived, saves fileName */
init(byReferencingURL url: NSURL) /* When archived, saves url, supports progressive loading */
现在正因为如此,如果你有一个 T
类,它是 NSImage
的子类,你会期望 T(named:)
到返回 T
的实例,但如果您不提供自己的实现,将调用父类(super class)实现,正如我们在声明中看到的那样,它返回一个 NSImage
。
为什么它与所有其他初始化程序不同?我不知道。
也许 NSImage
使用的底层实现和缓存机制导致了这种不一致。但更重要的是 +[NSImage imageNamed:]
不会调用任何 NSImage
指定的初始化程序。
看看我用 Cocoa Touch 的 UIImage
制作的这个例子:https://gist.github.com/bartekchlebek/d61154add8525218ae3a
你可以在那里看到,我创建了一个名为 MyImage
的 UIImage
子类,我覆盖了所有 UIImage
的初始化程序,将 NSLog
并返回 nil。
在
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
我调用 -[MyImage imageNamed:]
并注意到,放置在指定初始化程序中的 NSLog
都没有被打印。我还打印了用 -[MyImage imageNamed:]
创建的实例的类,你可以看到它不是 MyImage
而是 UIImage
(把除了我在所有初始化器中返回 nil
的事实之外 ;) )
总之,+[UIImage imageNamed:]
不会调用任何 UIImage
的指定初始化器,因此覆盖它们不会使 +[MyImage imageNamed:]
返回 MyImage
的实例。我希望 NSImage
的行为方式相同。
这种微妙之处导致 Swift
不继承 named:
初始化器,因为 swift 要求 便利初始化器通过指定的初始化器,而 Objective -C 没有要求。
关于objective-c - Obj-C 工厂方法未暴露给 Swift 子类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26551100/
我正在尝试创建键值对并推送到数组,但我得到的只是 [Obj Obj]、[Obj Obj]。我创建了一个传递名称和值的函数,用于将键和值分配给 JavaScript 对象。这是我的代码。不确定我错过了什
似乎在 for 循环的上下文中,关于对象的语法发生了变化。 为什么 console.log() 不应该运行相同的东西?第一个按预期运行,第二个呈现错误“steve is not defined”: v
在 Ady Osmani 的 blog post关于 js 中的命名空间,他提到了 5 个常见做法来测试先前定义的命名空间/对象是否存在以防止覆盖。我在这里复制我关注的 3 个: var myAppl
有没有办法(我怀疑它涉及继承和多态)来区分OBJ o, OBJ& o, const OBJ& o?我希望在 3 个不同的程序中使用相同的代码,并使用相同的方法名称调用不同的方法。 int main()
我正在寻找一个正则表达式来分割这种内容: obj.method(obj.attr,obj.attr2) 我希望拆分返回一个数组: ["obj", "method(obj.attr, obj.attr2
我想知道这些方法中哪种更好: var Obj = Obj || {}; 或 if (Obj === undefined || typeof Obj !== 'object') { Obj = {}
我正在尝试将一个值推送到数组的属性,如下所示 var obj = {}; obj.a = (obj.a || []).push(10); console.log( typeof obj.a );
为了避免误解,我们首先要就某些词语的含义达成一致。以下含义并非普遍接受的含义,我仅建议将它们作为此问题的背景。 function -- Function 的一个实例。它有一个与其关联的过程。 obje
我总是不确定哪个是正确的以及该使用哪个。 通常我会进行(obj == null)检查。我认为最好直接问。 我应该使用以下哪一项: if (obj == null) { alert(
我正在处理一些使用 pygraph 模块的类,当我使用 add_node() 方法时,它总是出现“node xxx already in graph”。所以我尝试使用 deepcopy() 创建一个新
在 this page您可以看到以下示例,了解如何实现数组的indexOf: if (!Array.prototype.indexOf) { Array.prototype.indexOf = f
(1) 和 (2) 之间是否存在任何重要差异(语义、性能相关等)? var obj = obj || {}; var obj = Object(obj); 上下文。第一个是我在 Django 的模板和
我想知道 obj !== obj 什么时候可以为真? 这是我在书上看到的一行代码,我很纳闷。 var result = class2type[(obj == null || obj !== obj)]
我有时会看到这种模式...... obj.method.call(obj, arg) 我不明白为什么它不同于... obj.method(arg) 为什么要使用第一种模式? 我的天啊,似乎引起了很
我刚刚在一段 React 代码中发现了以下结构(名称已更改): 据我了解,bind 只是执行相应的函数,并将函数的 this 设置为第一个参数,并向其传递更多参数。由于 func 已经是我们想要的
当我们查看Underscore.js源码时,我们可以看到如下内容: _.isObject = function (obj) { return obj === Object(obj);
我在将项目发布到本地系统时收到此错误 Copying file obj\Debug\build.force to obj\Release\Package\PackageTmp\obj\Debug\bu
我有一个类型为 Expression> 的现有表达式;它包含类似 cust => cust.Name 的值. 我还有一个父类,其字段类型为 T .我需要一个接受上述作为参数并生成一个以父类 ( TMo
我在当前目录中有 add.c sub.c 并通过 makefile 编译它们。 我做了以下事情: program 1: objs=$(patsubst %.cpp, %.o, $(wildcard *
这个问题在这里已经有了答案: Is there a difference between copy initialization and direct initialization? (9 个回答)
我是一名优秀的程序员,十分优秀!