【发布时间】:2019-11-17 05:12:48
【问题描述】:
我有一个ViewModel 类,它是一个ObservableObject,我在初始化其相应视图时对其进行了初始化。似乎如果我对视图内的ViewModel 有任何绑定,ViewModel 就会泄露。
例如,如果我在工作表中显示所述视图,则每次显示工作表时,都会分配一个新引用,并且在关闭工作表时不会释放它。引用计数随着我展示工作表的次数不断增加。
我是否遗漏了什么,或者@ObservedObject 属性包装器不应该以这种方式使用?
这是一个展示问题的简单示例。永远不会为 ViewModel 调用 deinit 函数
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")
}
}
}
}
}
编辑:添加了显示工作表的代码
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
}
}
编辑 2:内存图截图
编辑 3:奇怪的是,用 VStack 替换 Form 视图可以消除泄漏。如果我使用 List 视图而不是 VStack,泄漏又回来了。
【问题讨论】:
-
即使在表单中添加了表单,我也无法在提供的代码中获得超过一个 INIT 的输出。但这是一个 INIT,因为您在 NewContactView 初始化期间创建了 ViewModel 的实例。并且没有 DEINIT 因为视图被呈现。请更新问题的真实演示。
-
@Asperi 为了简洁起见,我没有包含显示工作表的代码。我很抱歉。请查看我的编辑。