类FooClass应该只允许通过其sharedInstance进行交互.我试图通过不允许任何人访问FooClass的init()来防止滥用.
我尝试了一些不同的方法,但都没有工作:
使用私人关键字:
class FooClass: NSObject { // singleton static let sharedInstance = FooClass() let value: String private override init() { self.value = "asdf" } } // this should be a compile error,but it is not let foo = FooClass()
使用@available:
class FooClass: NSObject { // singleton // COMPILE ERROR - "init is unavailable. use sharedInstance" static let sharedInstance = FooClass() let value: String @available(*,unavailable,message="use sharedInstance") override init() { self.value = "asdf" } } // COMPILE ERROR - as it should be let foo = FooClass()
我也试过使用内部,但仍然没有运气.
UPDATE
如果将第一个版本移动到自己的文件,则第一个版本可以工作,但是该类的ObjC版本仍然允许调用init.有任何想法吗?
SWIFT_CLASS("_TtC11SwiftToObjC8FooClass") @interface FooClass : NSObject + (FooClass * __nonnull)sharedInstance; @property (nonatomic,readonly,copy) Nsstring * __nonnull value; - (nonnull instancetype)init OBJC_DESIGNATED_INITIALIZER; @end
解决方法
这个答案解决了Swift 2.在Swift 3中,看起来方法的访问级别从Swift正确导入到Objective-C,并且不需要标记为NS_UNAVAILABLE以禁止它可用.当然,Swift中没有新方法,因此仍需要将其标记为NS_UNAVAILABLE以正确维护单例.
只要将类放入自己的文件中,您的第一种方法就会起作用.访问控制关键字private表示定义的功能仅在包含文件中可用.
但是,如您所说,在Objective-C中使用Swift类将删除私有程序给您的保护.我相信这是因为标记为private的任何内容都不会在编译器生成的导入头文件中有条目.因此,从NSObject继承的init函数是可用的,因为它没有被重写.
我找到的解决方案是创建另一个头文件,显式声明一个无法调用的init函数.
Swift课程:
@objc(FooClass) class FooClass: NSObject { // singleton static let sharedInstance = FooClass() let value: String private override init() { self.value = "asdf" } }
Objective-C标题:
@interface FooClass (SharedInstance) + (instancetype) new NS_UNAVAILABLE; - (instancetype) init NS_UNAVAILABLE; @end
您还必须阻止new,因为如果不这样做,则可以通过它创建类的实例.
测试:
FooClass* foo = [FooClass sharedInstance]; // All good here FooClass* foo2 = [[FooClass alloc] init]; // 'init' is unavailable FooClass* foo3 = [FooClass new]; // 'new' is unavailable
我这里有一个基本的示例项目:SharedInstance project git
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。