Hi Jonathan

Thank you for your detailed response. I have made some progress since I
posted this message and have some new questions about some of the things I
have discovered.

1. Calling the GC did help with things however because the GC takes a
millisecond or whatever to execute the UI suffers and gets a stutter
effect. Is there a way to avoid this?

In terms of memory management and making requests based on the individual
activities my biggest concern is really the user flooding the app with
network requests. Something else that has an influence on this in my app is
that I am also loading images within my activities and caching them. I
suspect that this was a cause for getting out of memory.

I have come up with a solution based on the ImageLoader in the MWC app. I
have managed to hack together a version of this ImageLoader that unifies
queuing of the image requests and the data requests. This seemed good for
managing the memory but it now has some other issues. Ideally what I think
I need is:
 - Two queues one for images and one for the data requests.
 - The worker for processing the queues should limit the number of requests
across the queues.
 - The data queue should take higher priority to the image queue. So the
data queue is always processed before the image queue.
 - If the activity is destroyed before loading all of the data/images it
should dequeue any outstanding requests. Any ideas how I might dequeue if I
am using something like the ImageLoader from the MWC app?

What are your thoughts on this design for managing the requests?

I still need to make individual requests for each activity as the data
retrieved from each request is specific to the individual activity and it
can only be retrieved on the per activity basis. I need to have realtime up
to date info for each activity.

If I initiate the all of the requests from the Application class as you
suggest and then raise the events to update the activities from the
application class when raising the event should it always be done on the UI
thread using SynchronizationContext.Post? Or is it ok to raise it on
another thread, then in the event handler in the Activity call
RunOnUIThread?

Thanks again
Liam


On Mon, Jul 2, 2012 at 8:13 AM, Jonathan Pryor <j...@xamarin.com> wrote:

> General questions:
>
> 1. Does calling GC.Collect() anywhere help with your memory use? It could
> just be that the GC's allocation limits aren't entirely appropriate, and
> calling GC.Collect() will help:
>
>         http://docs.xamarin.com/android/advanced_topics/garbage_collection
>
> 2. Are there lots of peer instances that can be more quickly collected?
>
>
> http://docs.xamarin.com/android/advanced_topics/garbage_collection#Cross-VM_Object_Collections
>
> http://docs.xamarin.com/android/advanced_topics/garbage_collection#Helping_the_GC
>         http://docs.xamarin.com/android/advanced_topics/diagnostics
>
> http://docs.xamarin.com/android/advanced_topics/diagnostics#Global_Reference_Messages
>
> 2(b): How is your gref count? You can enable gref logging with:
>
>         adb shell setprop debug.mono.log gref
>
> then every monodroid-gref message will contain the current gref count.
>
> On Jun 28, 2012, at 10:57 PM, Liam Houlahan wrote:
> > The scenario that I have for my app is. On first start up each day the
> app will do an initial large download. By large I mean a couple of hundred
> kbs.
> > After the large download when the user accesses parent activities for
> the first time they will download all of the child data.
>
> How big is the child data download? How much memory does that
> approximately take, post-download? (You can use GC.GetTotalMemory(false)
> before your web request and after it returns to see how much memory has
> been used, approximately.)
>
> > To update the UI for the large download I am using event handlers in my
> activities assigned to a static event in my service class. For the other
> smaller requests I am using callbacks to the Activities.
>
> You'd need to provide more detail regarding how _exactly_ you do this;
> done wrong, this could mean that no Activities can ever be collected (as
> the static event is referring to them), so you'd have a steadily increasing
> gref count and increasing memory use.
>
> An alternative approach is to instead use
> Android.App.Application.SynchronizationContext.Post(...) to invoke
> something on the UI thread.
>
> > Some more specific things that I am wandering about are:
> >
> >  - What is the best way to update the UI from async webrequest methods?
> i.e. using action callbacks or events.
>
> Most direct would be Activity.RunOnUiThread(), but if that isn't available
> then using Application.SynchronizationContext will work, and in >= 4.2.3
> then System.Threading.SynchronizationContext will actually work (wow!), but
> note that SynchronizationContext.Current is [ThreadStatic], so you'd need
> to either pass the SynchronizationContext to the worker thread (via a
> closure?) or use TAP (which iirc should implicitly pass
> SynchronizationContext.Current...on .NET, but I haven't tested on Mono;
> test accordingly).
>
> >  - Should database access and async webrequests be initiated from
> threads separate to the UI thread or is it best to just call them from the
> UI thread? If it is better to use separate threads where should these
> threads be started? e.g. in the activity.
>
> Async web requests should be fine. Database access _may_ be fine; it
> depends on the performance characteristics of SQLite and your data set. If
> you're confident that it won't block the UI thread for long periods of
> time, it's likely safe, but if you're not confident of that you should move
> the writes onto another thread.
>
> >  - For manually calling the Garbage Collector, should the GC be called
> after each web request? If it should which part of the app should the GC be
> called in and when? i.e in the service class after the download or in the
> activity after displaying the updated results?
>
> Where (1) consistently works in reducing memory use, and (2) is
> convenient. ;-)
>
> There's a bit of trial and error here...
>
> >  - If a user opens an activity that starts a webrequest then goes back
> opens another activity starting more requests how should the app deal with
> this? Should the requests from the first activity be cancelled before
> starting the next etc. If I am using a callback to the activity for an
> async webrequest and the activity is destroyed before the webrequest
> finishes what is the result of this?
>
> Again, it depends on your code structure. The most obvious way would
> result in the Async web request referencing the launching Activity, which
> would thus prevent the GC from collecting the Activity instance, so when
> the web request returns it'll act on a ("dead") Activity instance. This has
> the added downside that if the user starts, "exits", then starts your app,
> you may launch the same web request twice, downloading more data than is
> necessary.
>
> Instead, what I would suggest is that your web requests know ~nothing of
> the UI. Instead, you should e.g. subclass Android.App.Application [0], and
> have a data cache in the Application instance. The Web request would update
> this data cache.
>
> Next, Activity.OnCreate() can register with an Application event, and
> Activity.OnDestroy() can unregister the Activity with the Application.
> (Both are important, to allow the Activity instances to be collectable!)
> Next, updates to the Application data cache can raise the Application
> event, allowing Activities to be notified that data is available/has
> changed.
>
> The result of such an architecture is that, with some logic, you can
> ensure that only one Web request is generated within a short time period
> (instead of one/Activity), all data is (reasonably) up-to-date, and all
> instances can be collected when they are no longer used.
>
>  - Jon
>
> [0]
> https://github.com/xamarin/monodroid-samples/blob/master/SanityTests/Hello.cs#L92
>
> _______________________________________________
> Monodroid mailing list
> Monodroid@lists.ximian.com
>
> UNSUBSCRIBE INFORMATION:
> http://lists.ximian.com/mailman/listinfo/monodroid
>
_______________________________________________
Monodroid mailing list
Monodroid@lists.ximian.com

UNSUBSCRIBE INFORMATION:
http://lists.ximian.com/mailman/listinfo/monodroid

Reply via email to