无法暂停活动。 NPE

Posted

技术标签:

【中文标题】无法暂停活动。 NPE【英文标题】:Unable to pause activity. NPE 【发布时间】:2015-01-14 07:49:18 【问题描述】:

当我使用主页按钮或切换到横向时,我发现运行时异常:无法暂停活动。 NullPointerException。请帮助。

01-14 09:23:49.435:E/androidRuntime(331):致命异常:主 01-14 09:23:49.435: E/AndroidRuntime(331): java.lang.RuntimeException: 无法暂停活动 com.example.actiontest/com.example.actiontest.MainActivity: java.lang.NullPointerException 01-14 09:23:49.435: E/AndroidRuntime(331):在 android.app.ActivityThread.performPauseActivity(ActivityThread.java:2354) 01-14 09:23:49.435: E/AndroidRuntime(331): 在 android.app.ActivityThread.performPauseActivity(ActivityThread.java:2311) 01-14 09:23:49.435: E/AndroidRuntime(331): 在 android.app.ActivityThread.handlePauseActivity(ActivityThread.java:2291) 01-14 09:23:49.435: E/AndroidRuntime(331): 在 android.app.ActivityThread.access$1700(ActivityThread.java:117) 01-14 09:23:49.435:E/AndroidRuntime(331):在 android.app.ActivityThread$H.handleMessage(ActivityThread.java:938) 01-14 09:23:49.435: E/AndroidRuntime(331): 在 android.os.Handler.dispatchMessage(Handler.java:99) 01-14 09:23:49.435:E/AndroidRuntime(331):在 android.os.Looper.loop(Looper.java:123) 01-14 09:23:49.435: E/AndroidRuntime(331):在 android.app.ActivityThread.main(ActivityThread.java:3683) 01-14 09:23:49.435:E/AndroidRuntime(331):在 java.lang.reflect.Method.invokeNative(Native Method) 01-14 09:23:49.435:E/AndroidRuntime(331):在 java.lang.reflect.Method.invoke(Method.java:507) 01-14 09:23:49.435: E/AndroidRuntime(331):在 com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839) 01-14 09:23:49.435: E/AndroidRuntime(331): 在 com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597) 01-14 09:23:49.435:E/AndroidRuntime(331):在 dalvik.system.NativeStart.main(Native Method) 01-14 09:23:49.435: E/AndroidRuntime(331):引起:java.lang.NullPointerException 01-14 09:23:49.435:E/AndroidRuntime(331):在 android.support.v4.app.FragmentManagerImpl.saveFragmentBasicState(FragmentManager.java:1576) 01-14 09:23:49.435: E/AndroidRuntime(331): 在 android.support.v4.app.FragmentManagerImpl.saveAllState(FragmentManager.java:1617) 01-14 09:23:49.435: E/AndroidRuntime(331): 在 android.support.v4.app.FragmentActivity.onSaveInstanceState(FragmentActivity.java:481) 01-14 09:23:49.435: E/AndroidRuntime(331): 在 com.actionbarsherlock.app.SherlockFragmentActivity.onSaveInstanceState(SherlockFragmentActivity.java:126) 01-14 09:23:49.435: E/AndroidRuntime(331): 在 android.app.Activity.performSaveInstanceState(Activity.java:1037) 01-14 09:23:49.435: E/AndroidRuntime(331): 在 android.app.Instrumentation.callActivityOnSaveInstanceState(Instrumentation.java:1181) 01-14 09:23:49.435: E/AndroidRuntime(331): 在 android.app.ActivityThread.performPauseActivity(ActivityThread.java:2336) 01-14 09:23:49.435: E/AndroidRuntime(331): ... 12 更多

MainActivity.java

package com.example.actiontest;

import android.os.Bundle;
import android.support.v4.app.FragmentManager;
import android.support.v4.app.FragmentTransaction;
import android.support.v4.view.ViewPager;

import com.actionbarsherlock.app.ActionBar;
import com.actionbarsherlock.app.ActionBar.Tab;
import com.actionbarsherlock.app.SherlockFragmentActivity;

public class MainActivity extends    SherlockFragmentActivity 
                          implements ActionBar.TabListener 

    private ViewPager viewPager;
    private TabsPagerAdapter mAdapter;
    private ActionBar actionBar;
    // Tab titles
    private String[] tabs =  "Top Rated", "Games", "Movies" ;

    @Override
    public void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        // Initilization
        actionBar = getSupportActionBar();
        viewPager = (ViewPager) findViewById(R.id.pager);

        mAdapter = new TabsPagerAdapter(getSupportFragmentManager());

        viewPager.setAdapter(mAdapter);
        //actionBar.setHomeButtonEnabled(false);
        actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);  

        // Adding Tabs
        for (String tab_name : tabs) 
            actionBar.addTab(actionBar.newTab().setText(tab_name)
                    .setTabListener(this));
        

        /**
         * on swiping the viewpager make respective tab selected
         * */
        viewPager.setOnPageChangeListener(new ViewPager.OnPageChangeListener() 

            @Override
            public void onPageSelected(int position) 
                // on changing the page
                // make respected tab selected
                actionBar.setSelectedNavigationItem(position);
            

            @Override
            public void onPageScrolled(int arg0, float arg1, int arg2) 
            

            @Override
            public void onPageScrollStateChanged(int arg0) 
            
        );
    

    @Override
    public void onTabReselected(Tab tab, FragmentTransaction transaction) 
    

    @Override
    public void onTabSelected(Tab tab, FragmentTransaction transaction) 
        //mSelected.setText("Selected: " + tab.getText());
        viewPager.setCurrentItem(tab.getPosition());
    

    @Override
    public void onTabUnselected(Tab tab, FragmentTransaction transaction) 
    

TabsPagerAdapter.java

package com.example.actiontest;

import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentManager;
import android.support.v4.app.FragmentPagerAdapter;

public class TabsPagerAdapter extends FragmentPagerAdapter 

    public TabsPagerAdapter(FragmentManager fm) 
        super(fm);
    

    @Override
    public Fragment getItem(int index) 

        switch (index) 
        case 0:
            // Top Rated fragment activity
            return new TopRatedFragment();
        case 1:
             // Top Rated fragment activity
            return new TopRatedFragment();
        case 2:
            // Top Rated fragment activity
            return new TopRatedFragment();
        

        return null;
    

    @Override
    public int getCount() 
        // get item count - equal to number of tabs
        return 3;
    


TopRatedFragment.java

package com.example.actiontest;

import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import com.actionbarsherlock.app.SherlockFragment;

public class TopRatedFragment extends SherlockFragment 

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) 

        View rootView = inflater.inflate(R.layout.fragment_top_rated, container, false);

        return rootView;
    

【问题讨论】:

SherlockFragmentActivity 中似乎有错误。请发布此课程。 它是来自 ActionBarSherlock 库的类。 异常丢失,当我添加以下代码时: @Override public void onSaveInstanceState(Bundle savedInstanceState) 但程序无法正常工作。而且我有新的警告:ActivityManager:警告:Activity 未启动,其当前任务已被置于最前面 【参考方案1】:

切换到FragmentStatePagerAdapter 不是正确的解决方案。 Android docs 明确指出

FragmentStatePagerAdapter 在有 large 数字时更有用 页面,更像是一个列表视图。当页面不可见时 用户,他们的整个片段可能会被销毁,只保留 该片段的保存状态。这允许寻呼机保持很多 与每个访问的页面相关的内存更少 FragmentPagerAdapter 以可能更多开销为代价 在页面之间切换。

既然你只有三个页面,使用FragmentPagerAdapter是正确的。

您的问题很可能源于使用旧版本的 v4 支持库。要测试是否是这种情况,只需将您的 TopRatedFragment#onSaveInstanceState() 方法覆盖为

@Override
public void onSaveInstanceState(Bundle outState) 
    super.onSaveInstanceState(outState);
    setUserVisibleHint(true);

如果这解决了您的问题,那么您的 v4 支持库版本在 FragmentManagerImpl#saveFragmentBasicState() 处存在以下错误

if (!f.mUserVisibleHint) 
    /*
    if (result == null) 
        result = new Bundle(); MISSING!
    */
    // Only add this if it's not the default value
    result.putBoolean(FragmentManagerImpl.USER_VISIBLE_HINT_TAG, f.mUserVisibleHint);

result 未初始化但仍调用putBoolean() 时会引发NPE。

【讨论】:

感谢您的回答。我不明白。我将 FragmentStatePagerAdapter 重命名为 FragmentPagerAdapter 并且之前的异常消失了。发生了什么事? 因为,正如您在对问题的最后评论中提到的那样,您可能仍然会覆盖您的 onSaveInstanceState() 方法。它可能会跳过对super.onSaveInstanceState() 的调用,因此也跳过了这个错误。 但我删除了 onSaveInstanceState()。哦,我创建了新项目,问题又回来了。旧项目有问题。 将源代码附加到 Eclipse 中的 v4 支持库,然后在行号 FragmentManager.java:1576 之前放置断点后进行调试。这肯定会证实这个问题。 好吧,我添加了 onSaveInstanceState 并且代码运行良好。这是我的问题的正常解决方案吗?【参考方案2】:

让您的TabsPagerAdapter 扩展FragmentStatePagerAdapter 而不是FragmentPagerAdapter,您就可以开始了。

【讨论】:

不客气。只需将答案标记为已接受,让遇到相同问题的其他人知道它有效。

以上是关于无法暂停活动。 NPE的主要内容,如果未能解决你的问题,请参考以下文章

为啥完成Activity会抛出NPE?

android.content.Context.getString 中的 NPE 导致应用程序在启动时崩溃

发送从活动数据到片段:NPE [重复]

邻域保持嵌入(NPE)

jpa 结果集映射抛出 NPE

NPE 在 NodeList 上调用 getLength()