Как создать протокол во время выполнения в Objective-C? - PullRequest
3 голосов
/ 11 апреля 2010

Прежде всего, я хочу прояснить, что я не говорю об определении протокола, и что я понимаю концепцию

@protocol someprotocol
- (void)method;
@end

Я знаю, что среда выполнения Obj-C позволяет создавать классы в RUNTIME, а также его ivars и методы. Также доступны для создания SEL-ы. Я думаю, что просто что-то упустил, но кто-нибудь знает, какую функцию вызывать для создания протокола во время выполнения? Основная причина этого заключается в том, что conformsToProtocol: работает, поэтому простое добавление соответствующих методов на самом деле не сокращает его.

Ответы [ 2 ]

8 голосов
/ 14 декабря 2012

Способ добавления протокола во время выполнения

Protocol *aProtocol = objc_allocateProtocol("TestingRuntimeDelegate");
AppDelegate *appInstance = (AppDelegate*)[[UIApplication sharedApplication] delegate];
NSLog(@"conformed Protocol ..%d",class_conformsToProtocol([self.delegate class], aProtocol));

protocol_addMethodDescription(aProtocol, @selector(itIsTestDelegate), "test", NO, NO);
objc_registerProtocol(aProtocol);
class_addProtocol([appInstance class], aProtocol);
//NSLog(@"adding Protocol %d",class_addProtocol([appInstance class], aProtocol));

if ([self.delegate conformsToProtocol:@protocol(TestDelegate)])
{
    NSLog(@"conformed Protocol ..");
}
else
{
    NSLog(@"conformed Protocol ..%d",class_conformsToProtocol([appInstance class], aProtocol));
    class_conformsToProtocol([self.delegate class], aProtocol);
    [appInstance performSelector:@selector(itIsTestDelegate)];
}

Но метод делегата должен быть предварительно определен в этом классе, иначе он потерпит крах из-за нераспознанного селектора

или метод может быть добавлен во время выполнения. Тогда все будет работать нормально ..

Но следующий файл должен быть импортирован первым.

#include <objc/runtime.h>
1 голос
/ 11 апреля 2010

следующий вид работ, но правильный способ сделать это будет высоко ценится:

Protocol *proto = [Protocol alloc];
object_setInstanceVariable(proto, "protocol_name", &"mySuperDuperProtocol");
void *nada = NULL;
object_setInstanceVariable(proto, "protocol_list", &nada);
object_setInstanceVariable(proto, "class_methods", &nada);

struct objc_method_description_list *methods;
methods = malloc(sizeof(int) + sizeof(struct objc_method_description) * 1);
methods->count = 1;
methods->list[0].name = @selector(foobar:);
methods->list[0].types = "v:@";
object_setInstanceVariable(proto, "instance_methods", &methods);

class_addProtocol([self class], proto);
NSLog(@"%d %s", [self conformsToProtocol:proto], protocol_getName(objc_getProtocol("mySuperDuperProtocol")));

Первый параметр NSLog равен 1, а не когда строка, добавляющая протокол к классу, закомментирована, то есть протокол каким-то образом зарегистрирован в классе. С другой стороны, он не возвращается objc_getProtocol (вторая записанная запись - nil).

...