1

我正在尝试了解 SwiftUI 在内存管理方面是如何在内部工作的。我对此毫无疑问。

当我将 NavigationLink 添加到具有一些搜索功能并从云加载一些数据的第二个视图时。

现在当我回到根视图时,我的 observableObject 类仍在内存中。

有谁知道 SwiftUI 如何管理内存和释放对象?

这是我的实验的示例代码。

struct ContentView: View {
    var body: some View {
        NavigationView {
            DemoView(screenName: "Home")
                .navigationBarHidden(true)
        }
    }
}

struct DemoView:View {
    var screenName:String
    var body: some View {
        VStack{
             
            NavigationLink(destination: SecondView(viewModel:SecondViewModel())) {
                Text("Take Me To Second View")
            }
            
            Text(self.screenName)
        } 
    }
}


// Second View
class SecondViewModel:ObservableObject {
    @Published var search:String = ""
    @Published var user:[String] = []
    
    func fetchRecords() -> Void {
        DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 3) { [weak self] in
            self?.user = ["Hello", "There"]
        }
    }
    
}

struct SecondView:View {
    @ObservedObject var viewModel:SecondViewModel
    
    var body: some View {
        VStack {
            TextField("Search Here", text: $viewModel.search)
                .textFieldStyle(RoundedBorderTextFieldStyle())
                .padding()
            List(self.viewModel.user, id:\.self) { user in
                Text("User \(user)")
            }
        }.onAppear{
            self.viewModel.fetchRecords()
        }
    }
}

这就是我收到的内存图。

在此处输入图像描述

4

2 回答 2

1

好的,我可能不记得关于同一问题的其他类似帖子,但它的原因是因为你的SecondView,因为它是一个值,当你按下返回时,它仍然在NavigationView,只要直到另一个 NavigationLink 被激活。

所以你需要有不同的独立生命周期,SecondViewModel或者,如果保持原样,reset/cleanup为它添加一些,所以只剩下纯空对象,即

    }.onAppear{
        self.viewModel.fetchRecords()
    }.onDisappear {
        self.viewModel.cleanup()
    }
于 2020-10-19T13:26:55.080 回答
1

SwiftUI 中的对象生命周期与往常一样。当不再引用对象时,ARC 将释放对象。您可以添加deinit { print("deinit")} 到您的SecondViewModel并查看对象何时被释放。SecondViewModel是的,在您的情况下,每次评估主体时都会创建一个新对象DemoView,这可能不是您想要的。我猜你在视图层次结构之外初始化和存储SecondViewModel对象,并在DemoView.body.

于 2020-10-20T10:29:05.300 回答