gpt4 book ai didi

swift - 当从另一个 ViewController 类调用方法时,Xcode 发现一个 nil 变量,而该变量不应该存在

转载 作者:行者123 更新时间:2023-11-30 10:34:05 24 4
gpt4 key购买 nike

我在创建一个 MacOS 应用程序(我的第一个应用程序,不属于“您的第一个应用程序”教程的一部分)的过程中,涉及到主窗口中的大量用户选项,并且厌倦了我的 ViewController文件已经变得一团糟,从长远来看是无法维护的。

我决定将其分成较小的 block 来输入多个 View Controller ,以便使用 UIBuilder 中的容器 View 来嵌入 View 使其更易于管理,但我发现的所有教程要么是针对过时版本的 Xcode/Swift,要么是关于管理多个iOS 中的 View ,所以我不得不进行一些推断,但我可能做错了。

现在,当一个 ViewController 中的方法被另一个 ViewController 调用时,我遇到了错误,即使该方法在被自己的 View Controller 调用时可以正常工作。

要么我遗漏了一些明显的东西,要么我设置错误。

全局变量:

var inputPathUrl = URL?
var outputExtension: String = ""

@IBOutlets 和 InOutViewController 类的本地属性:

@IBOutlet weak var inputTextDisplay: NSTextField!
@IBOutlet weak var outputTextDisplay: NSTextField!
@IBOutlet weak var inputBrowseButton: NSButton!
@IBOutlet weak var outputBrowseButton: NSButton!

var outputDirectoryUrl: URL?
var inputFilePath: String = ""

@IBOutlets 用于 OptionsViewController

@IBOutlet weak var Button1: NSButton!
@IBOutlet weak var Button2: NSButton!
@IBOutlet weak var Button3: NSButton!
@IBOutlet weak var Button4: NSButton!
@IBOutlet weak var Button5: NSButton!

InOutViewController 类的方法:

@IBAction func InputBrowseClicked(\_ sender: Any) {  
let inputPanel = NSOpenPanel()
inputPanel.canChooseFiles = true
inputPanel.canChooseDirectories = false
inputPanel.allowsMultipleSelection = false
inputPanel.allowedFileTypes = \["aax"\]
let userChoice = inputPanel.runModal()
switch userChoice{
case .OK :
if let inputFileChosen = inputPanel.url {
inputFileUrl = inputFileChosen // define global variable that will be called by other methods in other classes to check if an input file has been chosen
updateInputText() // call methods to display path strings in text fields
updateOutputText()
}
case .cancel :
print("user cancelled")
default :
break
}
}

@IBAction func outputBrowseClicked(_ sender: Any) {
let outputPanel = NSOpenPanel()
outputPanel.canChooseFiles = false
outputPanel.canChooseDirectories = true
outputPanel.allowsMultipleSelection = false
let userChoice = outputPanel.runModal()
switch userChoice{
case .OK :
if let outputUrl = outputPanel.url {
outputDirectoryUrl = outputUrl
updateOutputText()
}
case .cancel :
print("user cancelled")
default:
break
}
}

func updateInputText() {
// call getOutputOption method to see which radio button is selected
OptionsViewController().getOutputOption()
if inputFileUrl != nil {
inputFilePath = inputFileUrl!.path
inputTextDisplay.stringValue = inputFilePath
}
}
func updateOutputText() {
// derive output file path and name from input if no output location is chosen
if inputFileUrl != nil && outputDirectoryUrl == nil {
let outputDirectory = inputFileUrl!.deletingPathExtension()
let outputDirectoryPath = outputDirectory.path
let outputPath = outputDirectoryPath + "(outputExtension)"
outputTextDisplay.stringValue = outputPath
} else if inputFileUrl != nil && outputDirectoryUrl != nil {
// derive default file name from input but use selected output path if one is chosen
let outputDirectoryPath = outputDirectoryUrl!.path
let outputFile = inputFileUrl!.deletingPathExtension()
let outputFilename = outputFile.lastPathComponent
// derive file extension from getOutputOption method of OptionsViewController class
let outputPath = outputDirectoryPath + "/" + outputFilename + "(outputExtension)"
outputTextDisplay.stringValue = outputPath
}
}

最后一行 (outputTextDisplay.stringValue = outputPath) 是我遇到 fatal error 的地方,但仅当我从 @IBAction 调用此方法时OptionsViewController 中的输出格式单选按钮可在选择不同的文件扩展名时更新输出显示。当我从 InOutViewController 中的 actions 方法调用该方法时,它工作正常。

以下是 OptionsViewController 类中的 @IBAction 方法和 getOutputOption 方法:

@IBAction func radioButtonClicked(_ sender: Any) {
getOutputOption()
// update display with new file extension
InOutViewController().updateOutputText()
}

func getOutputOption() {
// make sure an input file has been chosen
if inputFileUrl != nil {
// check which radio button is selected and derive output file format based on selection
// not sure why I need to specify the button isn't nil, since one is ALWAYS selected, but I was getting a fatal error without doing so
if (Button1 != nil) && Button1.state == .on {
outputExtension = ".extA"
} else if (Button2 != nil) && Button2.state == .on {
outputExtension = ".extB"
} else if (Button3 != nil) && Button3.state == .on {
outputExtension = ".extC"
} else if (Button4 != nil) && Button4.state == .on {
outputExtension = ".extD"
} else {
outputExtension = ".extE"
}
}
}

我确信我错过了一些明显的东西,但就像我说的,这是我第一次使用多个 View Controller ,我不确定我是否正确实现了它们,而且我只编写了一些代码几周了,所以我无法发现哪里出了问题。

最佳答案

我的猜测是outputTextDisplayInOutViewController中的一个IBOutlet,并且它被声明为隐式解包。 (类似这样的:)

var outputTextDisplay: UITextField!

如果您从 InOutViewController 中的 IBAction 之一引用这样的变量,一切都会很好,因为此时您的 View Controller 的 View 已加载。

另一方面,如果您从另一个 View Controller 调用 updateOutputText(),则您的 InOutViewController 的 View 可能尚未加载,因此输出文本显示仍然为零。当一个变量被声明为隐式解包时(在类型末尾使用 ! ),那么任何时候你引用它,编译器都会强制解包它,如果它为 nil,你就会崩溃。

您应该更改 updateOutputText() 以使用 ? 来解包变量。这可以阻止隐式解包的选项崩溃。像这样的事情:

func updateOutputText() {
// derive output file path and name from input if no output location is chosen
if inputFileUrl != nil && outputDirectoryUrl == nil {
let outputDirectory = inputFileUrl!.deletingPathExtension()
let outputDirectoryPath = outputDirectory.path
let outputPath = outputDirectoryPath + "(outputExtension)"
outputTextDisplay?.stringValue = outputPath //Note the `?`
} else if inputFileUrl != nil && outputDirectoryUrl != nil {
// derive default file name from input but use selected output path if one is chosen
let outputDirectoryPath = outputDirectoryUrl!.path
let outputFile = inputFileUrl!.deletingPathExtension()
let outputFilename = outputFile.lastPathComponent
// derive file extension from getOutputOption method of OptionsViewController class
let outputPath = outputDirectoryPath + "/" + outputFilename + "(outputExtension)"
outputTextDisplay?.stringValue = outputPath //Note the `?`
}
}
}

代码 outputTextDisplay?.stringValue = outputPath 使用“可选链接”,这会导致编译器检查 outputTextDisplay 是否为 nil,如果为 nil,则停止执行代码.

请注意,如果进行此更改,当您调用 updateOutputText() 函数时,您的字符串值将不会安装到 outputTextDisplay 字段中。您需要在调用 viewDidLoad() 后安装该值。 (也许在您的 viewDidLoad 或 viewWillAppear 中。)

编辑:

这段代码:

@IBAction func radioButtonClicked(_ sender: Any) {
getOutputOption()
// update display with new file extension
InOutViewController().updateOutputText()
}

这是非常错误的。为了响应用户单击单选按钮,InOutViewController() 位会创建一个 InOutViewController 的一次性实例,并尝试调用其 updateOutputText() 方法,然后忘记新创建的 InOutViewController。不要这样做。

您需要一种方法来跟踪屏幕上的 subview Controller 。为了向您展示如何做到这一点,您需要解释如何创建各种 View Controller 。您使用嵌入转场吗?

关于swift - 当从另一个 ViewController 类调用方法时,Xcode 发现一个 nil 变量,而该变量不应该存在,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58459129/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com