Sau*_*pal 1 objective-c nsstring ios
我只是编写以下代码用于测试目的:
NSString *aStr = [[NSString alloc] initWithFormat:@"Foo"];
aStr = [aStr initWithFormat:@"Bar"];//Crashed here
Run Code Online (Sandbox Code Playgroud)
我收到以下错误:
*** initialization method -initWithFormat:locale:arguments: cannot be sent to an abstract object of class __NSCFString: Create a concrete instance!
Run Code Online (Sandbox Code Playgroud)
如果我写下面的代码同样的事情发生
NSString *aStr = [NSString alloc];
aStr = [aStr initWithFormat:@"Foo"];
aStr = [aStr initWithFormat:@"Bar"]; //Crashed here
Run Code Online (Sandbox Code Playgroud)
通过谷歌,我知道initWithFormat将返回NSCFString对象.我的问题是,如果NSCFString是派生类,NSString那么为什么我不能调用该initWithFormat方法NSCFString.如果可以停止可见性,我该如何在代码中实现.
让我们对NSString类集群如何在内部进行调查:
NSString *factory = [NSString alloc];
NSString *theInstance = [factory initWithString:@"I am constant"];
NSLog(@"factory class: %@, instance class: %@", [factory class], [theInstance class]);
Run Code Online (Sandbox Code Playgroud)
输出是:
factory class: NSPlaceholderString, instance class: __NSCFConstantString
Run Code Online (Sandbox Code Playgroud)
如您所见,该alloc方法返回一个实例NSPlaceholderString.它是一个"工厂"类,它实现了init...声明的所有方法NSString.这些方法返回的具体(私有)子类NSString.它__NSCFConstantString在此示例中返回.
如果您将第一行更改为
NSString *factory = [NSMutableString alloc];
Run Code Online (Sandbox Code Playgroud)
输出将更改为:
NSPlaceholderMutableString,实例类:__ NSCFString
因此,可变和不可变字符串有不同的工厂类,并且这些工厂返回不同的子类.
您甚至可以在iOS运行时标题中检查私有子类的层次结构:此处和此处.
现在让我们看看当我们调用刚刚创建initWithString:的实例时会发生什么__NSCFConstantString.
[theInstance initWithString:@"Crash"];
Run Code Online (Sandbox Code Playgroud)
如你所料 - 它崩溃了.在stacktrace中我们可以看到-[NSString initWithCharactersNoCopy:length:freeWhenDone:]调用该方法,抛出异常:
'NSInvalidArgumentException',原因:'***初始化方法-initWithCharactersNoCopy:length:freeWhenDone:无法发送到类__NSCFConstantString的抽象对象:创建一个具体的实例!'
所以我们可以猜测NSString类中的这个初始化器实际上是一个抽象方法(有点 - 在Objective-C中没有抽象方法,因此它在调用时抛出异常).
此方法在工厂类中实现NSPlaceholderString.但它并没有在所有具体的子类中实现,所以如果你调用任何init...方法,它将调用NSString抛出异常的实现.
让我们把它们放在一起,构建NSString类集群的一小部分.它真的很简单,可能与真正的实现完全不同,但我只想展示这个想法.
@interface NSPlaceholderString : NSString
@end
@interface __NSCFConstantString : NSString
@end
@implementation NSString
+ (instancetype)alloc {
return [[NSPlaceholderString alloc] init];
}
- (instancetype)initWithCharactersNoCopy:(unichar *)characters length:(NSUInteger)length freeWhenDone:(BOOL)freeBuffer {
[NSException raise:NSInvalidArgumentException format:@" initialization method -initWithCharactersNoCopy:length:freeWhenDone: cannot be sent to an abstract object of class %@: Create a concrete instance!'", [self class]];
return nil;
}
- (instancetype)initWithString:(NSString *)aString {
//this method has to call the "abstract" initializer somewhere. The real implementation is probably more complex, this single line is here for simplicity
return [self initWithCharactersNoCopy:[aString UTF8String] length:[aString length] freeWhenDone:YES];
}
@end
@implementation NSPlaceholderString
- (instancetype)initWithCharactersNoCopy:(unichar *)characters length:(NSUInteger)length freeWhenDone:(BOOL)freeBuffer {
__NSCFConstantString *concreteClassInstance = ...; // create the concrete instance.
return concreteClassInstance;
}
@end
@implementation __NSCFConstantString
//implement all the needed methods here. But do NOT implement initWithCharactersNoCopy:length:freeWhenDone:
@end
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
535 次 |
| 最近记录: |