Reputation: 1581
I also tried using the ApplicationContext
but it still leaks for some reason.
Found a similar post about the issue here AdActivity leak on AdMob (SDK 7.0) for Android but without an answer.
Also tried to set the adlistener and the ad to null in onDestroy() but without any luck and still leaks the activity.
My code called in onCreate()
private void refreshInterstitial(){
mInterstitialAd = new InterstitialAd(this);
mInterstitialAd.setAdUnitId("AD_ID");
mInterstitialAd.loadAd(new AdRequest.Builder().addTestDevice(AdRequest.DEVICE_ID_EMULATOR).addTestDevice("877BCC97E130A0DC62B2E5770D854496").build());
mInterstitialAd.setAdListener(new AdListener() {
@Override
public void onAdLoaded() {
mInterstitialAd.show();
}
@Override
public void onAdClosed() {
}
});
}
Leakcanary Leak Trace
┬───
│ GC Root: Global variable in native code
│
├─ mx instance
│ Leaking: UNKNOWN
│ ↓ mx.a
│ ~
├─ com.google.android.gms.ads.internal.webview.w instance
│ Leaking: UNKNOWN
│ mContext instance of com.google.android.gms.ads.internal.webview.ay, not wrapping activity
│ View#mParent is null
│ View#mAttachInfo is null (view detached)
│ View.mWindowAttachCount = 1
│ ↓ w.a
│ ~
├─ com.google.android.gms.ads.internal.webview.aa instance
│ Leaking: YES (View detached and has parent)
│ mContext instance of com.google.android.gms.ads.internal.webview.ay, not wrapping activity
│ View#mParent is set
│ View#mAttachInfo is null (view detached)
│ View.mWindowAttachCount = 1
│ ↓ aa.mListenerInfo
├─ android.view.View$ListenerInfo instance
│ Leaking: YES (aa↑ is leaking)
│ ↓ View$ListenerInfo.mOnClickListener
├─ com.google.android.gms.ads.nonagon.ad.webview.f instance
│ Leaking: YES (aa↑ is leaking)
│ ↓ f.a
├─ com.google.android.gms.ads.nonagon.ad.webview.l instance
│ Leaking: YES (aa↑ is leaking)
│ ↓ l.e
├─ com.google.android.gms.ads.nonagon.ad.event.bs instance
│ Leaking: YES (aa↑ is leaking)
│ ↓ bs.a
├─ java.util.HashMap instance
│ Leaking: YES (aa↑ is leaking)
│ ↓ HashMap.table
├─ java.util.HashMap$Node[] array
│ Leaking: YES (aa↑ is leaking)
│ ↓ HashMap$Node[].[1]
├─ java.util.HashMap$Node instance
│ Leaking: YES (aa↑ is leaking)
│ ↓ HashMap$Node.key
├─ com.google.android.gms.ads.nonagon.shim.k instance
│ Leaking: YES (aa↑ is leaking)
│ ↓ k.a
├─ com.google.android.gms.ads.internal.client.ae instance
│ Leaking: YES (aa↑ is leaking)
│ ↓ ae.a
├─ com.google.android.gms.internal.ads.zzuc instance
│ Leaking: YES (aa↑ is leaking)
│ ↓ zzuc.zzcbw
├─ com.test.Activity$1 instance
│ Leaking: YES (aa↑ is leaking)
│ Anonymous subclass of com.google.android.gms.ads.AdListener
│ ↓ EqualizerActivity$1.this$0
╰→ com.test.Activity instance
Leaking: YES (ObjectWatcher was watching this because Activity received Activity#onDestroy() callback and Activity#mDestroyed is true)
key = 40a1eb8e-c9e6-4062-b5f7-053e642e812f
watchDurationMillis = 5288
retainedDurationMillis = 258
Upvotes: 9
Views: 2918
Reputation: 31
I faced the same issue when testing my app with AdMob interstitial ads on an Android 5.1 device. It has 2 GB of RAM, and it threw an OutOfMemoryError
after I used my app for not more than 20 minutes. I followed AdMob's official guide to implement interstitial ads on my app. When I tested it, Android Profiler detected a memory leak after closing an activity with interstitial ads. I was using the latest version of the AdMob SDK (23.3.0).
Apparently, Google hasn't done a pretty good job in addressing memory leaks in their SDK. Fortunately, there's a fix, but it is not as simple as passing an applicationContext
to InterstitialAd.load()
within the activity class and setting interstitialAd
and its FullScreenContentCallback
to null
in onDestroy()
. Doing these is not enough to eliminate memory leaks.
Instead, you must ensure you load and show interstitial ads within a single instance: a singleton object. Singh has provided a good solution to this problem, but I will elaborate further and share what I did to eliminate memory leaks due to the AdMob SDK.
Here's a basic structure of the custom AdFactory
singleton (import statements are excluded):
interface AdFactoryInterface {
fun onInterstitialAdDismissedFullScreenContent()
fun onInterstitialAdFailedToShowFullScreenContent()
}
object AdFactory {
var interstitialAd: InterstitialAd? = null
private var mContextReference: WeakReference<Context>? = null
private var adFactoryInterface: AdFactoryInterface? = null
fun initialize(context: Context, afi: AdFactoryInterface) {
mContextReference = WeakReference(context)
adFactoryInterface = afi
}
fun loadInterstitialAds() {
val context = mContextReference?.get()
?: throw IllegalStateException("AdFactory has not been initialized")
if (adFactoryInterface == null) {
throw IllegalStateException("adFactoryInterface cannot be null")
}
val adRequest = AdRequest.Builder().build()
InterstitialAd.load(context, "your_ad_id", adRequest,
object: InterstitialAdLoadCallback() {
override fun onAdLoaded(p0: InterstitialAd) {
super.onAdLoaded(p0)
interstitialAd = p0
interstitialAd?.fullScreenContentCallback =
object: FullScreenContentCallback() {
override fun onAdDismissedFullScreenContent() {
super.onAdDismissedFullScreenContent()
interstitialAd = null
adFactoryInterface?.onInterstitialAdDismissedFullScreenContent()
}
override fun onAdFailedToShowFullScreenContent(p0: AdError) {
super.onAdFailedToShowFullScreenContent(p0)
interstitialAd = null
adFactoryInterface?.onInterstitialAdFailedToShowFullScreenContent()
}
}
}
override fun onAdFailedToLoad(p0: LoadAdError) {
super.onAdFailedToLoad(p0)
interstitialAd = null
}
}
)
}
fun showInterstitialAd(activity: Activity) {
if (interstitialAd != null) {
interstitialAd?.show(activity)
}
}
fun clear() {
interstitialAd?.fullScreenContentCallback = null
interstitialAd = null
mContextReference = null
adFactoryInterface = null
}
}
Here's a description of each method in the singleton object:
initialize(context: Context, afi: AdFactoryInterface)
: Initializes the singleton. You can call this method in your activity class as follows: AdFactory.initialize(applicationContext, this)
. Your activity class must implement AdFactoryInterface
.loadInterstitialAds()
: Loads an interstitial ad. This method must be called after initialize(context: Context, afi: AdFactoryInterface)
.showInterstitialAd(activity: Activity)
: Shows the interstitial ad.clear()
: Removes all references to the activity and the interstitial ad. This must be called in onDestroy()
, or the activity to be destroyed may leak memory.If you need to perform actions (e.g. starting another activity) after the user dismisses the interstitial ad or the ad fails to show, override onInterstitialAdDismissedFullScreenContent()
and onInterstitialAdFailedToShowFullScreenContent()
in your activity class. I have tried this approach and tested it on Android 5.1 and Android 12 platforms, and Android Profiler no longer reports any memory leaks. It's complicated, but that's the way to do it.
Upvotes: 0
Reputation: 1156
// Singleton pattern using Kotlin's object declaration.
object AdInterstitialCacheManager {
private var interstitialAd: InterstitialAd? = null
// Loads an interstitial ad using Google's AdManager with custom targeting.
private fun loadGoogleInterstitialAd(context: Context, adUnitId: String) {
if (interstitialAd != null) return
val pref = Preferences().apply { getPreference(context) }
val adRequest = AdManagerAdRequest.Builder()
.addCustomTargeting(Constants.Ads.CUSTOM_TARGETING_CATEGORY, listOf("बड़ी ख़बरें", "ListingPage"))
.addCustomTargeting(Constants.Ads.CUSTOM_TARGETING_PRICE_TAG, pref.getAdsPriceCategory().orEmpty())
.setPublisherProvidedId(pref.getPPID())
.build()
loadAd(context, adUnitId, adRequest)
}
// Common method to load an interstitial ad, with a fallback to Google's AdManager on failure.
private fun loadAd(context: Context, adUnitId: String, adRequest: AdRequest) {
InterstitialAd.load(context, adUnitId, adRequest, object : InterstitialAdLoadCallback() {
override fun onAdLoaded(ad: InterstitialAd) {
interstitialAd = ad
}
override fun onAdFailedToLoad(adError: LoadAdError) {
if (adRequest is AdManagerAdRequest) {
interstitialAd = null // If already using Google AdManager, do not retry.
} else {
loadGoogleInterstitialAd(context, adUnitId)
}
}
})
}
// Displays the loaded interstitial ad and clears it from cache after being shown.
fun showInterstitialAd(activity: Activity) {
interstitialAd?.fullScreenContentCallback = object : FullScreenContentCallback() {
override fun onAdDismissedFullScreenContent() {
interstitialAd = null
}
override fun onAdShowedFullScreenContent() {
interstitialAd = null
}
}
interstitialAd?.show(activity)
}
// Clears the interstitial ad from cache, allowing a new ad to be loaded.
fun clearAdViewCache() {
interstitialAd = null
}
}
Upvotes: 0
Reputation: 111
I was facing the same issue in interstitial ad and it is resolved by setting the FullScreenContentCallback to null in all of its overridden methods.
interstitial.setFullScreenContentCallback(new FullScreenContentCallback() {
@Override
public void onAdDismissedFullScreenContent() {
super.onAdDismissedFullScreenContent();
interstitial.setFullScreenContentCallback(null);}}
Upvotes: 1
Reputation: 21
This is working for me: (in ActivityMain()):
MobileAds.initialize(WeakReference(applicationContext).get()){}
in fragments:
adView= AdView(WeakReference(requireActivity().application).get())
LeakCanary shows:
==================================== HEAP ANALYSIS RESULT ==================================== 0 APPLICATION LEAKS.
Upvotes: 2
Reputation: 1156
According to InterstitialAd Docs:
A single InterstitialAd object can be used to request and display multiple interstitial ads over the course of an activity's lifespan, so you only need to construct it once.
After looking at your code again, I noticed you re-construct mInterstitialAd
every time refreshInterstitial()
method is called. But according to the docs above, you should only construct mInterstitialAd
once during onCreate()
.
In your case, the main cause of the memory leak: you still have an active listener (which is bound to Activity lifespan), yet you reconstruct a new InterstitialAd
instance with another listener.
So, the solution is to reuse InterstitialAd
instance and its listener without re-assigning. I suggest to simplify your refreshInterstitial()
method to this:
private void refreshInterstitial() {
mInterstitialAd.loadAd(new AdRequest.Builder().addTestDevice(AdRequest.DEVICE_ID_EMULATOR).addTestDevice("877BCC97E130A0DC62B2E5770D854496").build());
}
Then put mInterstitialAd
assignment to onCreate()
. This solution is similar to the one you can find here.
Upvotes: 3