Android应用程序长时间闲置后崩溃

Android应用程序长时间闲置后崩溃,android,android-layout,android-activity,crash,android-resources,Android,Android Layout,Android Activity,Crash,Android Resources,我现在正面临着一个非常奇怪的问题,我的手机。如果我使用我正在开发的应用程序,我不会有任何问题,直到我长时间停止使用它(当这种情况发生时,我相信应用程序的实例会关闭)。为了模拟这种行为,我下载了一个“填充内存”的应用程序。我不理解的是错误:每次我在填充RAM后尝试使用应用程序时,我都会遇到ResourceNotFound异常,至少对我来说没有意义。从跟踪中可以看出,错误出现在以下行中: mView = inflater.inflate(mLayout,container, false); 以下是

我现在正面临着一个非常奇怪的问题,我的手机。如果我使用我正在开发的应用程序,我不会有任何问题,直到我长时间停止使用它(当这种情况发生时,我相信应用程序的实例会关闭)。为了模拟这种行为,我下载了一个“填充内存”的应用程序。我不理解的是错误:每次我在填充RAM后尝试使用应用程序时,我都会遇到ResourceNotFound异常,至少对我来说没有意义。从跟踪中可以看出,错误出现在以下行中:

mView = inflater.inflate(mLayout,container, false);
以下是跟踪:

11-25 08:22:09.496: E/AndroidRuntime(6386): FATAL EXCEPTION: main
11-25 08:22:09.496: E/AndroidRuntime(6386): Process: com.papinotas, PID: 6386
11-25 08:22:09.496: E/AndroidRuntime(6386): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.papinotas/com.papinotas.views.activities.TeacherMainActivity}: android.content.res.Resources$NotFoundException: Resource ID #0x0
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2305)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2363)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.app.ActivityThread.access$900(ActivityThread.java:161)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1265)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.os.Handler.dispatchMessage(Handler.java:102)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.os.Looper.loop(Looper.java:157)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.app.ActivityThread.main(ActivityThread.java:5356)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at java.lang.reflect.Method.invokeNative(Native Method)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at java.lang.reflect.Method.invoke(Method.java:515)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1265)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1081)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at dalvik.system.NativeStart.main(Native Method)
11-25 08:22:09.496: E/AndroidRuntime(6386): Caused by: android.content.res.Resources$NotFoundException: Resource ID #0x0
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.content.res.Resources.getValue(Resources.java:2036)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.content.res.Resources.loadXmlResourceParser(Resources.java:3265)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.content.res.Resources.getLayout(Resources.java:1852)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.view.LayoutInflater.inflate(LayoutInflater.java:396)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at com.androidinit.base.PlaceHolderFragment.onCreateView(PlaceHolderFragment.java:29)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.app.Fragment.performCreateView(Fragment.java:1700)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.app.FragmentManagerImpl.moveToState(FragmentManager.java:890)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.app.FragmentManagerImpl.moveToState(FragmentManager.java:1062)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.app.FragmentManagerImpl.moveToState(FragmentManager.java:1044)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.app.FragmentManagerImpl.dispatchActivityCreated(FragmentManager.java:1853)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.app.Activity.performCreate(Activity.java:5429)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1105)
11-25 08:22:09.496: E/AndroidRuntime(6386):     at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2269)
11-25 08:22:09.496: E/AndroidRuntime(6386):     ... 11 more
11-25 08:22:09.506: W/ActivityManager(847):   Force finishing activity com.papinotas/.views.activities.TeacherMainActivity
11-25 08:22:09.516: W/ContextImpl(847): Calling a method in the system process without a qualified user: android.app.ContextImpl.sendBroadcast:1479 com.android.server.am.ActivityStack.startPausingLocked:1006 com.android.server.am.ActivityStack.finishActivityLocked:3162 com.android.server.am.ActivityStack.finishTopRunningActivityLocked:3005 com.android.server.am.ActivityStackSupervisor.finishTopRunningActivityLocked:3142 
11-25 08:22:09.526: D/CrashAnrDetector(847): processName: com.papinotas
11-25 08:22:09.526: D/CrashAnrDetector(847): broadcastEvent : com.papinotas data_app_crash
11-25 08:22:09.526: V/SmartFaceService - 3rd party pause(847): onReceive [android.intent.action.ACTIVITY_STATE/com.papinotas/pause]
11-25 08:22:09.526: W/ContextImpl(847): Calling a method in the system process without a qualified user: android.app.ContextImpl.sendBroadcast:1479 com.android.server.analytics.data.collection.application.CrashAnrDetector.broadcastEvent:296 com.android.server.analytics.data.collection.application.CrashAnrDetector.processDropBoxEntry:254 com.android.server.analytics.data.collection.application.CrashAnrDetector.access$100:60 com.android.server.analytics.data.collection.application.CrashAnrDetector$1.onReceive:102 
以下是跟踪的完整类:

package com.androidinit.base;

import android.app.Fragment;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;

    public class PlaceHolderFragment extends Fragment {

    private int mLayout;
    private View mView;
    private BaseActivity mActivity;

    public PlaceHolderFragment() {
    }

    public PlaceHolderFragment(int mLayout, BaseActivity mActivity) {

        this.mLayout = mLayout;
        this.mActivity = mActivity;

    }

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

            mView = inflater.inflate(mLayout,container, false);
            mActivity.setActivityView(mView);

        return mView;
    }
}

此外,以下是我认为在BaseActivity中可能很重要的方法,我使用这些方法创建我的所有活动(我的所有活动都扩展了BaseActivity)

我还没有找到任何解决这个问题的方法,这个问题发生在我的Galaxy S4,Android 4.4.3中。
非常感谢您的帮助。

您将收到
ResourceNotFoundException
,因为您请求加载ID为0的资源

您正在请求加载ID为
0
的资源,因为
mLayout
为0

mLayout
为0,因为它是片段的数据成员,并且正在重新创建片段,可能是因为进程在后台终止

坦白地说,IMHO,将布局ID传递给片段是一种代码味道,而将
BaseActivity
传递给片段则是代码中的一个彻头彻尾的错误。如果片段需要活动,请对片段调用
getActivity()
。完全删除双参数构造函数

假设实际上有合法的理由传入布局ID,请使用factory方法模式和参数
Bundle
,而不是双参数构造函数。参数
Bundle
会自动保存为片段实例状态的一部分,因此当进程终止并稍后重新创建时,它不会丢失

我在中使用工厂方法/参数
Bundle
模式,以便片段知道
ViewPager
的哪个页面。该
位置
存储在参数
Bundle
中,并在片段需要时从该
Bundle
中检索:

/***
  Copyright (c) 2012-14 CommonsWare, LLC
  Licensed under the Apache License, Version 2.0 (the "License"); you may not
  use this file except in compliance with the License. You may obtain a copy
  of the License at http://www.apache.org/licenses/LICENSE-2.0. Unless required
  by applicable law or agreed to in writing, software distributed under the
  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS
  OF ANY KIND, either express or implied. See the License for the specific
  language governing permissions and limitations under the License.

  From _The Busy Coder's Guide to Android Development_
    http://commonsware.com/Android
 */

package com.commonsware.android.pager;

import android.app.Fragment;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.EditText;

public class EditorFragment extends Fragment {
  private static final String KEY_POSITION="position";

  static EditorFragment newInstance(int position) {
    EditorFragment frag=new EditorFragment();
    Bundle args=new Bundle();

    args.putInt(KEY_POSITION, position);
    frag.setArguments(args);

    return(frag);
  }

  @Override
  public View onCreateView(LayoutInflater inflater,
                           ViewGroup container,
                           Bundle savedInstanceState) {
    View result=inflater.inflate(R.layout.editor, container, false);
    EditText editor=(EditText)result.findViewById(R.id.editor);
    int position=getArguments().getInt(KEY_POSITION, -1);

    editor.setHint(String.format(getString(R.string.hint), position + 1));

    return(result);
  }
}

当片段被销毁并重新创建时,您的mLayout将获取其默认值,即0。比如说,当你们将应用程序更改为RAM填充应用程序并返回到你们的应用程序时,就会发生这种情况

所以您需要做的是使用onSavedInstanceState方法并将所需的变量存储在其中。当调用onCreateView并且参数savedInstanceState不为空时,您必须从Bundle中检索变量

阅读此文,了解我想说的内容:

然后看起来是这样的:

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

    if (savedInstanceState != null) {
        mLayout = savedInstanceState.getInt("layout_id");
    }

    mView = inflater.inflate(mLayout,container, false);
    // Return the Activity this fragment is currently associated with.
    BaseActivity activity = (BaseActivity) getActivity();
    activity.setActivityView(mView);

    return mView;
}

@Override
public void onSaveInstanceState(Bundle outState) {
    outState.putInt("layout_id", mLayout);
    super.onSaveInstanceState(outState);
}

感谢您的回复,是否有任何理由认为传递layoudd是一种代码味道?那我怎么知道要充气呢?我目前正在使用这种方法,因为我认为这是重新创建菜单的最佳方法,而不必在每个活动中重复代码。@Waclock:“我怎么知道要膨胀什么?”——将该决定放在片段本身中。我从2011年初就开始研究碎片,我从来没有看到碎片之外的东西告诉碎片膨胀的布局。片段经常被视为本地控制器(在MVC中使用术语“控制器”),如果片段不知道自己的小部件是什么,这是不可能的。一个片段应该是决定它使用什么布局的片段,这样它就知道它的小部件、它的事件处理程序、填充小部件所需的模型数据等等。感谢您的回复。我用安卓系统编程才几个星期,所以我仍然掌握着诀窍。实际上,我正在努力遵循良好的实践,但在Android中很难做到这一点。我会接受你的答案,因为我觉得它比维尔普的答案更正确(尽管他的答案完美无瑕)。再次感谢@沃克尔:他的解决方案和我的大致相当。您通常会使用他的方法(
onSaveInstanceState()
)处理片段动态获取的数据(例如,从联系人列表中选择联系人的结果)。对于传递到片段中的数据,这两种技术都会起作用,不过我的技术更符合谷歌的建议;而且它工作得完美无缺,谢谢:)
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
        Bundle savedInstanceState) {

    if (savedInstanceState != null) {
        mLayout = savedInstanceState.getInt("layout_id");
    }

    mView = inflater.inflate(mLayout,container, false);
    // Return the Activity this fragment is currently associated with.
    BaseActivity activity = (BaseActivity) getActivity();
    activity.setActivityView(mView);

    return mView;
}

@Override
public void onSaveInstanceState(Bundle outState) {
    outState.putInt("layout_id", mLayout);
    super.onSaveInstanceState(outState);
}