我需要使用
swift在iOS中创建代理模式
我已经尝试使用Objective C,这是代码
MyProtocol.h
#import <Foundation/Foundation.h>
@protocol MyProtocol <NSObject>
@required
-(void)testMessage;
@end
TestBO.h
#import <Foundation/Foundation.h>
#import "MyProtocol.h"
@interface TestBO : NSObject <MyProtocol>
@end
TestBO.m
#import "TestBO.h"
@implementation TestBO
-(void)testMessage{
NSLog(@"Test Message");
}
@end
TestProxyHandler.h
#import <Foundation/Foundation.h>
@interface TestProxyHandler : NSProxy
@property (nonatomic, strong) id object;
- (instancetype)initWithProtocol:(Protocol *)protocol andObject:(Class)clazz;
- (void)forwardInvocation:(NSInvocation *)invocation;
- (NSMethodSignature *)methodSignatureForSelector:(SEL)selector;
@end
TestProxyHandler.m
#import "TestProxyHandler.h"
#import "TestBO.h"
@implementation TestProxyHandler
- (instancetype)initWithProtocol:(Protocol *)protocol andObject:(Class)clazz{
if ([clazz conformsToProtocol:@protocol(MyProtocol)]) {
self.object = [[clazz alloc] init];
}else{
NSLog(@"Error it does not conform to protocol");
}
return self;
}
- (void)forwardInvocation:(NSInvocation *)invocation{
NSString *selString = NSStringFromSelector(invocation.selector);
NSLog(@"Called %@",selString);
[invocation invokeWithTarget:self.object];
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)selector {
return [self.object methodSignatureForSelector:selector];
}
@end
我用它调用了它
id <MyProtocol> delegate = (TestBO *)[[TestProxyHandler alloc] initWithProtocol:@protocol(MyProtocol) andObject:[TestBO class]];
[delegate testMessage];
但即使初始化器显示该消息,我也无法在Swift中使其工作
TestHandler.swift
import Foundation
class TestHandler: NSProxy {
var object: AnyObject
convenience override init(`protocol`: Protocol, andObject clazz: AnyClass) {
if clazz.conformsToProtocol() {
self.object = clazz()
}
else {
NSLog("Error it does not conform to protocol")
}
}
}
有没有人有任何线索在swift中做到这一点?
编辑:
在java中,您可以使用Proxy.newProxyInstance调用创建方法的运行时实现,但这可以在iOS中实现吗?用swift?任何线索?
最佳答案 与Objective C和Swift相比,Swift提供了对运行时语言访问的极其有限的访问.所以根据我的研究到现在为止还不能做到:(
我甚至尝试在swift中继承NSProxy类,但是无法调用super.init并且代码从不编译,但是在Objective C中同样的东西也可以工作
所以我最终采用了这种方法
我使用创建了一个协议
@objc protocol SomeProt {
// Some method
}
请注意关键字@objc在协议之前是必不可少的,否则您将无法将其作为变量传递,同时添加@objc将协议的使用限制为目标c运行时功能,因此不要期望在swift中获得协议的完整功能
public func someMethod(`protocol` : Protocol, implementation : AnyClass) {
let isImplemented : Bool = implementation.conformsToProtocol(`protocol`)
// some code
}
如果你需要在某些词典中使用它或者它应该符合NSCopying类的地方那么使用
NSStringFromProtocol
和
NSProtocolFromString
方法
现在我已经写了一个目标c辅助类来进行初始化
ObjcHelper.h
#import <Foundation/Foundation.h>
@interface ObjcHelper : NSObject
+(NSObject *)objectForClass:(Class)clazz;
@end
ObjcHelper.m
#import "ObjcHelper.h"
@implementation ObjcHelper
+ (NSObject *)objectForClass:(Class)clazz{
return [[clazz alloc] init];
}
@end
现在用它
let prot : SomeProt = ObjcHelper.objectForClass(NSClassFromString("PROT_HANDLER_CLASS_NAME")) as! SomeProt
但是将来如果有人能提供更好的答案,请务必在此发布