swift - Swift 模块/类如何解决缺少对 "protected"成员的语言支持的问题?

标签 swift access-control

我遇到这样一种情况,我在模块中定义了一个可重用的基类,我想提供某些只能由子类而不是该子类的外部用户调用的函数。

我正在编写一个框架并将其打包为 Swift 模块。我的框架的一部分包括一个基类,可以对其进行子类化以添加功能,但是派生类也有更进一步的外部目的。想象一下定义一种新的 View :它派生自 UIViewNSView,然后提供额外的逻辑,然后由另一方实例化。

在这种情况下,我是定义类似于 UIView 的类的人,该类旨在被子类化,并且随之而来的是许多私有(private) UIView 内部东西,比如测量、安排,谁知道呢,内部的东西。

关键是,这个新 View 类的最终用户不想看到支持子类化的体系结构的内部结构,那些应该完全在子类所代表的黑盒子里。

令我震惊的是,这在 Swift 中是不可能的。

我真的不明白为什么 Swift 摆脱了 protected 访问控制。 According to Apple ,我只想公开给子类的函数“在子类之外并不是很有用,所以保护并不重要”。

我错过了什么吗?这是 Swift 根本无法支持的一整类设计模式吗?

我想到的一个想法是,我或许可以将我类(class)的公共(public)-公共(public)和私有(private)-公共(public)部分分成两部分,也许使用协议(protocol),公共(public)-公共(public)用户只能看到公共(public)协议(protocol)和“私有(private)” "公共(public)用户也会看到“私有(private)”协议(protocol)。唉,这似乎是为了过去免费的东西而进行的大量工程。

最佳答案

FWIW — 自从 Swift 中有访问控制之前,我一直在要求在 Swift 中提供更好的访问控制(包括 protected)。现在,在我们被告知尝试使用 Swift 方法进行访问控制 3.5 年之后,Swift 一直是我近 3 年的主要语言,我仍然认为访问控制范式很笨拙,无法对易于理解的概念进行建模几乎所有相似的语言。

对我来说最大的缓和因素是 Swift 让我在 95% 的时间里不再使用继承和子类化,我认为这是一件好事。所以这个问题出现的次数比其他情况少。但是对于您所描述的情况,没有一种等效的方法可以仅使用协议(protocol)和协议(protocol)扩展来完成您正在做的事情,因此您要么用可能有害的内部细节污染公共(public) API,要么使用一些解决方法(比如下一个)它具有尽可能少的公共(public) API 暴露,并以样板和笨拙为代价模拟您想要的东西。

也就是说,我采用的方法在某种程度上受到了 Objective C 的启发,其中也没有真正的 protected 访问控制,但约定是声明一个公共(public) API header (客户端代码将导入该 header )和引用)和一个特殊的“+Subclassing” header ,只有子类才会在其实现中导入,使它们能够看到非公共(public)消费的内部结构。

在 Swift 中,这也不是直接可能的,但是给定一个这样的类:

open class SomeClass {
    private var foo: String
    private var bar: Data
    public init(){
        foo = "foo"
        bar = Data()
    }
    private func doInternalThing() {
        print(foo)
    }
}

您可以通过扩展添加一个嵌套的“ protected ”包装器(必须与您的类声明在同一个文件中),它采用类(或子类)的实例并将 protected 级别的内部结构公开为一种代理人:

// Create a nested "Protected" type, which can accept an instance of SomeClass (or one of its subclasses) and expose the internal / protected members on it
public extension SomeClass {
    public class Protected {
        unowned private var someClass: SomeClass
        public var foo: String {
            get {
                return someClass.foo
            }
            set {
                someClass.foo = newValue
            }
        }
        public init(_ someClass: SomeClass) {
            self.someClass = someClass
        }
        public func doInternalThing() {
            someClass.doInternalThing()
        }
    }
}

在框架之外,在客户端应用程序中, protected 成员在这样的子类中访问:

class SomeSubclass: SomeClass {
    private lazy var protected: SomeClass.Protected = { SomeClass.Protected(self) }()
    func doSomething() {
        protected.foo = "newFoo"  // Accesses the protected property foo and sets a new value "newFoo"
        protected.doInternalThing() // Prints "newFoo" by calling the protected method doInternalThing which prints the foo property.
    }
}

这种方法有利也有弊。缺点主要是您需要编写的样板数量,以便将所有属性和函数从 Protected 包装器映射到实际的类实例,如上所示。此外,无法避免这样一个事实,即消费者会将 SomeClass.Protected 视为一种公开可见的类型,但希望很明显它不应该被使用,并且很难随意使用它,所以它不会发生。

优点是在创建子类时没有太多样板文件或给客户带来痛苦,并且很容易声明一个惰性的“ protected ”var 来获得所需的 API。非子类偶然或无意中偶然发现或使用此 API 的可能性很小,而且它大部分是根据需要隐藏的。 SomeSubclass 的实例不会在代码完成或外部代码中显示任何额外的 protected API。

我鼓励任何其他认为访问控制——或者在这种情况下,API 可见性和组织——比现在的 Swift 更容易的人通过 Swift 论坛、Twitter 或 bugs.swift 让 Swift 团队知道。组织。

关于swift - Swift 模块/类如何解决缺少对 "protected"成员的语言支持的问题?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49097205/

相关文章:

security - 领域驱动设计中的访问控制

ios - DragInteraction itemsForBeginning 未触发

Swift 将数据发送到不同 Storyboard中的 View Controller

security - PouchDB/CouchDB 的每个文档用户访问控制

swift - 无法将属性声明为开放,因为其类型使用内部类型 (Typealias)

命名空间中的 C++ 前向声明和友元

java - 在 Java 中使用默认构造函数扩展类

ios - 在 inputAccessoryView 上方插入 subview

ios - 由于未捕获的异常 'NSInvalidArgumentException' 而终止应用程序 - ios google 登录

ios - CocoaPods 未检测到隐式依赖关系