- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我在使用 [NSMethodSignature getArgumentTypeAtIndex] 函数时遇到了非常奇怪的行为。它返回 BOOL 类型的“@”字符,根据 objective-c type encodings 错误.如果我使用 objc\runtime.h 库方法 method_getTypeEncoding BOOL 类型正确表示为“B”,但我不明白为什么它不能与更高级别的 NSMethodSignature 一起使用。以下代码演示了该问题:
-(void)viewDidAppear:(BOOL)animated
{
[super viewDidAppear:animated];
NSInvocation* inv = [NSInvocation invocationWithMethodSignature:[self methodSignatureForSelector:@selector(viewDidAppear:)]];
const char* encFromGetArgument = [[inv methodSignature] getArgumentTypeAtIndex:2];
const char* encFromMethodSignature = method_getTypeEncoding(class_getInstanceMethod([self class], @selector(viewDidAppear:)));;
const char* methodEncodingPure = [[[[NSString stringWithUTF8String:encFromMethodSignature] componentsSeparatedByCharactersInSet:[NSCharacterSet decimalDigitCharacterSet]] componentsJoinedByString:@""] UTF8String];//remove stack sizes
NSLog(@"BOOL arg from NSMethodSignature: %s", encFromGetArgument);
NSLog(@"BOOL arg from objc/runtime.h: %c", methodEncodingPure[3]);//first type is for return, second is target, third is selector
}
令人惊讶的是(至少对于男性而言)上面打印了以下内容:
BOOL arg from NSMethodSignature: @
BOOL arg from objc/runtime.h: B
我目前正在使用我自己的实现来避免这种奇怪的行为,但是我想知道我是否遗漏了什么或者它只是一个错误。我唯一的线索是 BOOL 是原始的,因此在调用 objective-c 方法时不能直接使用它。但是,当我尝试检查它时 [object isKingOfClass:[NSNumber class]] 返回 NO。
更新
好的,我已将 XCode 更新到最新版本 (6.1 6A1052d),情况大大改善。但是我现在的问题是区分 unsigned char 编码和真正的 bool 编码。我知道在旧版本中 BOOL 是作为 char 的 typedef,但是我如何才能完成真正的 char 与 BOOL 编码?我现在的结果是:
对于Simulator iPhone6和真机iPhone6我收到了:
argument 2: -------- -------- -------- --------
type encoding (B) 'B'
flags {}
BOOL arg from NSMethodSignature: B
BOOL arg from objc/runtime.h: B
这很棒,但是对于模拟器 iPhone4s 和真实设备 iPhone5 我得到:
argument 2: -------- -------- -------- --------
type encoding (c) 'c'
flags {isSigned}
BOOL arg from NSMethodSignature: c
BOOL arg from objc/runtime.h: c
我很确定,如果我检查 iPhone5s,它会得到与 iPhone6 相同的输出(因为我认为这都是关于 64 位架构的)。所以我现在的问题是如何正确解决旧设备,如何为它们区分 BOOL 的字符?或者我是否应该假设如果编码是“c”并且参数等于“1”我们有 YES,而对于“0”我们没有?
最佳答案
char *buf1 = @encode(BOOL);
NSLog(@"bool type is: %s", buf1);
在 32 位模拟器上,encode(BOOL) 返回“c”,而在 64 位模拟器上,它返回“B”。在 Build Settings 中,将 Architectures 更改为 $(ARCHS_STANDARD_32_BIT)
,然后它将为您返回“c”。
在 objc/objc.h 文件中。
#define OBJC_BOOL_DEFINED
/// Type to represent a boolean value.
#if !defined(OBJC_HIDE_64) && TARGET_OS_IPHONE && __LP64__
typedef bool BOOL;
#else
typedef signed char BOOL;
// BOOL is explicitly signed so @encode(BOOL) == "c" rather than "C"
// even if -funsigned-char is used.
#endif
BOOL 在 32 位中是 typeof signed char。您可以将它与 unsigned char 区分开来。
@encode(char) --> 'c'
@encode(unsigned char) --> 'C'
您可以从 here 判断设备是 32 位还是 64 位, 如果是 32 位, 'c' 对 BOOL 校验有效。
关于ios - NSMethodSignature 的 BOOL 编码错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26459754/
我最近开始为 IOS 开发并与调用对象作斗争。 我有一个类“Location”,派生自 NSManagedObject(它是 Coredata 模型的一部分) @interface Location
在 ObjC 中我有: NSMethodSignature *ms = [[object.target class] instanceMethodSignatureForSelector:object
我正在使用此方法将方法分派(dispatch)给委托(delegate),不幸的是,我发现大多数时候 NSMethodSignature 为 nil,这是因为选择器来自协议(protocol)。我想知
我需要一种方法来在运行时检查给定 block 的参数数量和参数类型(我目前正在编写的某些对象映射库需要这个,我正在将字符串格式的值映射到选择器,想要相同的 block )。 我尝试了以下示例中的代码,
我在使用 [NSMethodSignature getArgumentTypeAtIndex] 函数时遇到了非常奇怪的行为。它返回 BOOL 类型的“@”字符,根据 objective-c type
我正在尝试在协议(protocol)方法中获取 block 的签名。 这是一个示例协议(protocol): @protocol ProtocolSample - (void) doSomethin
我如何在 swift 2 中编写这个类? NSMethodSignature 和 NSInvocation 在 swift 2 中不再存在 @implementation MyAuth - (void
在苹果的 documentation这个方法的最后说了以下内容: 特殊注意事项 此方法自 Mac OS X v10.0 起可用,在 Mac OS X v10.5 中公开。仅支持应用程序运行所针对的运行
我在手机上运行以下代码,其中“对象”是猫,它是动物的子类。动物有一个属性“颜色”: NSLog(@"Object: %@", object); NSLog(@"Color: %@", [object
我是一名优秀的程序员,十分优秀!