在Java中,拥有实用程序功能列表非常普遍
public class Utils {
private Utils() {
}
public static void doSomething() {
System.out.println("Utils")
}
}
如果我在Swift中,应该使用
class
还是struct
实现类似的目的?还是没关系?类
class Utils {
private init() {
}
static func doSomething() {
print("class Utils")
}
}
结构
struct Utils {
private init() {
}
static func doSomething() {
print("struct Utils")
}
}
最佳答案
我认为,有关此问题的讨论必须从对依赖注入,它是什么以及它解决什么问题的理解开始。
依赖注入
编程就是将小部件组装成抽象的,能完成很多事情的程序集。很好,但是大型程序集很难测试,因为它们非常复杂。理想情况下,我们要测试小型零件及其组装方式,而不是测试整个组件。
为此,单元测试和集成测试非常有用。但是,每个全局函数调用(包括对静态函数的直接调用,它们实际上只是一个不错的小名称空间中的全局函数)都是责任。这是一个没有接缝的固定结,可以通过单元测试将其断开。例如,如果您有一个直接调用排序方法的视图控制器,则无法隔离该排序方法来测试视图控制器。有一些后果:
sort
方法由使用它的每段代码进行测试)。这不利于定期运行它们,这很重要。 动态调度会引入接缝。接缝是代码中可配置性的要点。可以更改一种实现的位置,然后放入另一种实现的位置。例如,您可能需要一个
MockDataStore
,BetaDataStore
和ProdDataStore
,具体取决于环境。如果所有这三种类型都符合一个通用协议,则可以编写依赖代码以依赖该协议,该协议允许根据需要交换这些不同的实现。为此,对于您希望能够隔离的代码,您永远不要使用全局函数(例如
foo()
),或直接调用静态函数(实际上是命名空间中的全局函数),例如FooUtils.foo()
。如果要用foo()
替换foo2()
或用FooUtils.foo()
替换BarUtils.foo()
,则不能。依赖注入是“注入”依赖关系(取决于配置,而不是对其进行硬编码的一种做法。)而不是对依赖关系硬编码
FooUtils.foo()
,而是创建一个Fooable
接口,该接口需要函数foo
。在依赖代码中(会调用foo
的类型,您将存储一个Fooable
类型的实例成员。当您需要调用foo
时,请调用self.myFoo.foo()
这样,您将调用已提供(“注入”)到Fooable
实例的任何self
实现。在构造时,它可以是MockFoo
,NoOpFoo
,ProdFoo
,它不在乎,它知道它的myFoo
成员具有foo
函数,可以调用它来照顾所有这是foo的需求。上面相同的事情也可以实现基类/子类关系,对于这些意图和目的,它的行为就像协议/符合类型的关系一样。
交易工具
如您所见,Swift在Java中提供了更多的灵活性。编写函数时,可以选择使用:
在每个地方都有一个合适的时间和地点。 Java将选项2和3推倒了(主要是选项2),而Swift使您可以更经常地依靠自己的判断。我将讨论每种情况,何时使用或不使用。
1)全局功能
这些对于其中一种实用程序功能可能很有用,在这种情况下,以特定方式将它们“分组”并没有太大好处。
优点:
foo
,而不是FooUtils.foo
)而导致的短名称缺点:
2)实例功能
优点:
self
的成员),几乎总是比全局状态更可取。 缺点:
MathUtils
对象,则只需使用其pow
实例方法,该方法实际上并不使用任何实例数据(例如MathUtils().pow(2, 2)
)3)静态功能
优点:
缺点:
4)类功能
对于类,
static func
类似于final class func
。 Java支持这些功能,但是在Swift中,您也可以具有非最终类功能。唯一的区别是它们支持覆盖(由子类覆盖)。所有其他优点/缺点与静态功能共享。我应该使用哪一个?
这取决于。
如果您要编写的代码是想进行测试的代码,那么全局函数就不是候选对象。您必须使用基于协议或继承的依赖注入。如果代码不具有某种实例状态(并且永远不会需要它),则静态函数可能是适当的,而当需要实例状态时,则应该使用实例函数。如果不确定,则应该选择一个实例函数,因为如前所述,将一个函数从静态转换为实例是一项API重大更改,如果可能的话,您应该避免这样做。
如果新功能真的很简单,则可能是全局功能。例如。
print
,min
,abs
,isKnownUniquelyReferenced
等。但前提是没有有意义的分组。有一些例外情况需要注意:func formatDecimal(_: Decimal) -> String { ... }
func formatCurrency(_: Price) -> String { ... }
func formatDate(_: Date) -> String { ... }
func formatDistance(_: Measurement<Distance>) -> String { ... }
如果将这些功能归为一类,可以更好地表达。在这种情况下,我们不需要实例状态,因此我们不需要使用实例方法。另外,有一个
FormattingUtils
实例是有意义的(因为它没有状态,没有任何东西可以使用该状态),因此,禁止创建实例是一个好主意。一个空的enum
就是这样做的。enum FormatUtils {
func formatDecimal(_: Decimal) -> String { ... }
func formatCurrency(_: Price) -> String { ... }
func formatDate(_: Date) -> String { ... }
func formatDistance(_: Measurement<Distance>) -> String { ... }
}
这种逻辑分组不仅“有意义”,而且还具有使您更进一步支持这种类型的依赖注入的额外好处。您需要做的就是将接口提取到新的
FormatterUtils
协议中,将此类型重命名为ProdFormatterUtils
,并更改依赖代码以依赖协议而不是具体类型。 func enableLED(pin: Int) { ... }
func disableLED(pin: Int) { ... }
func checkLEDStatus(pin: Int) -> Bool { ... }
我们不仅可以从上面的第1点开始应用重构,而且还可以注意到
pin: Int
是重复的参数,可以将其更好地表示为类型的实例。相比:class LED { // or struct/enum, depending on the situation.
let pin: Int
init(pin: Int)? {
guard pinNumberIsValid(pin) else { return nil }
self.pin = pin
}
func enable() { ... }
func disable() { ... }
func status() -> Bool { ... }
}
与从第1点进行重构相比,这将 call 站点从
LEDUtils.enableLED(pin: 1)`
LEDUtils.disableLED(pin: 1)`
至
guard let redLED = LED(pin: 1) else { fatalError("Invalid LED pin!") }
redLED.enable();
redLED.disable();
这不仅更好,而且现在我们有了一种方法,可以使用
Int
和LED
来清楚地区分期望任何旧整数的函数和期望LED引脚号的函数。我们还为所有与LED有关的操作提供了一个中心位置,并为我们可以验证引脚号确实有效的位置提供了一个中心点。您知道如果您提供了LED
实例,则pin
是有效的。您不需要自己检查它,因为您可以依靠已经检查过的它(否则这个LED
实例将不存在)。 关于ios - 您通常使用类或结构来定义实用程序功能列表吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56963022/