Swift类中的静态vs类函数/变量?


416

以下代码在Swift 1.2中进行编译:

class myClass {
    static func myMethod1() {
    }
    class func myMethod2() {
    }
    static var myVar1 = ""
}

func doSomething() {
    myClass.myMethod1()
    myClass.myMethod2()
    myClass.myVar1 = "abc"
}

静态函数和函数有什么区别?我应该使用哪一个?何时使用?

如果我尝试定义另一个变量class var myVar2 = "",它说:

类中尚不支持的类存储属性;您是说“静态”吗?

如果支持此功能,则静态变量和变量之间有什么区别(即,当两者都在类中定义时)?我应该使用哪一个?何时使用?

(Xcode 6.3)


Answers:


690

staticclass与类两者关联的方法,而不是一个类的实例。不同之处在于子类可以覆盖class方法。他们不能覆盖static方法。

class 属性在理论上将以相同的方式起作用(子类可以覆盖它们),但是在Swift中还不可能。


89
那么,final class一个类中的函数和“静态”函数有什么区别?
hippo_san 2015年

57
@hippo_san,在基类中,两者在功能上是相同的。但是,final在子类中使用时,可用于切断进一步的覆盖。两者都有自己的位置,我想说在类函数上使用staticfinal时很简单,取决于您的样式选择。
安德鲁·罗宾逊

8
嗯,所以static func foo(){}在Swift中就像public static final foo(){}在Java中一样?
Supuhstar 2015年

3
@Supuhstar:基本上是。
mipadi 2015年

2
@mipadi我现在明白了。对于类函数,可以将“ static”替换为“ final class”,但是对于类中的属性,我们只能具有静态属性,而不能具有类属性。因此,“静态”关键字仍然有它的位置。
allenlinli

72

我在操场上尝试了mipadi的回答和评论。并考虑共享它。干得好。我认为mipadi的答案应标记为已接受。

class A{
    class func classFunction(){
    }
    static func staticFunction(){
    }
    class func classFunctionToBeMakeFinalInImmediateSubclass(){
    }
}

class B: A {
    override class func classFunction(){

    }

    //Compile Error. Class method overrides a 'final' class method
    override static func staticFunction(){

    }

    //Lets avoid the function called 'classFunctionToBeMakeFinalInImmediateSubclass' being overriden by subclasses

    /* First way of doing it
    override static func classFunctionToBeMakeFinalInImmediateSubclass(){
    }
    */

    // Second way of doing the same
    override final class func classFunctionToBeMakeFinalInImmediateSubclass(){
    }

    //To use static or final class is choice of style.
    //As mipadi suggests I would use. static at super class. and final class to cut off further overrides by a subclass
}

class C: B{
    //Compile Error. Class method overrides a 'final' class method
    override static func classFunctionToBeMakeFinalInImmediateSubclass(){

    }
}

27

关于OOP,答案太简单了:

子类可以覆盖方法,但不能覆盖静态方法。

除了您的文章之外,如果您想声明一个变量(就像一样class var myVar2 = ""),还应该按照以下步骤进行:

class var myVar2: String {
    return "whatever you want"
}

23

在我的一个项目中,我也感到困惑,发现这篇文章非常有帮助。在我的操场上也尝试过,这里是摘要。希望这可以帮助别人与存储性能和类型的功能staticfinalclass,覆盖类乏等。

class Simple {

    init() {print("init method called in base")}

    class func one() {print("class - one()")}

    class func two() {print("class - two()")}

    static func staticOne() {print("staticOne()")}

    static func staticTwo() {print("staticTwo()")}

    final func yesFinal() {print("yesFinal()")}

    static var myStaticVar = "static var in base"

    //Class stored properties not yet supported in classes; did you mean 'static'?
    class var myClassVar1 = "class var1"

    //This works fine
    class var myClassVar: String {
       return "class var in base"
    }
}

class SubSimple: Simple {
    //Successful override
    override class func one() {
        print("subClass - one()")
    }
    //Successful override
    override class func two () {
        print("subClass - two()")
    }

    //Error: Class method overrides a 'final' class method
    override static func staticOne() {

    }

    //error: Instance method overrides a 'final' instance method
    override final func yesFinal() {

    }

    //Works fine
    override class var myClassVar: String {
        return "class var in subclass"
    }
}

这是测试样本:

print(Simple.one())
print(Simple.two())
print(Simple.staticOne())
print(Simple.staticTwo())
print(Simple.yesFinal(Simple()))
print(SubSimple.one())
print(Simple.myStaticVar)
print(Simple.myClassVar)
print(SubSimple.myClassVar)

//Output
class - one()
class - two()
staticOne()
staticTwo()
init method called in base
(Function)
subClass - one()
static var in base
class var in base
class var in subclass

23

Swift 4中的测试显示了模拟器的性能差异。我使用“ class func”创建了一个类,并使用“ static func”构建了一个结构,并进行了测试。

静态函数是:

  • 无需编译器优化,速度提高20%
  • 启用优化-整个模块优化时,速度提高38%

但是,在iOS 10.3下的iPhone 7上运行相同的代码将显示完全相同的性能。

如果您想测试一下自己,这是Swift 4中针对Xcode 9的示例项目 https://github.com/protyagov/StructVsClassPerformance


这是在模拟器还是物理设备上?
mmr118 '18


7

除上述方法外,静态方法是静态调度,这意味着编译器知道在运行时将执行哪种方法,因为静态方法不能被覆盖,而类方法可以是动态调度,因为子类可以覆盖这些方法。


0

还有另外一个区别。类只能用于定义计算类型的类型属性。如果需要存储的类型属性,请使用static。

类别:-参考类型

struct:-值类型


0

class用于Reference Type(类)内部:

  • 计算属性
  • 方法
  • 可以被子类覆盖

static用于Reference TypeValue Type(类,枚举):

  • 计算属性和存储属性
  • 方法
  • 不能由子类更改
protocol MyProtocol {
//    class var protocolClassVariable : Int { get }//ERROR: Class properties are only allowed within classes
    static var protocolStaticVariable : Int { get }

//    class func protocolClassFunc()//ERROR: Class methods are only allowed within classes
    static func protocolStaticFunc()
}

struct ValueTypeStruct: MyProtocol {
    //MyProtocol implementation begin
    static var protocolStaticVariable: Int = 1

    static func protocolStaticFunc() {

    }
    //MyProtocol implementation end

//    class var classVariable = "classVariable"//ERROR: Class properties are only allowed within classes
    static var staticVariable = "staticVariable"

//    class func classFunc() {} //ERROR: Class methods are only allowed within classes
    static func staticFunc() {}
}

class ReferenceTypeClass: MyProtocol {
    //MyProtocol implementation begin
    static var protocolStaticVariable: Int = 2

    static func protocolStaticFunc() {

    }
    //MyProtocol implementation end

    var variable = "variable"

//    class var classStoredPropertyVariable = "classVariable"//ERROR: Class stored properties not supported in classes

    class var classComputedPropertyVariable: Int {
        get {
            return 1
        }
    }

    static var staticStoredPropertyVariable = "staticVariable"

    static var staticComputedPropertyVariable: Int {
        get {
            return 1
        }
    }

    class func classFunc() {}
    static func staticFunc() {}
}

final class FinalSubReferenceTypeClass: ReferenceTypeClass {
    override class var classComputedPropertyVariable: Int {
        get {
            return 2
        }
    }
    override class func classFunc() {}
}

//class SubFinalSubReferenceTypeClass: FinalSubReferenceTypeClass {}// ERROR: Inheritance from a final class

[参考与值类型]

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.