Flex – Event Handlers and Weak References

Did you know defining an event handler on an object in Flex can prevent it or other objects from being garbage collected even after all explicit references to the objects have been removed? If not, then keep reading.

Introduction

Previously, I discussed techniques for freeing up memory on video and images in Flex, but I neglected to include a discussion of event handlers. With the release of ActionScript 3, Adobe added the notion of strong references and weak references. Strong references are those references that, if present, will prevent the object from being garbage collected. The presence of only weak references, on the other hand, will allow an object to be garbage collected. When most developers consider garbage collection, they often consider only strong references, those references they explicitly know about. The age-old wisdom, that if you remove all active (strong) references to an object, it will become eligible for garbage collection, overlooks the fact that the object may have strong references from other objects, created via event handler registration.

Event Handlers are Strong References (by default)

Somewhat surprisingly, and perhaps in a move to preserve older source code, the commonly used method addEventListener() creates a strong reference to the event handler by default. That means even after removing all explicit references to an object, it may never be eligible for garbage collection because other objects have subscribed as listeners to the object. If you have a process that repeatedly creates such objects, you have an uncontrolled memory leak that will eventually crash your program. As an example, consider a screen saver application that displays a new Image or Video on the screen every 30 seconds by creating new display objects and discarding the existing ones. Without considering the event handlers on this application, the memory usage could grow without bound, even if you take proper steps to discard the display objects.

From the Adobe Flex 3 documentation:

One important consequence of this parameter involves working with display objects’ events. Normally, you might expect a display object to be removed from memory when it is removed from the display list. However, if other objects have subscribed as listeners to that display object, with the useWeakReference parameter set to false (the default), the display object will continue to exist in Flash Player’s or AIR’s memory even though it no longer appears on the screen.

Solution #1: Proper deconstructor

The most obvious solution to prevent event handlers from holding objects in memory is to call removeEventListener() on the original object for each event handler on it. The problem is Flex does not provide a listAllEventListeners() method nor a removeAllEventListeners() method in the Event Dispatcher API [Feature Request!], so unless you remembered to save a reference to the event handler when you created it, you’re not going to have much luck removing it. This solution requires you to manage a list of event handlers and their associated types, so each can be discarded at a later time. Also, remember to remove the reference to the event handler in the list, such that the act of maintaining a list of references does not prevent garbage collection.

Solution #2: Declare a weak reference

Maintaining a list of event handlers for every object is a bit cumbersome, so luckily Adobe has added a better solution: declare the reference weak when the event is added to the object. In ActionScript 3, Adobe added an overloaded version of addEventListener() that takes 5 parameters. By setting the 5th parameter (useWeakReference) to be true, the event handler and its associated objects will become eligible for garbage collection if the weak reference event handlers are the only ones left.

For example, let’s say you defined a timer to run for 10 seconds and then stop:

var myTimer:Timer = new Timer(10,0);
myTimer.addEventListener(TimerEvent.TIMER,myTimerHandler);

As the code is written, objects referenced by myTimerHandler cannot be garbage collected, even if the timer was never started. Alternatively, if you add the overloaded parameters to the method ,false,0,true, objects referenced by myTimerHandler can be garbage collected:

var myTimer:Timer = new Timer(10,0);
myTimer.addEventListener(TimerEvent.TIMER,myTimerHandler,false,0,true);

Keep in mind, there are situations when it makes sense to define event handlers using strong references even on temporary objects. Just remember to explicitly remove the event handler if you want to reclaim the allocated memory.

Some developers, though, have made cases that Solution #1 is the only correct solution, and that using weak references is not sufficient for proper memory management.

Conclusion: Refactor your code with overloaded parameters

My suggestion to the reader is search through their code for all occurrences of addEventListener. For each occurrence, decide whether the event handler should live for the life of the application and never be garbage collected, or can be removed before the application completes. If it is the former, and the event handler should live indefinitely for the application, then do nothing. If it is the latter, and the event handler may be discarded at some point, add the parameters ,false,0,true to the event handler creation. This will preserve the existing functionality but allow it to be garbage collected. Keep in mind, though, that the event handler will not be garbage collected if you maintain strong or explicit references to it elsewhere. In the previous example, if you defined myTimerHandler to be a class-level member function, then the event handler would not be eligible for garbage collection with or without the change to the event handler registration.

Error’d: European Edition

I traveled to the UK earlier this month and was greeted by an error screen a day for the first three days of my trip.

Day 1: London Subway Station
Nothing like a 5-foot-tall boot screen to make you feel at home in a foreign country.




Day 2: Cardiff Train Station
Found a Windows BSOD while waiting for the train. And they say Linux dominates the European market!




Day 3: London Bus
Found half-way up the stairs of one of London’s famous double decker public buses.




Bonus: Bed Bath & Beyond
So as not to forget American-based errors, here’s one I came across this morning (live for now). Those without time machines are at a severe disadvantage for this Bed Bath & Beyond rebate offer!


By the way, it was an excellent trip, with or without the BSOD’s following me around Europe!

Review: Marware Accent for iPhone 4

Accent 4 After waiting two months for the Marware Accent Case for the iPhone 4, I was excited when it finally reach my doorstep. I prefer “flip” cases to shell cases for my smart phones since they protect the screen when placed in a pants pocket or a woman’s purse. On top of that, since they are often made out of leather, they provide a nice cushion should the phone be dropped from a few feet. Unfortunately, the Marware Accent 4 for the iPhone 4 suffers a serious defect with regard to the camera flash and was not worth the wait. Read on for the rest of the review.

The Good

Overall, the Accent 4 case design is quite nice. Unlike older iPhone 3 cases that most iPhone 4 owners have been using for the last 2 months, the Accent 4 fits the iPhone 4 perfectly. The mute button and power adapter are fully exposed. The speakers on the bottom are partially exposed with thin mesh netting over them. The volume buttons are cover by a thin sheet of leather with indentations for each button. Finally, the death grip area is completely covered so no need to purchase an iPhone 4 band aid.

The Bad

During the two month wait for the case, I called Marware support a few times, hoping to get a solid delivery date. It was scheduled to be sent out mid July but the Marware customer service representative informed me that there was a problem with the hole for the camera and flash being too small. They said it would be delayed another few weeks while they redesign it. I’m sorry to say, though, that they did not resolve the issue with the camera as the test below demonstrates:

I tried a number of different variations in light rooms, dark rooms, and pitch black rooms, but the results were all the same. As shown in the picture, some of the reflection was off the top of the hole so I tried readjusting the position of the iPhone inside the case but even without the direct bounce, there is always an awful glare/haze covering the picture. All of the pictures I took with the iPhone 4 inside the Accent case came out with a glare similar to the one seen above.

iPhone - Camera/Flash hole

Examining the back, the hole is large enough for the camera and flash to be seen without obstruction but not large enough for the flash to function properly. The flash functions by spreading light outward, and the width of the leather is too large so as to catch the light from the flash, and reflect it back into the lens.

Contacting Marware Support

Having ordered my case through Marware directly, I contacted Marware support about this defect. Despite the numerous revisions, Marware checked with the shipping manager and assured me that I did indeed have the finished product. Unfortunately, as the pictures above show, this product is still quite defective. I have to assume someone just said “close enough” and shipped the product as is. The support representative indicated Marware was not planning another revision at this time, although offered to refund my money. After waiting two months for a case, I’m reluctant to wait another two more.

Marware Support Followup

UPDATE: Marware’s official response on the Accent 4 flash defect:

Hello Scott,

As per our phone conversation today, the case you received is the most recent one. There are no plans for re adjusting the camera hole again at this point. The only suggestion provided to me is that when you use the flash, to push the iPhone up out of the case a little. If you are not comfortable doing that we can either exchange for another case or give you a refund.

I look forward to hearing back from you.

Regards,
Name omitted
Marware Customer Support Team

Despite acknowledging the issue, Marware does not seem to be interested in fixing it. At this time, I cannot recommend purchasing this case as it is clearly defective.