-
-
Notifications
You must be signed in to change notification settings - Fork 532
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Potential Memory Issue when using any processors such as Resize #776
Comments
Just following up on this with more information, we've discovered this only happens with |
Hey @alexjameslittle, Could you please share the pipeline configuration that you are using? What is the usage pattern? Is the app loading large number of images with high resolution?
Resize and decompression use a lot of memory because both of these operations require bitmapping the image, so high memory usage is to be expected, especially if displaying large images.
Would you mind sharing some more details about how you are measuring the memory usage? There a distinction between persistent and transient memory. The only persistent memory the framework uses is managed by the internal memory cache that has strict limits and clears automatically on memory warnings. As long as the app periodically clears used memory, a high memory usage should not going to be a concern. It's nearly impossible to get a modern app to crash because of the high memory usage. It only happens if you hit a hard memory limit. You can find some more information about performance and caching in the documentation. It can help you guide how to adjust the configuration or the usage patterns to best fit your app's needs. |
Hey @kean, Of course, our pipeline is setup like this: let pipeline = ImagePipeline {
let dataLoader: DataLoader = {
let config = URLSessionConfiguration.default
config.urlCache = nil
return DataLoader(configuration: config)
}()
$0.dataLoader = dataLoader
let path = FileManager
.default
.containerURL(forSecurityApplicationGroupIdentifier: config.appGroupIdentifier)!
.appendingPathComponent(Self.cacheFolderName)
let cache = try? DataCache(path: path, filenameGenerator: Self.filenameGenerator)
cache?.sizeLimit = 1024 * 1024 * (5 * 1024) // 5GB
$0.dataCache = cache
$0.isStoringPreviewsInMemoryCache = true
$0.isProgressiveDecodingEnabled = true
$0.dataCachePolicy = .storeAll
} We've been using Nuke for almost 2 years without using any of the processors and we've not seen memory usage grow like it has until we started trialling the processors for resizing and blurring, we do expect an increase in memory when doing these operations, however it seems there is an usually large growth in memory that just doesn't happen if we don't ask the pipeline to store all images, we believe it is most likely a memory leak. We're seeing the memory usage of the app quickly going over 1gb after a scrolling through 20-30 resized images (jpegs at ~200kb, 1080x1440...resized to approx). If we use the If there is anything else I can do to help debug this, please let me know! |
There are multiple things that will contribute to high memory usage:
If you combine these four things together, and the app does in fact load progressive JPEGs, I would expect an extremely high memory usage. It isn't necessarily a problem if that's what you are going for in terms of the UI/UX. The high memory usage is not the indication that the app will be terminated by the system.
Resizing is usually a good thing. It will cause a quick spike in CPU and memory usage, but will ultimately result in a smaller footprint.
It seems unlikely, but I would not exclude is as a possibility. I don't think it's a popular option. I personally always go for storing the originals. I don't have a lot of practical experience with encoding and I'm not sure what performance characteristics to expect from it. I also wouldn't recommend using
I would appreciate if you could run the Leaks instruments and see if there are any. |
Yeah, we're aware we have a much higher ceiling to go to with the memory usage generally before the app gets terminated, however in this instance the memory usage is constantly rising to a point where which is causing visible performance issues to the user, if you continue to scroll through even more images the app then does terminates itself due to excessive memory usage. Whilst we have set the We chose We did check the leak instruments but there is no visible leaks in instruments, we will revert to |
It does indicate that the might be a leak somewhere.
Yes, that would be ideal. |
Where is best to send you the project as we'd prefer not to upload it publicly as it's for the company I work for. |
If it has any closed source code, I would not recommend sharing it. I should be able to do an investigation based on the details you provided. |
We created an example app without any closed source just now that demonstrates the issue. Feel free to use it to help debug. |
Thanks @alexjameslittle. I've tested the demo project and haven't identified anything beyond what I would expect, considering the size of the images. On my device, the memory footprint never grows beyond ~500 MB. When I background the app, it frees almost all of the memory, confirming that it's not persistent and that there are no leaks. I believe the main cause of the performance issues is the sheer size of the images (2000×2000px and higher) and the target size of the resize processors (1500×1500px on @3 devices). You can get a ~5x improvement in memory and CPU usage by reducing the target size used for I would also recommend loading images of a smaller size, if possible, to reduce the networking footprint. Image encoding does use a lot of CPU and memory, so if you disable it, you get an even better result. I haven't found any opportunities for optimizing it – it's just expensive. I did identify a regression in one of the recent Nuke versions that broke one of the optimizations that would skip redundant image decompression and prepared a fix #777. This issue is not as impactful as I would expect it, but the fix will ship in the upcoming version. I'm also planning to remove the |
hey @kean, I can also confirm the memory does get cleared on backgrounding, however for now we're going to go with your original suggestion and not store all images, only storing original image data. In terms of networking footprint & resizing, that demo application isn't an exact replica of what we have internally, our own pipeline is much more optimised but we will keep in mind all of your suggestions going forward. Good to know about the decompression fix & the removal of points based for resizing, I will keep an eye out for these updates! Thank you for investigating and the detailed response! |
We're seeing issues when fetching images for the first time and resizing them with LazyImageView, the memory usage continues to rise indefinitely. However we store those resized images on disk using nuke and on next app launch when nuke uses the already resized images, the memory usage is normal.
If we don't use any processors we don't see any issue with memory.
Edit: This seems to also be reproducible without processors if you use
skipDecompress
in the image request optionsThe text was updated successfully, but these errors were encountered: