我有一个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,泄漏又会回来。
对于 iOS 14,使用 @StateObject 因此它只会初始化一次
永远不要这样做:@ObservedObject private var viewModel = ViewModel(),它会一直创建视图模型。您需要在应用程序开始时创建此对象,并将其传递到不会重新创建的位置,或者您可以在某些不更新的视图中创建,因为仪表板取决于您的应用程序结构。