Android常见问题总结(四)

上一篇博客传送门:Android常见问题总结(三)

16. 动画有哪几类,各有什么特点

如今Android的动画主要有三种,分别是:逐帧(Frame)动画,补间(Tween)动画,属性(Property)动画

逐帧(Frame)动画

逐帧动画是最容易理解的动画,它要求我们把动画过程的每张静态图片都准备好,然后依次显示,利用人眼“视觉暂留”的原理形成动画效果。

例子:肥波跳舞?

素材准备(共27帧):

fat_po.xml:

<?xml version="1.0" encoding="utf-8"?>
<animation-list xmlns:android="http://schemas.android.com/apk/res/android"
    android:oneshot = "false" >

    <item android:drawable="@drawable/fat_po_f01" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f02" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f03" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f04" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f05" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f06" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f07" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f08" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f09" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f10" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f11" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f12" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f13" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f14" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f15" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f16" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f17" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f18" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f19" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f20" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f21" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f22" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f23" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f24" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f25" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f26" android:duration="60" />
    <item android:drawable="@drawable/fat_po_f27" android:duration="60" />

</animation-list>

animation-list的oneshot属性用于设置动画是否只播放一次,true是,false表示循环播放。

Activity布局xml:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:gravity="center">

    <ImageView
        android:id="@+id/image_view"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:background="@animator/fat_po"/>

</LinearLayout>

把逐帧(Frame)动画设置为ImageView的背景。

Activity的Java代码:

public class MyActivity extends Activity {

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		ImageView imageView = (ImageView)findViewById(R.id.image_view);
		// 开始动画,默认为停止
		((AnimationDrawable)imageView.getBackground()).start();
	}

}

由于逐帧动画默认是停止,因此我们需要调用其start方法才能播放动画。

效果如下:

补间(Tween)动画

相比与逐帧动画要求我们把动画的每一帧都列出来,补间动画只需要我们指定动画开始、动画结束等“关键帧”,而其中动画变化的“中间帧”由系统计算补齐。

例子:花瓣开合

动画材料:

花瓣动画xml文件:

close.xml

<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
    android:interpolator="@android:anim/linear_interpolator">

    <scale android:fromXScale="1.0"
        android:toXScale="0.01"
        android:fromYScale="1.0"
        android:toYScale="0.01"
        android:pivotX="50%"
        android:pivotY="50%"
        android:fillAfter="true"
        android:duration="3000"/>

    <alpha
        android:fromAlpha="1"
        android:toAlpha="0.05"
        android:duration="3000"/>

    <rotate
        android:fromDegrees="0"
        android:toDegrees="1800"
        android:pivotX="50%"
        android:pivotY="50%"
        android:duration="3000"/>
</set>

花瓣关闭xml文件,使用了scale(大小变化),alpha(透明度变化),rotate(旋转变化)三种动画,执行的速度是linear线性的

open.xml

<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
    android:interpolator="@android:anim/linear_interpolator">

    <scale android:fromXScale="0.01"
        android:toXScale="1.0"
        android:fromYScale="0.01"
        android:toYScale="1.0"
        android:pivotX="50%"
        android:pivotY="50%"
        android:fillAfter="true"
        android:duration="3000"/>

    <alpha
        android:fromAlpha="0.05"
        android:toAlpha="1"
        android:duration="3000"/>

    <rotate
        android:fromDegrees="1800"
        android:toDegrees="0"
        android:pivotX="50%"
        android:pivotY="50%"
        android:duration="3000"/>
</set>

与close动画类似。

界面的xml文件:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:gravity="center">

    <ImageView
        android:id="@+id/image_view"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:src="@drawable/flower"/>

</LinearLayout>

只有一个简单的imageView。

Activity的Java代码:

public class MyActivity extends Activity {

	private Handler handler = new Handler() {
		public void handleMessage(android.os.Message msg) {
			if (flag)
				imageView.startAnimation(close);
			else
				imageView.startAnimation(open);
			flag = !flag;
		};
	};

	private Animation open, close;
	private ImageView imageView;
	boolean flag = true; // 花瓣状态,true开,false合

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		imageView = (ImageView)findViewById(R.id.image_view);
		// 花瓣张开
		open = AnimationUtils.loadAnimation(this, R.anim.open);
		// 保留动画变化后的状态
		open.setFillAfter(true);
		// 花瓣关闭
		close = AnimationUtils.loadAnimation(this, R.anim.close);
		// 保留动画变化后的状态
		close.setFillAfter(true);
		// 设置重复任务
		new Timer().schedule(new TimerTask() {

			@Override
			public void run() {
				// 通过handler发送消息刷新UI
				handler.sendEmptyMessage(0);
			}
		}, 0, 3500);
	}

}

加载了close与open动画后,通过Timer的重复任务不断发送消息给handler,通过handler来播放动画。

效果如下:

对于补间(Tween)动画我们只需要准备动画开始与结束的关键帧即可,其中的“中间帧”使用Android提供的多种变换(Scale,Alpha,Rotate,translate)生成即可。

属性(Property)动画

从某种角度来看,属性动画是增强版的补间动画,属性动画的强大主要体现在两个方面:

  1. 补间动画只能定义两个关键帧在“透明度”、“旋转”、“缩放”、“位移”4个方面的变化,但属性动画可以定义任何属性的变化
  2. 补间动画只能对UI组件执行动画,但属性动画几乎可以对任何对象执行动画

由于属性动画比较复杂,本文在此不展开讨论了,推荐两篇大神博客供大家参考:

Android 属性动画(Property Animation) 完全解析 (上)

Android 属性动画(Property Animation) 完全解析 (下)

17. Handler、Looper消息队列模型,各部分的作用。

简单提提Handler、Looper模型中各部分的作用,主要有以下三部分:

  • MessageQueue:消息队列,存储待处理的消息
  • Looper:封装了消息队列与Handler,线程绑定,使用loop方法循环处理消息
  • Handler:消息处理的辅助类,里面封装了消息的投递、处理和获取等一系列操作

详细的情况请参考看我以前的博文:

Android Looper和Handler小结

18. 怎样退出终止App

测试:

依次打开3个Activity(ActivityA,ActivityB,ActivityC),并在第3个Activity中终止App

System.exit(0)

重写Application方法:

public class MyApplication extends Application {

	public void exit() {
		System.exit(0);
	}

}

切记在AndroidManifest中修改使用的Application

在第三个Activity中调用该方法:

public class ActivityC extends Activity {

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity);
		Button button = (Button)findViewById(R.id.btn);
		button.setText("Finish app");
		button.setOnClickListener(new OnClickListener() {

			@Override
			public void onClick(View v) {
				// 退出App
				((MyApplication)getApplication()).exit();
			}
		});
	}
}

失败,应用进程被杀死,然而过会应用重启了……并且ActivityA和ActivityB都被“复活了”,只杀死了ActivityC。

android.os.Process.killProcess(android.os.Process.myPid())

失败,与System.exit(0)一样,虽然杀死了进程,但过会就被重启了,并且ActivityA和ActivityB都被“复活了”。

ActivityManager am= (ActivityManager) this.getSystemService(Context.ACTIVITY_SERVICE);

am.killBackgroundProcesses(this.getPackageName());

失败,毫无反应。

自定义BaseActivity维护Activity列表

自定义一个BaseActivity:

public class BaseActivity extends Activity {

	// 维护一个Activity软引用的列表
	private static List<SoftReference<Activity>> list = new ArrayList<SoftReference<Activity>>();

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		list.add(new SoftReference<Activity>(this));
	}

	@Override
	protected void onDestroy() {
		super.onDestroy();
		list.remove(new SoftReference<Activity>(this));
	}

	/**
	 * 关闭所有的Activity
	 */
	public void finishAll() {
		for (SoftReference<Activity> sr : list) {
			if (sr.get() != null) {
				sr.get().finish();
			}
		}
	}

}

对于ActivityA、ActivityB和ActivityC继承BaseActivity而不是Activity,在ActivityC中调用finishAll方法即可关闭所有Activity进而退出App。

finishAffinity()

直接关闭相同任务栈中的所用Activity,与上一个方法效果差不多,但是是Android自带的,方便多了。

综上所述,测试结果如下:

  1. System.exit(0):只能关闭当前Activity,关闭进程可能导致数据存储问题,不推荐
  2. android.os.Process.killProcess(android.os.Process.myPid()):同上

  3. ActivityManager am= (ActivityManager) this.getSystemService(Context.ACTIVITY_SERVICE);

    am.killBackgroundProcesses(this.getPackageName()):测试无效

  4. 自定义BaseActivity维护Activity列表:可以关闭依次启动的所用Activity,进而退出整个App
  5. finishAffinity():可以关闭同一个任务栈中的所有Activity,Android自带方法,比较方便

19. Asset目录与res目录的区别

Asset目录和res目录均为Android中用来存放资源的目录,其中:

  • asset目录下存放的资源代表应用无法直接访问的原生资源,应用程序需要通过AssetManager以二进制流的形式来读取资源
  • res目录下的资源可通过R资源清单类访问,Android SDK会在编译时在R类中为他们创建对应的索引项

20. Android怎么加速启动Activity

个人认为,影响Activity启动时间的主要有两个地方:

  1. onCreate、onStart、onResume等回调方法的执行时间
  2. Activity对应的界面的inflate时间

对于第一点,我们应该尽量减少在这些回调方法中执行耗时操作(涉及数据库,图片等),如果一定要执行耗时操作,可以考虑新开子线程处理。

对于第二点,我们应该合理使用各种xml的优化标签,并界面上减少View的嵌套层数与绘制时间。(可参考Android常见问题总结(三)中的问题14)

时间: 2024-10-16 06:39:45

Android常见问题总结(四)的相关文章

Android学习Scroller(四)——实现拉动后回弹的布局

MainActivity如下: package cc.testscroller2; import android.os.Bundle; import android.app.Activity; /** * Demo描述: * 实现可以拉动后回弹的布局. * 类似于下拉刷新的. * * 参考资料: * 1 http://gundumw100.iteye.com/blog/1884373 * 2 http://blog.csdn.net/gemmem/article/details/7321910

Android切近实战(四)

上一节我们看了系统参数的主界面,大家应该还有印象,如下 那本节我们来看一下修改和删除. 上节我已经介绍了系统参数修改以及删除的WebService,如下 其中系统参数修改的描述如下 系统参数删除的定义如下 接下来我们需要知道的是如何实现修改和删除按钮的功能.记得上节我们使用系统提供的SimpleAdapter去展示listview的数据.这样是无法实现按钮的响应的.所以在实现这两个按钮的功能之前,首先需要让他们能够响应点击事件.所以需要我们自己定义Adapter. public class cu

Android:图解四种启动模式 及 实际应用场景解说

在一个项目中会包括着多个Activity,系统中使用任务栈来存储创建的Activity实例,任务栈是一种"后进先出"的栈结构.举个栗子,若我们多次启动同一个Activity.系统会创建多个实例依次放入任务栈中.当按back键返回时,每按一次,一个Activity出栈,直到栈空为止.当栈中无不论什么Activity.系统就会回收此任务栈. 上面这个样例中的Activity并没有设置启动模式,你会发现多次启动同一个Activity.而系统却创建了多个实例,白白浪费内存,这样的情况Andro

Android自定义视图四:定制onMeasure强制显示为方形

这个系列是老外写的,干货!翻译出来一起学习.如有不妥,不吝赐教! Android自定义视图一:扩展现有的视图,添加新的XML属性 Android自定义视图二:如何绘制内容 Android自定义视图三:给自定义视图添加"流畅"的动画 Android自定义视图四:定制onMeasure强制显示为方形 上一篇开发之后的效果如上图.不过看着这张图,需要注意的不是我们自定义视图展示了什么,而是这个视图的大小和位置.你会看到这个折线图有一个特定的大小(size).这个size是怎么定的呢?现在的代

[转][Android]Android数据的四种存储方式

android.database.sqlite类 SQLiteQueryBuilder java.lang.Object android.database.sqlite.SQLiteQueryBuilder public class SQLiteQueryBuilderextends Object This is a convience class that helps build SQL queries to be sent to SQLiteDatabase objects. 构造方法摘要

Android数据的四种存储方式SharedPreferences、SQLite、Content Provider和File (一) —— 总览

Android数据的四种存储方式SharedPreferences.SQLite.Content Provider和File (一) —— 总览 作为一个完成的应用程序,数据存储操作是必不可少的.因此,Android系统一共提供了四种数据存储方式.分别是:SharePreference.SQLite.Content Provider和File.由于Android系统中,数据基本都是私有的的,都是存放于“data/data/程序包名”目录下,所以要实现数据共享,正确方式是使用Content Pro

android 实践项目四

android 实践项目四 本周主要是开发android baidumap实现公交的查询 1.权限的取得和对屏幕的支持 1 <uses-permission android:name="android.permission.BAIDU_LOCATION_SERVICE" > 2 </uses-permission> 3 <uses-permission android:name="android.permission.ACCESS_NETWORK

Android学习笔记四:添加Source

问题描述 Source not foundThe JAR file D:\.....\sdk\platforms\android-20\android.jar has no source attachment. 问题原因及解决办法 1. 使用SDK Manager下载最新版本的Sources for Android SDK 一般文件下载目录默认在SDK下的sources文件中即 \adt-bundle-windows-x86_64-20130522\sdk\sources\android-20

Android 常见问题收集 (持续更新)

gridview中设置item的高度. 一.BaseAdapter中的getView中设置 AbsListView.LayoutParams param = new AbsListView.LayoutParams( android.view.ViewGroup.LayoutParams.MATCH_PARENT, parent.getWidth()/2+DpPx.dip2px(context,20));//-20 10spacing 5padding + 40 textview.height