【问题标题】:Why class cannot be initialised in Switch statement in Swift为什么不能在 Swift 的 Switch 语句中初始化类
【发布时间】:2014-12-21 13:18:30
【问题描述】:

想知道为什么 Swift switch 语句不允许像其他语言那样实例化类以及如何解决这个问题。如果有人能在这方面提供帮助,我会很高兴。 在示例中,我创建了一个简单的 Vehicle 类,并尝试根据 classSetter 值通过 switch 实例化其子类。但是,如果 print 语句的最后一行在 switch(或任何其他类型的条件)语句中实例化,则不能打印任何类的 name 属性。

import UIKit

class Vehicle {
    var name: String {return ""}
}

class Car: Vehicle {
    override var name: String {return "Car"}
}

class Bike: Vehicle {
    override var name: String {return "Bike"}
}

var classSetter:Int = 1

switch classSetter {
case 1:
    println("Initializing Car")
    var vehicle = Car()
case 2:
    println("Initialized Bike")
    let vehicle = Bike()
default:
    println("just defaulted")
}

println("Name property from initialization is \(vehicle.name)")

【问题讨论】:

  • 非常感谢您的宝贵意见。唯一仍不清楚的是如何在所有文本输出中去掉“可选”。
  • 刚刚更新了我的答案以涵盖这个

标签: swift switch-statement


【解决方案1】:

您的两个vehicles 是在开关的{ } 中声明的,因此它们只存在于该块中(即它们的“范围”)。它们不存在于它之外,所以你不能在那里引用它们,因此会出现错误。

对此的默认解决方案(其他答案正在给出)是将vehicle 声明为开关外部的var,但这里有另一种选择:将开关包装在闭包表达式中并从中返回一个值。为什么要这样做?因为那时你可以使用let 而不是var 来声明vehicle

let vehicle: Vehicle = { // start of a closure expression that returns a Vehicle
    switch classSetter {
    case 1:
        println("Initializing Car")
        return Car()
    case 2:
        println("Initialized Bike")
        return Bike()
    default:
        println("just defaulted")
        return Vehicle()
    }
}()  // note the trailing () because you need to _call_ the expression

println("Name property from initialization is \(vehicle.name)")

如果 ifswitch 是表达式(即评估为结果)那就太好了,所以你不需要这个闭包,但现在这是一个合理的解决方法。

注意,这里有几个使用var 方法的答案建议将vehicle 设为可选值(即Vehicle?)。这不是必需的——只要保证代码在使用之前为 vehicle 赋值(编译器会为你检查),它就不必是可选的。但我仍然认为闭包表达式版本是更好的方式。

顺便说一句,您可能需要考虑为Vehicle 使用协议而不是基类,因为这样您就不必为Vehicle 提供默认但无效的name 实现:

protocol Vehicle {
    var name: String { get }
}

// one of the benefits of this is you could
// make Car and Bike structs if desired
struct Car: Vehicle {
    var name: String {return "Car"}
}

struct Bike: Vehicle {
    var name: String {return "Bike"}
}

虽然这意味着您不能从 Vehicle() 的 switch 语句中获得默认返回。但无论如何,这可能会很糟糕——可选的Vehicle?nil 代表失败可能是更好的选择:

let vehicle: Vehicle? = {
    switch classSetter {
    case 1:
        println("Initializing Car")
        return Car()
    case 2:
        println("Initialized Bike")
        return Bike()
    default:
        println("no valid value")
        return nil
    }
}()

// btw since vehicle is a Vehicle? you need to unwrap the optional somehow,
// one way is with optional chaining (will print (nil) here)
println("Name property from initialization is \(vehicle?.name)")

如果您根本不希望出现这种情况,您可以考虑将不同类型车辆的指标设为枚举,以便它只能是一组有效车辆中的一个:

enum VehicleKind {
    case Bike, Car
}

let classSetter: VehicleKind = .Car

let vehicle: Vehicle = {
    switch classSetter {
    case .Car:
        println("Initializing Car")
        return Car()
    case .Bike:
        println("Initialized Bike")
        return Bike()
    // no need for a default clause now, since
    // those are the only two possibilities
    }
}()

【讨论】:

  • 恭喜,这个答案突出了一些在 Objective-C 中不可用的 Swift 概念。
  • 感谢您的精彩回答。在删除“可选”时效果很好,我在以下情况下使用了底部答案:stackoverflow.com/questions/26347777/…
【解决方案2】:

你的假设不正确。

变量vehicle的作用域在switch里面。然后您尝试在交换机外部访问它。

您需要在开关之外创建变量。你仍然可以在里面实例化它。

var vehicle: Vehicle?

Switch... {
    Case
        vehicle = Car()
}

println(vehicle)

写在我的手机上,所以无法给出完整的正确代码,但这会给你一个想法。

【讨论】:

    【解决方案3】:

    您正在做的事情也不适用于其他语言。您需要在输入switch 之前声明vehicle 变量,以便它与println 处于同一范围内。之后,您可以在 switch 中分配您需要的任何内容。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2019-05-30
      • 2016-05-12
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多