Есть ли способ зарегистрировать все значения свойств экземпляра Objective-C

Мне просто интересно, есть ли быстрый и простой способ распечатать в журнале все различные значения свойств для моего класса для целей отладки. Например, я хотел бы знать, какие значения всех BOOL, плавающих и т.д. Есть.

Ответы

Ответ 1

Этот вопрос кажется ответом на ваш вопрос.

Обновление:

Мне стало любопытно, и я стал каталогом:

//Using Xcode 4.5.2 - iOS 6 - LLDB - Automatic Reference Counting

//NSObject+logProperties.h    
@interface NSObject (logProperties)
- (void) logProperties;
@end

//NSObject+logProperties.m
#import "NSObject+logProperties.h"
#import <objc/runtime.h>

@implementation NSObject (logProperties)

- (void) logProperties {

    NSLog(@"----------------------------------------------- Properties for object %@", self);

    @autoreleasepool {
        unsigned int numberOfProperties = 0;
        objc_property_t *propertyArray = class_copyPropertyList([self class], &numberOfProperties);
        for (NSUInteger i = 0; i < numberOfProperties; i++) {
            objc_property_t property = propertyArray[i];
            NSString *name = [[NSString alloc] initWithUTF8String:property_getName(property)];
            NSLog(@"Property %@ Value: %@", name, [self valueForKey:name]);
        }
        free(propertyArray);
    }    
    NSLog(@"-----------------------------------------------");
}

@end

Включите его в свой класс: #import "NSObject+logProperties.h"

и вызовите [self logProperties]; для этих свойств!

Ответ 2

Текущие ответы просто показывают, как это сделать для свойств. Если вы хотите, чтобы каждая переменная экземпляра была распечатана, вы могли бы сделать что-то вроде ниже.

- (void)logAllProperties {
    unsigned int count;
    Ivar *ivars = class_copyIvarList([self class], &count);
    for (unsigned int i = 0; i < count; i++) {
        Ivar ivar = ivars[i];

        const char *name = ivar_getName(ivar);
        const char *type = ivar_getTypeEncoding(ivar);
        ptrdiff_t offset = ivar_getOffset(ivar);

        if (strncmp(type, "i", 1) == 0) {
            int intValue = *(int*)((uintptr_t)self + offset);
            NSLog(@"%s = %i", name, intValue);
        } else if (strncmp(type, "f", 1) == 0) {
            float floatValue = *(float*)((uintptr_t)self + offset);
            NSLog(@"%s = %f", name, floatValue);
        } else if (strncmp(type, "@", 1) == 0) {
            id value = object_getIvar(self, ivar);
            NSLog(@"%s = %@", name, value);
        }
        // And the rest for other type encodings
    }
    free(ivars);
}

Хотя я бы не стал предлагать делать это на практике, но если это для целей отладки, то это прекрасно. Вы можете реализовать это как категорию на NSObject и держать ее лежащей для использования при отладке. Если он завершен для всех типов кодировок, он может сделать очень хороший небольшой метод.

Ответ 3

да, одним из способов было бы запросить все свойства, а затем использовать KVC, например:

//properties
unsigned int cProperties = 0;
objc_property_t *props = class_copyPropertyList(self.class, &cProperties);
for(int i = 0; i < cProperties; i++) {
    const char *name = property_getName(props[i]);
    NSLog(@"%@=%@", name, [self valueForKey:name];
}

альтернативный способ состоит в том, чтобы пройти все методы класса, получить возвращаемый тип, вызвать и распечатать его

Ответ 4

Быстро и грязно было бы переопределить debugDescription:

-(NSString*)debugDescription {
    NSString *str = [[NSString alloc] initWithFormat:@"My BOOL 1: %d, My Float: %f", self.myBool, self.myFoat];
    return str;
}

Конечно, если ваш объект сложный, это может занять много времени.