为什么在创建实例时我的 IBOutlet 连接变量没有实例化?

Why are my IBOutlet-connected vars not instantiated when I create an instance?

我设置了一些全局变量等于 class 实例化后的参数。 class 定义如下:

import Foundation
import UIKit

class LogIn : UIViewController{

    var test:Int = 1

    @IBOutlet var login_icon: UIImageView!

    @IBOutlet var username_field: UITextField!
    @IBOutlet var password_field: UITextField!
    @IBOutlet var login_button: UIButton!
    @IBOutlet var createaccount_button: UIButton! 

}

全局变量是这样实例化的(在 viewDidAppear 方法中):

func showLoginView(){
    self.startup_animation.image = self.seq_6
    let login_view = LogIn()
    self.login_icon = login_view.login_icon.image //Doesnt work
    self.username_field = login_view.username_field //Doesnt work
    self.password_field = login_view.password_field //Doesnt work
    self.login_button = login_view.login_button //Doesnt work
    self.createaccount_button = login_view.createaccount_button  //Doesnt work
    self.int_testval = login_view.test //Works  
}

实例化后所有 IBOutlets 连接的变量都是 nil 值,但 var test 实例化很好。这是为什么?

谢谢!

其他上下文 showLoginView 函数与 viewDidAppear 函数一起调用。整个 class 看起来是这样的:

class StartupViewController: UIViewController {

var images: [UIImage]!

var animatedImage: UIImage!

@IBOutlet var startup_animation: UIImageView!
@IBOutlet var startup_text: UIImageView!

var login_icon: UIImage!
var username_field: UITextField!
var password_field: UITextField!
var login_button: UIButton!
var createaccount_button: UIButton!


var seq_1 : UIImage!
var seq_2 : UIImage!
var seq_3 : UIImage!
var seq_4 : UIImage!
var seq_5 : UIImage!
var seq_6 : UIImage!
var seq_7 : UIImage!
var seq_8 : UIImage!

var rec_1 : UIImage!
var rec_2 : UIImage!
var rec_3 : UIImage!
var rec_4 : UIImage!
var rec_5 : UIImage!
var rec_6 : UIImage!

override func viewDidLoad() {
    super.viewDidLoad()

}

override func viewDidAppear(_ animated: Bool) {

    super.viewDidAppear(animated)

    seq_1 = UIImage(named: "Eceipt_seq1")
    seq_2 = UIImage(named: "Eceipt_seq2")
    seq_3 = UIImage(named: "Eceipt_seq3")
    seq_4 = UIImage(named: "Eceipt_seq4")
    seq_5 = UIImage(named: "Eceipt_seq5")
    seq_6 = UIImage(named: "Eceipt_seq6")
    seq_7 = UIImage(named: "Eceipt_seq7")
    seq_8 = UIImage(named: "Eceipt_seq8")

    images = [seq_1 ,seq_2, seq_3, seq_4, seq_5, seq_6, seq_7, seq_8]

    animatedImage = UIImage.animatedImage(with: images, duration: 0.8)

    startup_animation.image = animatedImage

    let transition_text = UIImage(named: "Eceipt_logo")

    UIView.transition(with: self.startup_text, duration: 3.0, options: .transitionCrossDissolve,
                      animations: {
                        self.startup_text.image = transition_text
    },
                      completion: { (finished: Bool) -> () in
                        self.showLoginView()
    })

}

func showLoginView(){
    self.startup_animation.image = self.seq_6
    let login_view = LogIn()
    self.login_icon = login_view.login_icon.image
    self.username_field = login_view.username_field
    self.password_field = login_view.password_field
    self.login_button = login_view.login_button
    self.createaccount_button = login_view.createaccount_button
}

我认为您误解了 @IBOutlet 的用法。

@IBOutlet 只是表示它是 xib 或故事板文件中视图或其他对象的出口。

你的观点是 nil 不是因为它们是出口,而是因为它们是隐式展开的可选值,如果你不给它一个值,它就是 nil

test 不是零,因为 1) 它不是可选类型,并且 2) 它的值为 1.

我建议您将 @IBOutlet 放在 UIViewControllerUIView 子 class 中,并将 storyboard/xib 文件中的适当视图连接到它.这样,它们将在 viewDidLoad 方法或 awakeFromNib 方法被调用时被初始化。

编辑:

您似乎已经控制+拖动视图到 LogIn class。在这种情况下,请检查出口左侧的圆圈是否已填满。如果是,则表示它们已正确连接。您只需要等到 awakeFromNib 被调用。如果不是,请尝试重新连接它们。或者,只需将所有插座直接连接到视图控制器,这样可以省去很多麻烦。

编辑 2:

现在我明白了你想要做什么,下面是你应该做的。您似乎只想呈现 LogIn VC。因此,与其将每个视图分配给 self,不如使用 present 方法正确呈现它。

let vc = UIStoryboard.main!.instantiateViewController(withIdentifier: "Login")
present(vc, animated: true, completion: nil)

记得在情节提要中为您的登录 VC 提供情节提要 ID "Login"。

我认为问题在于您通过 let login_view = LogIn() 实例化了 LogIn 视图控制器。当您使用空初始化程序实例化视图控制器时,每个 IBOutlet 都将为 nil,因为视图控制器不是故事板(或 xib)中的 "taken"。如果你想使用 iboutles,你必须从故事板或 xib 中实例化视图控制器;在您的情况下(使用情节提要),您应该通过以下方式实例化它:

let storyboard = UIStoryboard(name: "Main", bundle: nil) // name is the filename of the storyboard that contains the vc
let vc = storyboard.instantiateViewController(withIdentifier: "LogIn") as! LogIn // you can cast to have the exact view controller type
present(vc, animated: true, completion: nil)

确保故事板中的视图控制器具有标识符 "LogIn"。