在 View 中初始化 @ObservedObject 时出现内存泄漏

A.A*_*ino 5 ios swift swiftui

我有一个ViewModel类,ObservableObject并且在初始化其相应的视图时初始化它。似乎如果我ViewModel对视图内部有任何绑定,ViewModel则泄漏了。

例如,如果我在工作表内显示所述视图,则每次我展示工作表时,都会分配一个新的引用,并且当工作表被解除时它不会被释放。引用计数随着我展示工作表的次数不断增加。

我是否遗漏了什么,或者@ObservedObject属性包装器不应该以这种方式使用?

这是一个展示问题的简单示例。该deinit函数永远不会被调用ViewModel

struct NewContactView: View {

    class ViewModel: ObservableObject {

        @Published var firstName = ""
        @Published var lastName = ""
        @Published var email = ""
        @Published var phoneNumber = ""

        init() {
            print("INIT")
        }

        deinit {
            print("DEINIT")
        }

    }

    @ObservedObject private var viewModel = ViewModel()

    var didCreateNewContact: (Contact) -> Void = { _ in }

    var body: some View {
        NavigationView {
            Form {
                Section(header: Text("Names")) {
                    TextField("First Name", text: $viewModel.firstName)
                    TextField("Last Name", text: $viewModel.lastName)
                }

                Section(header: Text("Details")) {
                    TextField("Email", text: $viewModel.email)
                    TextField("Phone Number", text: $viewModel.phoneNumber)
                }

                Button(action: {}) {
                    Text("Save")
                }
            }
        }
    }
}

Run Code Online (Sandbox Code Playgroud)

编辑:添加了显示工作表的代码

struct ContactsListView: View {

    @EnvironmentObject var contactStore: ContactStore

    @State private var isCreatingNewContact = false

    var body: some View {

            List(contactStore.contacts) { contact in
                ContactListItem(contact: contact)
            }
            .navigationBarItems(trailing: Button(action: { self.isCreatingNewContact = true }) {
                Image(systemName: "plus")
            })


            .sheet(isPresented: $isCreatingNewContact) {
                NewContactView(didCreateNewContact: self.createNewContactHandler)
        }
    }


    private func createNewContactHandler(_ contact: Contact) {
        contactStore.contacts.append(contact)
        isCreatingNewContact = false
    }

}
Run Code Online (Sandbox Code Playgroud)

编辑 2:内存图截图

内存图

编辑 3:奇怪的是,用 VStack 替换 Form 视图可以消除泄漏。如果我使用列表视图而不是 VStack,泄漏又会回来。

zdr*_*kin 4

对于 iOS 14,使用 @StateObject 因此它只会初始化一次

永远不要这样做:@ObservedObject private var viewModel = ViewModel(),它会一直创建视图模型。您需要在应用程序开始时创建此对象,并将其传递到不会重新创建的位置,或者您可以在某些不更新的视图中创建,因为仪表板取决于您的应用程序结构。