当前位置:  编程技术>移动开发
本页文章导读:
    ▪Managing the Activity Lifecycle-Stopping and Restarting an Activity        Managing the Activity Lifecycle---Stopping and Restarting an Activity原文地址:https://developer.android.com/training/basics/activity-lifecycle/stopping.html ---------------------------------------------- Properly stopping and restarting your act.........
    ▪ ListView运用ViewHolder优化不成反被OOM        ListView使用ViewHolder优化不成反被OOMGoogle官方推荐ListView的优化方式是convertView与ViewHolder的结合使用。这样缓存的方式,确实可以很大程度上压缩getView的生成时间,提高ListView的绘制效率,尤其.........
    ▪ popwindow实现上拉菜单弹出和收缩效果       popwindow实现下拉菜单弹出和收缩效果别的不多说先上图吧 以前做这样的功能,我都是自定义一个LinerLayout,然后控制它的隐藏和显示,还需要添加弹出和隐藏的动画效果。另外还需要处理点.........

[1]Managing the Activity Lifecycle-Stopping and Restarting an Activity
    来源: 互联网  发布时间: 2014-02-18
Managing the Activity Lifecycle---Stopping and Restarting an Activity

原文地址:https://developer.android.com/training/basics/activity-lifecycle/stopping.html

----------------------------------------------

Properly stopping and restarting your activity is an important process in the activity lifecycle that ensures your users perceive that your app is always alive and doesn't lose their progress. There are a few of key scenarios in which your activity is stopped and restarted:

  • The user opens the Recent Apps window and switches from your app to another app. The activity in your app that's currently in the foreground is stopped. If the user returns to your app from the Home screen launcher icon or the Recent Apps window, the activity restarts.
  • The user performs an action in your app that starts a new activity. The current activity is stopped when the second activity is created. If the user then presses the Back button, the first activity is restarted.
  • The user receives a phone call while using your app on his or her phone.

在activity的生命周期里,合适地停止和重启你的activity是一个非常重要的过程,这确保你的用户可以感到你的app总是活跃的而不是丢失了他们的进程。在以下几种情况下你的activity需要被停止或是重启:

  • 用户打开了“最近使用的应用程序”窗口,然后从你的app切换到另一个app。那么你的app里正在前台运行的activity就停止了。如果用户从主菜单里的开启图标或者“最近使用的应用程序”窗口返回到你的app,那activity就会重启。
  • 用户在你的app里执行了一个开始新的activity的动作。那么当前的activity在第二个activity创建时就停止了。如果用户后来就按下了返回按钮,那么第一个activity就会重启。
  • 用户在使用你的app时接到了一个电话。

The Activity class provides two lifecycle methods, onStop() and onRestart(), which allow you to specifically handle how your activity handles being stopped and restarted. Unlike the paused state, which identifies a partial UI obstruction, the stopped state guarantees that the UI is no longer visible and the user's focus is in a separate activity (or an entirely separate app).

Activity类提供了两个生命周期方法,onStop() 和 onRestart(),它们允许你明确地控制你的activity该如何停止和重启。和暂停状态(只是局部的UI被阻塞)不同,停止状态确保UI对用户不再可见,而且用户的焦点在单独的一个activity上(或是完全是另一个单独的app)。

Note: Because the system retains your Activity instance in system memory when it is stopped, it's possible that you don't need to implement the onStop() and onRestart() (or even onStart() methods at all. For most activities that are relatively simple, the activity will stop and restart just fine and you might only need to use onPause() to pause ongoing actions and disconnect from system resources.

注意:因为当你的Activity实例停止时系统在内存中仍然保留了它,你可能不需要实现onStop() 和 onRestart()(甚至是onStart()方法。对于大多数相对简单的activity,它们可以好好地停止和重启,而你可能只需要使用onPause()来暂停正在进行中的动作并且和系统资源分离)。

Figure 1. When the user leaves your activity, the system calls onStop() to stop the activity (1). If the user returns while the activity is stopped, the system calls onRestart() (2), quickly followed by onStart() (3) and onResume() (4). Notice that no matter what scenario causes the activity to stop, the system always calls onPause() before calling onStop().

图1. 当用户离开你的activity时,系统调用onStop()来停止activity (1)。如果当activity停止时用户再次返回,系统调用onRestart() (2),紧接着调用onStart() (3)和onResume() (4)。你应该注意到无论哪种情况下使得activity停止,系统都将先调用onPause()再调用onStop()。

 

Stop Your Activity —— 停止你的活动

When your activity receives a call to the onStop() method, it's no longer visible and should release almost all resources that aren't needed while the user is not using it. Once your activity is stopped, the system might destroy the instance if it needs to recover system memory. In extreme cases, the system might simply kill your app process without calling the activity's final onDestroy() callback, so it's important you use onStop() to release resources that might leak memory.

当你的activity收到一个onStop()方法的调用时,它将不再可见并且应该释放几乎所有用户在不使用它的所不需要的资源。一旦你的activity停止,如果系统需要回收系统内存,它就可能会销毁这个实例。在极端情况下,系统可能会杀死你的app进程而不需要调用activity的终极onDestroy()回调函数,因为使用onStop()来释放那些可能造成内存泄露的资源是很重要的。

Although the onPause() method is called before onStop(), you should use onStop() to perform larger, more CPU intensive shut-down operations, such as writing information to a database.

尽管onPause()方法在onStop()之前被调用,你应该使用onStop()来进行更大量的、更密集的CPU操作,例如向数据库写入信息。

For example, here's an implementation of onStop() that saves the contents of a draft note to persistent storage:

例如,下面onStop()的实现中永久保存了一个草稿记录的内容:

@Override
protected void onStop() {
    super.onStop();  // Always call the superclass method first

    // Save the note's current draft, because the activity is stopping
    // and we want to be sure the current note progress isn't lost.
    ContentValues values = new ContentValues();
    values.put(NotePad.Notes.COLUMN_NAME_NOTE, getCurrentNoteText());
    values.put(NotePad.Notes.COLUMN_NAME_TITLE, getCurrentNoteTitle());

    getContentResolver().update(
            mUri,    // The URI for the note to update.
            values,  // The map of column names and new values to apply to them.
            null,    // No SELECT criteria are used.
            null     // No WHERE columns are used.
            );
}

 

When your activity is stopped, the Activity object is kept resident in memory and is recalled when the activity resumes. You don’t need to re-initialize components that were created during any of the callback methods leading up to the Resumed state. The system also keeps track of the current state for each View in the layout, so if the user entered text into an EditText widget, that content is retained so you don't need to save and restore it.

当你的activity停止时,Activity对象被保留在内存中,然后在activity重新开始时被再次调用。你不需要再次初始化那些在任何回调函数中创建的部件。系统还将跟踪布局中每个View的当前状态,因此如果用户在EditText窗口中输入文字,那这些内容会被保留因此你不需要保存和重新载入它们。

Note: Even if the system destroys your activity while it's stopped, it still retains the state of the View objects (such as text in an EditText) in a Bundle (a blob of key-value pairs) and restores them if the user navigates back to the same instance of the activity (the next lesson talks more about using a Bundle to save other state data in case your activity is destroyed and recreated).

注意:即使在你的activity停止时系统销毁了它,系统仍然在一个Bundle(一个键-值对的二进制大对象)中保留了View对象的状态(例如一个EditText中的文本),如果用户导航到同样的activity实例系统将重新载入它们(next lesson中将探讨更多的关于当你的activity销毁和重建时如何使用一个Bundle来保存其他状态)。

 

Start/Restart Your Activity —— 启动/重启你的活动

When your activity comes back to the foreground from the stopped state, it receives a call to onRestart(). The system also calls the onStart() method, which happens every time your activity becomes visible (whether being restarted or created for the first time). The onRestart() method, however, is called only when the activity resumes from the stopped state, so you can use it to perform special restoration work that might be necessary only if the activity was previously stopped, but not destroyed.

当你的activity从停止状态返回到前台时,它收到一个onRestart()的调用。系统还会调用onStart()方法,这个方法在你的activity变得可见时都会发生(不管是被重启还是第一次被创建)。但是,onRestart()方法仅在activity从停止状态重新开始时才会被调用,因此你需要使用它来执行特殊的恢复工作(仅在activity之前停止了但未被销毁的情况下的必要的工作)。

It's uncommon that an app needs to use onRestart() to restore the activity's state, so there aren't any guidelines for this method that apply to the general population of apps. However, because your onStop() method should essentially clean up all your activity's resources, you'll need to re-instantiate them when the activity restarts. Yet, you also need to instantiate them when your activity is created for the first time (when there's no existing instance of the activity). For this reason, you should usually use the onStart() callback method as the counterpart to the onStop() method, because the system calls onStart() both when it creates your activity and when it restarts the activity from the stopped state.

一个app需要使用onRestart()来重新载入activity的状态是不常见的,因此对于这个应用于母体app的方法没有任何指导。但是,因为你的onStop()方法本质上应该清除你的activity的所有资源,在activity重启时你将需要重新实例化它们。然而,在你的activity第一次创建(这时没有任何存在的activity实例)时,你还需要实例化它们。出于这个原因,你通常应该使用onStart()回调函数作为onStop()方法的对应版本,因为系统在创建你的activity和停止时重启activity都会调用onStart()。

For example, because the user might have been away from your app for a long time before coming back it, the onStart() method is a good place to verify that required system features are enabled:

例如,因为用户在返回时可能已经离开你的app很长一段时间了,onStart()方法是一个核实系统性能是可用的的一个绝佳的地方:

@Override
protected void onStart() {
    super.onStart();  // Always call the superclass method first
    
    // The activity is either being restarted or started for the first time
    // so this is where we should make sure that GPS is enabled
    LocationManager locationManager = 
            (LocationManager) getSystemService(Context.LOCATION_SERVICE);
    boolean gpsEnabled = locationManager.isProviderEnabled(LocationManager.GPS_PROVIDER);
    
    if (!gpsEnabled) {
        // Create a dialog here that requests the user to enable GPS, and use an intent
        // with the android.provider.Settings.ACTION_LOCATION_SOURCE_SETTINGS action
        // to take the user to the Settings screen to enable GPS when they click "OK"
    }
}

@Override
protected void onRestart() {
    super.onRestart();  // Always call the superclass method first
    
    // Activity being restarted from stopped state    
}

When the system destroys your activity, it calls the onDestroy() method for your Activity. Because you should generally have released most of your resources with onStop(), by the time you receive a call to onDestroy(), there's not much that most apps need to do. This method is your last chance to clean out resources that could lead to a memory leak, so you should be sure that additional threads are destroyed and other long-running actions like method tracing are also stopped.

当系统销毁你的activity时,它将为你的Activity调用onDestroy()方法。因为你通常应该在onStop()里释放几乎所有的资源,等到你收到一个onDestroy()的调用时,大多数app不需要做太多的事情。这个方法是你清除那些会造成内存泄露的资源的最后的机会,因此你应该确保附加的线程被全部销毁,其他长期运行的动作,像追踪函数等,也同样被停止了。


    
[2] ListView运用ViewHolder优化不成反被OOM
    来源: 互联网  发布时间: 2014-02-18
ListView使用ViewHolder优化不成反被OOM

Google官方推荐ListView的优化方式是convertView与ViewHolder的结合使用。这样缓存的方式,确实可以很大程度上压缩getView的生成时间,提高ListView的绘制效率,尤其是在快速滚动时。

在我的项目中,ListView的item布局比较复杂,而且列表很长,用户可能会频繁滚动,使用convertView与ViewHolder来优化是理所当然的。使用之后,刚开始确实在用户体验上有一定提升,但是最后却带了一个非常严重的问题——OOM。

       在2.X的机器上反反复复创建和销毁Activity,内存一直往上快速增长,最后OOM,很明显是内存泄漏导致的,但是奇怪的是在4.X的机器上,却没有此问题。

       与系统版本相关的问题,向来是比较头疼。如果是context, listener以及handler这些容易发生内存泄漏的地方出了问题,肯定是所有的系统都会内存泄漏。用MAT查了一遍,只能看到Activity有多个实例存在,没有找到泄漏的根源。想想发生内存泄漏的本质,是销毁对象时外部还有对象持有该对象的引用。

       把Activity的代码又Review了几遍,真的没发现什么可疑的地方。苦思冥想无果,那我就就胡思乱想好了,连那些不太可能出问题的地方也查一遍。在检查ViewHolder的实现方式时,发觉有一个地方与常用的方式不一样。在我的项目中,由于getView外面还封装了一层,其默认的tag已经被占用,要把ViewHolder添加到View中,只能使用setTag的另外一个使用键值对的版本。

  思前想后,还是没有什么头绪。最后只能去怀疑那些不太可能出问题的地方了。在检查到ViewHolder的使用时,发现了一个不同寻常的地方,使用了一个带id参数的setTag版本。当初写这样写的原因,是因为默认的Tag已经被其他地方占用了,这里只能使用一个键值对形式的Tag。会不会是这里的原因,先看看源码再说。

       先看了一下2.2的系统View.java源码,很快就隐隐约约发现了问题的所在。默认的tag在View中是一个私有的成员变量;而键值对形式的Tag,是先存到一个SparseArray中,然后添加到一个以view为键WeakHashMap中,既然拿view来做键,这个WeakHashMap就肯定是静态的。虽然使用了WeakHashMap,寄希望与靠系统来做回收,但是那向来是不可靠的,即使我手动GC,都不会释放。为了确认是这个地方的问题,我查了2.3,4.0,4.1,4.2的源码,发现2.X的处理方式一致,而4.X中tag都是直接插入到一个私有的SparseArray中。

       Google官方说使用ViewHolder可以优化ListView的性能,确实没错,错就错在我的使用方式上。确认了问题所在,只能忍痛去掉了ViewHolder。去掉后,内存泄漏的问题就没有了。至于性能上,损失是必然的,只能先解决燃眉之急,性能的问题等以后再优化。

       最保险的做法是,不要使用带键值对版本的tag。ViewHolder还是还标准的方式来使用。


    
[3] popwindow实现上拉菜单弹出和收缩效果
    来源: 互联网  发布时间: 2014-02-18
popwindow实现下拉菜单弹出和收缩效果

别的不多说先上图吧



以前做这样的功能,我都是自定义一个LinerLayout,然后控制它的隐藏和显示,还需要添加弹出和隐藏的动画效果。另外还需要处理点击空白处和返回该键下拉菜单自动关闭。今天在eoe上无意看到popwindow也能实现这样的功能,而且根本不需要处理动画效果和空白处、返回键点击自动关闭的功能。哎,看来我真的out了,直接上代码:


这里是自定义的PopMenu 类,里面对popwindow进行了封装,实现下拉菜单的效果

package com.su.testpopwindow;


import java.util.ArrayList;


import android.content.Context;
import android.graphics.drawable.BitmapDrawable;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.view.ViewGroup.LayoutParams;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.BaseAdapter;
import android.widget.ListView;
import android.widget.PopupWindow;
import android.widget.TextView;
import com.su.testpopwindow.R;


public class PopMenu {
private ArrayList<String> itemList;
private Context context;
private PopupWindow popupWindow;
private ListView listView;




public PopMenu(Context context) {
this.context = context;
itemList = new ArrayList<String>();


View view = LayoutInflater.from(context)
.inflate(R.layout.popmenu, null);


// 设置 listview
listView = (ListView) view.findViewById(R.id.listView);
listView.setAdapter(new PopAdapter());
listView.setFocusableInTouchMode(true);
listView.setFocusable(true);


popupWindow = new PopupWindow(view, 100, LayoutParams.WRAP_CONTENT);
popupWindow = new PopupWindow(view, context.getResources()
.getDimensionPixelSize(R.dimen.popmenu_width),
LayoutParams.WRAP_CONTENT);


// 这个是为了点击“返回Back”也能使其消失,并且并不会影响你的背景(很神奇的)
popupWindow.setBackgroundDrawable(new BitmapDrawable());
}


// 设置菜单项点击监听器
public void setOnItemClickListener(OnItemClickListener listener) {
listView.setOnItemClickListener(listener);
}


// 批量添加菜单项
public void addItems(String[] items) {
for (String s : items) {
itemList.add(s);
}
}


// 单个添加菜单项
public void addItem(String item) {
itemList.add(item);
}


// 下拉式 弹出 pop菜单 parent 右下角
public void showAsDropDown(View parent) {
popupWindow.showAsDropDown(parent,
10,
// 保证尺寸是根据屏幕像素密度来的
context.getResources().getDimensionPixelSize(
R.dimen.popmenu_yoff));


// 使其聚集
popupWindow.setFocusable(true);
// 设置允许在外点击消失
popupWindow.setOutsideTouchable(true);
// 刷新状态
popupWindow.update();
}


// 隐藏菜单
public void dismiss() {
popupWindow.dismiss();
}


// 适配器
private final class PopAdapter extends BaseAdapter {


@Override
public int getCount() {
return itemList.size();
}


@Override
public Object getItem(int position) {
return itemList.get(position);
}


@Override
public long getItemId(int position) {
return position;
}


@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if (convertView == null) {
convertView = LayoutInflater.from(context).inflate(
R.layout.pomenu_item, null);
holder = new ViewHolder();
convertView.setTag(holder);


holder.groupItem = (TextView) convertView
.findViewById(R.id.textView);


} else {
holder = (ViewHolder) convertView.getTag();
}


holder.groupItem.setText(itemList.get(position));


return convertView;
}


private final class ViewHolder {
TextView groupItem;
}
}
}


这是主activity,点击button,弹出下拉菜单

package com.su.testpopwindow;


import android.app.Activity;
import android.content.Context;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import android.widget.Button;
import android.widget.AdapterView.OnItemClickListener;


public class TestPullPopWindowActivity extends Activity {
private PopMenu popMenu;
private Context context;


/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
context = TestPullPopWindowActivity.this;
popMenu = new PopMenu(context);
popMenu.addItems(new String[] { "停发", "延期", "删除", "备注", "修改","我们" });
// 菜单项点击监听器
popMenu.setOnItemClickListener(popmenuItemClickListener);


Button button1 = (Button) findViewById(R.id.button1);
button1.setOnClickListener(new View.OnClickListener() {


@Override
public void onClick(View v) {
popMenu.showAsDropDown(v);
}
});


}


// 弹出菜单监听器
OnItemClickListener popmenuItemClickListener = new OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position,
long id) {
System.out.println("下拉菜单点击" + position);
popMenu.dismiss();
}
};
}


最后附上源码下载地址:

http://download.csdn.net/detail/duanyu218/5045677


    
最新技术文章:
▪Android开发之登录验证实例教程
▪Android开发之注册登录方法示例
php开源软件 iis7站长之家
▪Android实现将已发送的短信写入短信数据库的...
▪Android发送短信功能代码
▪Android根据电话号码获得联系人头像实例代码
▪Android中GPS定位的用法实例
▪Android实现退出时关闭所有Activity的方法
▪Android实现文件的分割和组装
▪Android录音应用实例教程
▪Android双击返回键退出程序的实现方法
▪Android实现侦听电池状态显示、电量及充电动...
▪Android获取当前已连接的wifi信号强度的方法
▪Android实现动态显示或隐藏密码输入框的内容
▪根据USER-AGENT判断手机类型并跳转到相应的app...
▪Android Touch事件分发过程详解
▪Android中实现为TextView添加多个可点击的文本
▪Android程序设计之AIDL实例详解
▪Android显式启动与隐式启动Activity的区别介绍
▪Android按钮单击事件的四种常用写法总结
▪Android消息处理机制Looper和Handler详解
▪Android实现Back功能代码片段总结
▪Android实用的代码片段 常用代码总结
▪Android实现弹出键盘的方法
▪Android中通过view方式获取当前Activity的屏幕截...
▪Android提高之自定义Menu(TabMenu)实现方法
▪Android提高之多方向抽屉实现方法
▪Android提高之MediaPlayer播放网络音频的实现方法...
▪Android提高之MediaPlayer播放网络视频的实现方法...
▪Android提高之手游转电视游戏的模拟操控
 


站内导航:


特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

©2012-2021,,E-mail:www_#163.com(请将#改为@)

浙ICP备11055608号-3