Error: WebView.destroy() called while still attached 的解决办法

本文详细介绍了在Android开发中遇到的Webview页面闪退问题及其解决方法,包括错误解释、常见错误代码示例及正确的处理步骤。通过实例演示,帮助开发者避免常见的Webview使用陷阱。

今天在改Webview页面的时候出现了闪退。看Log发现是这个问题Error: WebView.destroy() called while still attached。


谷歌了一下。发现只有stackoverflow解释这个问题比较清楚,另外鄙视下百度,啥都搜不到。不过一些问题能用中文描述的,还是百度相关内容比较多。


现在总结下这个问题的解决办法。先大概解释下这个错误,这个错误从字面意思来说是当你结束webview的时候,Webview还依附在父控件下,使用解决这个问题

就是在WebView.destroy()前要解除他们之间的依附关系。一般会碰上这个问题应该是这样对webview进行了操作:

protected void onDestroy()
{
    if (adView != null)
    {
        adView.destroy();
    }
}


附上一段详细的正确代码:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:id="@+id/webviewRelativeLayout"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent">

<WebView
    android:id="@+id/webView1"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:layout_below="@+id/headerAlarmsWebViewTxt"
    android:layout_marginBottom="0dip"
    android:hapticFeedbackEnabled="true"
    android:overScrollMode="never"
    android:scrollbarAlwaysDrawVerticalTrack="false"
    android:scrollbars="none" />

 </RelativeLayout>
Then you assign it to an instance variable e.g. :

_layout = (RelativeLayout) findViewById(R.id.webviewRelativeLayout);
webView = (WebView) findViewById(R.id.webView1);
and on Destroy do something like this:

@Override
protected void onDestroy() {
    super.onDestroy();
    _layout.removeView(webView);
    webView.removeAllViews();
    webView.destroy();
}


一般看了这段代码就能明白问题所在了。


行 39460: 06-12 16:37:16.163 W/dalvikvm( 3220): threadid=1: thread exiting with uncaught exception (group=0x41d31bc0) 行 39462: 06-12 16:37:16.164 E/CrashHandler( 3220): 线程名称:Thread[main,5,main] / 异常:java.lang.NullPointerException 行 39464: 06-12 16:37:16.164 W/System.err( 3220): java.lang.NullPointerException 行 39466: 06-12 16:37:16.166 W/System.err( 3220): at com.android.org.chromium.base.PathUtils.getDataDirectory(PathUtils.java:55) 行 39468: 06-12 16:37:16.167 W/System.err( 3220): at com.android.org.chromium.content.browser.ResourceExtractor.getOutputDirFromContext(ResourceExtractor.java:308) 行 39470: 06-12 16:37:16.167 W/System.err( 3220): at com.android.org.chromium.content.browser.ResourceExtractor.<init>(ResourceExtractor.java:266) 行 39472: 06-12 16:37:16.168 W/System.err( 3220): at com.android.org.chromium.content.browser.ResourceExtractor.get(ResourceExtractor.java:232) 行 39474: 06-12 16:37:16.169 W/System.err( 3220): at com.android.org.chromium.content.browser.AndroidBrowserProcess.init(AndroidBrowserProcess.java:62) 行 39476: 06-12 16:37:16.169 W/System.err( 3220): at com.android.org.chromium.android_webview.AwBrowserProcess$1.run(AwBrowserProcess.java:51) 行 39478: 06-12 16:37:16.169 W/System.err( 3220): at com.android.org.chromium.base.ThreadUtils.runOnUiThreadBlocking(ThreadUtils.java:56) 行 39480: 06-12 16:37:16.169 W/System.err( 3220): at com.android.org.chromium.android_webview.AwBrowserProcess.start(AwBrowserProcess.java:46) 行 39482: 06-12 16:37:16.170 W/System.err( 3220): at com.android.webview.chromium.WebViewChromiumFactoryProvider.startChromiumLocked(WebViewChromiumFactoryProvider.java:190) 行 39484: 06-12 16:37:16.170 W/System.err( 3220): at com.android.webview.chromium.WebViewChromiumFactoryProvider.ensureChromiumStartedLocked(WebViewChromiumFactoryProvider.java:108) 行 39486: 06-12 16:37:16.171 W/System.err( 3220): at com.android.webview.chromium.WebViewChromiumFactoryProvider.startYourEngines(WebViewChromiumFactoryProvider.java:265) 行 39488: 06-12 16:37:16.171 W/System.err( 3220): at com.android.webview.chromium.WebViewChromium.init(WebViewChromium.java:218) 行 39490: 06-12 16:37:16.171 W/System.err( 3220): at android.webkit.WebView.<init>(WebView.java:534) 行 39492: 06-12 16:37:16.172 W/System.err( 3220): at android.webkit.WebView.<init>(WebView.java:501) 行 39494: 06-12 16:37:16.172 W/System.err( 3220): at android.webkit.WebView.<init>(WebView.java:481) 行 39498: 06-12 16:37:16.173 W/System.err( 3220): at android.webkit.WebView.<init>(WebView.java:470) 行 39500: 06-12 16:37:16.174 W/System.err( 3220): at android.webkit.WebView.<init>(WebView.java:460)
最新发布
06-20
### Android WebView NullPointerException 问题分析及解决方案 在 Android 开发中,`WebView` 是一个常用的组件,用于加载和显示网页内容。然而,在某些情况下,可能会遇到 `NullPointerException`,尤其是在与 Chromium 相关的路径处理方法(如 `com.android.org.chromium.base.PathUtils.getDataDirectory()`)发生异常时。 #### 1. 异常原因分析 `NullPointerException` 的出现通常与以下情况相关: - **上下文无效**:如果 `WebView` 初始化时使用的 `Context` 对象无效或已被销毁,可能会导致此类异常[^1]。 - **文件路径问题**:Chromium 在初始化时需要访问特定的文件目录。如果设备存储权限不足或路径不可用,可能导致异常[^2]。 - **多线程操作**:如果在非主线程中初始化或操作 `WebView`,可能会引发线程安全问题,从而导致 `NullPointerException`[^3]。 #### 2. 解决方案 ##### (1) 确保上下文有效 确保在初始化 `WebView` 时使用的是正确的 `Context` 对象。例如,应使用 `Activity` 或 `Application` 的上下文,而不是短期有效的上下文对象。代码示例如下: ```java // 使用 Application Context 初始化 WebView WebView webView = new WebView(getApplicationContext()); ``` ##### (2) 检查存储权限 从 Android 6.0(API 级别 23)开始,应用需要动态请求权限。如果未授予存储权限,可能导致路径访问失败。确保在运行时检查并请求必要的权限: ```java if (ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) { ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, REQUEST_CODE); } ``` ##### (3) 避免在非主线程中初始化 WebView `WebView` 的初始化必须在主线程中完成。如果需要在后台线程中执行任务,请确保将 `WebView` 的初始化移至主线程: ```java runOnUiThread(new Runnable() { @Override public void run() { WebView webView = new WebView(MainActivity.this); webView.loadUrl("https://www.example.com"); } }); ``` ##### (4) 配置 WebView 的缓存路径 可以通过设置自定义缓存路径来避免与系统默认路径相关的异常。例如: ```java String appCachePath = getApplicationContext().getCacheDir().getAbsolutePath(); webView.getSettings().setAppCachePath(appCachePath); webView.getSettings().setAllowFileAccess(true); webView.getSettings().setAppCacheEnabled(true); ``` ##### (5) 更新 WebView 版本 如果问题仍然存在,可能是由于系统自带的 WebView 版本过旧。建议提示用户更新 WebView 组件,或考虑使用第三方库(如 [Crosswalk](https://crosswalk-project.org/))替代系统 WebView[^4]。 #### 3. 示例代码 以下是一个完整的 `WebView` 初始化示例,包含上述解决方案: ```java public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // 设置 WebView 缓存路径 String appCachePath = getApplicationContext().getCacheDir().getAbsolutePath(); WebView webView = new WebView(getApplicationContext()); WebSettings webSettings = webView.getSettings(); webSettings.setJavaScriptEnabled(true); webSettings.setAppCachePath(appCachePath); webSettings.setAllowFileAccess(true); webSettings.setAppCacheEnabled(true); // 加载 URL webView.loadUrl("https://www.example.com"); // 将 WebView 添加到布局 setContentView(webView); } } ``` ###
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值