[已解决]项目中应用来回切换后,就会提示“已停止运行”

2023-10-12 13:58

本文主要是介绍[已解决]项目中应用来回切换后,就会提示“已停止运行”,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

经过debug,我发现是应用在调用我们的一个全局变量的时候,这个全局变量为空了,但是我们明明是有设置,而且在不切换之前也是有值的

原因分析:

最后在网上查找了一下,原来在内存较少的时候,来回切换的时候gc 会去将一些内存释放掉的,全局变量会被重新设置为默认值


模拟的方法:

1: 首先声明一个静态的全局变量类:

package com.example.kodulf.crashdemo;/*** Created by Kodulf on 2017/5/12.*/public class StaticClass {public static String staticString;public static int[] intArray;
}

2: 创建一个MainActivity

package com.example.kodulf.crashdemo;import android.content.Intent;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;public class MainActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);StaticClass.staticString = "hello";//初始化静态类里面的数组,给一个很大的内存亮StaticClass.intArray = new int[1024*1024*25];//启动第二个ActivityIntent intent = new Intent(this,SecondActivity.class);startActivity(intent);}
}

3: 创建SecondActivity

package com.example.kodulf.crashdemo;import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;public class SecondActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_second);Log.d("kodulf",""+StaticClass.staticString);//下面这里// 在当前显示的是SecondActivity的时候切换到后台,然后再切换回来,在内存较少的情况下,会报错// 在当前显示的是SecondActivity的时候切换到后台,还有一种让他直接报错的方法,就是通过一些内存管理软件,在当前显示的是SecondActivity的时候切换到后台,点击内存清理,然后再切换回来 就会报错了Log.d("kodulf",""+StaticClass.intArray.length);}}


4: 最直接的模拟方法:

启动该应用,默认的会在MainActivity的时候初始化那个静态的int数组,然后会自动启动SecondActivity

这个时候将该应用切换到后台

找到一个内存管理的工具,例如360软件助手,进行内存清理,

再将该应用切换回来,就会直接报错了



解决办法:

将全局变量尽量使用常量,如果要涉及到修改的全局变量,那么最好不要直接食用

1: 可以使用sharepreference 去保存然后。

2: 或者在onSaveInstanceState的时候保存这个全局变量,onRestoreInstanceState 的时候再给这个全局变量重新赋值就好了。

package com.example.kodulf.crashdemo;import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;public class SecondActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_second);Log.d("kodulf",""+StaticClass.staticString);//下面这里// 在当前显示的是SecondActivity的时候切换到后台,然后再切换回来,在内存较少的情况下,会报错// 在当前显示的是SecondActivity的时候切换到后台,还有一种让他直接报错的方法,就是通过一些内存管理软件,在当前显示的是SecondActivity的时候切换到后台,点击内存清理,然后再切换回来 就会报错了Log.d("kodulf",""+StaticClass.intArray.length);}@Overrideprotected void onSaveInstanceState(Bundle outState) {super.onSaveInstanceState(outState);outState.putSerializable("intArray",StaticClass.intArray);}@Overrideprotected void onRestoreInstanceState(Bundle savedInstanceState) {super.onRestoreInstanceState(savedInstanceState);StaticClass.intArray = (int[])savedInstanceState.getSerializable("intArray");}
}




+++++++++++++++++++++++++++++++++++++++++

参考:http://blog.csdn.net/hs_73/article/details/50911133

后期测试的时候,发现有个偶然现象,当应用被切换到后台一段时间再切换回来的时候会莫名的崩溃,而且在log的错误信息,云测以及友盟的错误列表上都没有发现有错误日志输出。确实是比较棘手,但这确实是应用的bug,无法忽视,但是这个问题又无法很容易的重现。后期查找资料分析一下,可能是跟这几个原因有关:

1.没有很正确的理解getContext() , getActivity() ,Activity.this ,getApplicationContext的区别以及用法,乱用导致:

Context:上下文(指的就是这个应用运行所依赖的运行环境), Context分类:
       ①.Activity:具体的上下文环境;Activity.this的context属于activity ,activity 摧毁他就摧毁
       ②.ApplicationContext:整体的上下文环境;getApplicationContext()通过该方法获取,getApplicationContext() 生命周期是整个应用,应用摧毁它才摧毁 
       ③getContext();. getContext获取的是当前对象所在的Context
对于绝大多数场景,使用Activity或者是ApplicationContext效果都一样.
activity.this要返回一个activity,而getApplicationContext()就不一定返回一个activity
      但是有些特殊情况下,使用Activity这种上下文的地方,不可以使用ApplicationContext;
      也有些地方,使用ApplicationContext的地方,也不能使用Activity:一般是需要对全局做初始化操作的时候.

 2.应用被切换到后台以后,由于Android的内存回收机制,Activity切换到后台之后,由于内存不够,此Activity被系统回收了,一段时间之后回到该应用程序,Activity被重新实例化了。而Activity被系统销毁时,附属在该Activity的Fragment并没有被销毁,在Activity的onSaveInstanceState里面将Fragment状态保存起来了,所以Activity重新创建了,但是FragmentA和FragmentB还是之前的,而此时FragmentA和FragmentB所附属的Activity已经被系统回收了,这次再调用getActivity时返回了null,我们看看FragmentActivity源码中的onSaveInstanceState方法:

[java]  view plain copy
  1. protected void onSaveInstanceState(Bundle outState)  
  2.    {  
  3.      super.onSaveInstanceState(outState);  
  4.      Parcelable p = mFragments.saveAllState();  
  5.      if (p != null) {  
  6.        outState.putParcelable("android:support:fragments", p);  
  7.     }  
  8.    }  
由上面源码可以看出,FragmentActivity确实在onSaveInstanceState方法里面将Fragment的状态保存了。 解决方法其实很简单,我们只要让FragmentActivity被系统回收的时候,不保存Fragment的状态即可,即在FragmentActivity中重写onSaveInstanceState方法,并且注释掉super.onSaveInstanceState(outState)就行了。
[java]  view plain copy
  1. @Override  
  2.  protected void onSaveInstanceState(Bundle outState) {  
  3. //        super.onSaveInstanceState(outState);  
  4. }  

参考:http://www.cnblogs.com/liuling/p/2015-9-21-1.html

3.存在某些全局静态变量被释放回收掉,再次打开时,由于缺少值,在一些配置较低的终端上容易出现。

4.还比如,从一个界面跳转到第二个界面时,存在传值,当在第二个界面时应用被切换到后台,当你再切换到前台时,在这第二个activity里,在其onResume()方法中做了数据访问用到了前一个界面的传值,当这个值已经被回收掉,这时就会crash掉了,多做些非空判断总归是有利无弊的。

造成这个问题的原因可能很多,以上是我个人的项目经验,仅代表个人观点,欢迎补充指正。




这篇关于[已解决]项目中应用来回切换后,就会提示“已停止运行”的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/196313

相关文章

qt5cored.dll报错怎么解决? 电脑qt5cored.dll文件丢失修复技巧

《qt5cored.dll报错怎么解决?电脑qt5cored.dll文件丢失修复技巧》在进行软件安装或运行程序时,有时会遇到由于找不到qt5core.dll,无法继续执行代码,这个问题可能是由于该文... 遇到qt5cored.dll文件错误时,可能会导致基于 Qt 开发的应用程序无法正常运行或启动。这种错

电脑提示xlstat4.dll丢失怎么修复? xlstat4.dll文件丢失处理办法

《电脑提示xlstat4.dll丢失怎么修复?xlstat4.dll文件丢失处理办法》长时间使用电脑,大家多少都会遇到类似dll文件丢失的情况,不过,解决这一问题其实并不复杂,下面我们就来看看xls... 在Windows操作系统中,xlstat4.dll是一个重要的动态链接库文件,通常用于支持各种应用程序

一文详解如何在idea中快速搭建一个Spring Boot项目

《一文详解如何在idea中快速搭建一个SpringBoot项目》IntelliJIDEA作为Java开发者的‌首选IDE‌,深度集成SpringBoot支持,可一键生成项目骨架、智能配置依赖,这篇文... 目录前言1、创建项目名称2、勾选需要的依赖3、在setting中检查maven4、编写数据源5、开启热

SpringBoot项目配置logback-spring.xml屏蔽特定路径的日志

《SpringBoot项目配置logback-spring.xml屏蔽特定路径的日志》在SpringBoot项目中,使用logback-spring.xml配置屏蔽特定路径的日志有两种常用方式,文中的... 目录方案一:基础配置(直接关闭目标路径日志)方案二:结合 Spring Profile 按环境屏蔽关

SpringBoot排查和解决JSON解析错误(400 Bad Request)的方法

《SpringBoot排查和解决JSON解析错误(400BadRequest)的方法》在开发SpringBootRESTfulAPI时,客户端与服务端的数据交互通常使用JSON格式,然而,JSON... 目录问题背景1. 问题描述2. 错误分析解决方案1. 手动重新输入jsON2. 使用工具清理JSON3.

MySQL 设置AUTO_INCREMENT 无效的问题解决

《MySQL设置AUTO_INCREMENT无效的问题解决》本文主要介绍了MySQL设置AUTO_INCREMENT无效的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录快速设置mysql的auto_increment参数一、修改 AUTO_INCREMENT 的值。

关于跨域无效的问题及解决(java后端方案)

《关于跨域无效的问题及解决(java后端方案)》:本文主要介绍关于跨域无效的问题及解决(java后端方案),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录通用后端跨域方法1、@CrossOrigin 注解2、springboot2.0 实现WebMvcConfig

Go语言中泄漏缓冲区的问题解决

《Go语言中泄漏缓冲区的问题解决》缓冲区是一种常见的数据结构,常被用于在不同的并发单元之间传递数据,然而,若缓冲区使用不当,就可能引发泄漏缓冲区问题,本文就来介绍一下问题的解决,感兴趣的可以了解一下... 目录引言泄漏缓冲区的基本概念代码示例:泄漏缓冲区的产生项目场景:Web 服务器中的请求缓冲场景描述代码

解决JSONField、JsonProperty不生效的问题

《解决JSONField、JsonProperty不生效的问题》:本文主要介绍解决JSONField、JsonProperty不生效的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录jsONField、JsonProperty不生效javascript问题排查总结JSONField

github打不开的问题分析及解决

《github打不开的问题分析及解决》:本文主要介绍github打不开的问题分析及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、找到github.com域名解析的ip地址二、找到github.global.ssl.fastly.net网址解析的ip地址三