谈到Toast,我想只要是做过Android开发的恐怕没有不知道的吧,毫无疑问,Toast是一个非常方便的组件,能在任何能获取上下文的地方进行弹窗提示,既然是弹窗那么我们就可以知道,Toast应该是Window的一部分了。
在这篇文章中,不对Window讲解,默认这部分内容是已知的。
我对Toast的理解最先来自 艺术探索 这本书,在了解了其基本原理之后,我觉得自己去摸索下源码来巩固自己的理解,于是有了这篇文章。本文在写作过程当作由于是边看源码,边写作,可能稍显混乱。我的基本思路如下:
Toast的调用过程如下:
Toast.makeText(this,"test",Toast.LENGTH_LONG).show()
上述代码可以看出,最先是调用了Toast的makeText方法,那么首先深入其中查看:
public static Toast makeText(Context context, CharSequence text, @Duration int duration) {
Toast result = new Toast(context);
LayoutInflater inflate = (LayoutInflater)
context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
View v = inflate.inflate(com.android.internal.R.layout.transient_notification, null);
TextView tv = (TextView)v.findViewById(com.android.internal.R.id.message);
tv.setText(text);
result.mNextView = v;
result.mDuration = duration;
return result;
}
这就是Toast的makeText的实现,我们来分析一下。
首先,第一行代码,Toast result = new Toast(context),去调用Toast的构造方法来初始化了一个Toast,是怎么初始化的呢?深入进去看看:
public Toast(Context context) {
mContext = context;
mTN = new TN();
mTN.mY = context.getResources().getDimensionPixelSize(
com.android.internal.R.dimen.toast_y_offset);
mTN.mGravity = context.getResources().getInteger(
com.android.internal.R.integer.config_toastDefaultGravity);
}
从代码中我们可以看到,Toast的初始化需要上下文,这也是我们在调用Toast的时候,都必须获取上下文的原因。然后新建了一个mTN对象,mTN是一个binder对象,用于IPC,在这部分代码中,这个TN的初始化主要是进行了Window布局元素的创建,具体如下,逻辑还是比较简单的:
TN() {
// XXX This should be changed to use a Dialog, with a Theme.Toast
// defined that sets up the layout params appropriately.
final WindowManager.LayoutParams params = mParams;
params.height = WindowManager.LayoutParams.WRAP_CONTENT;
params.width = WindowManager.LayoutParams.WRAP_CONTENT;
params.format = PixelFormat.TRANSLUCENT;
params.windowAnimations = com.android.internal.R.style.Animation_Toast;
params.type = WindowManager.LayoutParams.TYPE_TOAST;
params.setTitle("Toast");
params.flags = WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON
| WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
| WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE;
}
接下来继续看,在创建出一个mTN之后,就是为mTN的两个成员进行赋值:mY,mGravity。
完成了Toast的初始化之后,跳出来看下面的代码:
LayoutInflater inflate = (LayoutInflater)
context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
View v = inflate.inflate(com.android.internal.R.layout.transient_notification, null);
TextView tv = (TextView)v.findViewById(com.android.internal.R.id.message);
tv.setText(text);
这一部分的代码,我想不用我说了吧?不过还是说说
在这部分代码中是加载了一个布局,这个布局相当简单,就是一个TextView,然后找到布局中的TextView,然后将我们需要显示的Toast显示在这个地方。
完成了数据的绑定之后,接下来继续看:
result.mNextView = v;
result.mDuration = duration;
这两行代码,将之前我们所初始化的view,以及我们传入的需要延时的时长传入。
到此,makeText方法里的逻辑就分析完了,现在来总结以下,它干了什么事:
可以看出,正如方法名所显示的那样,其主要就是确定Toast需要显示的位置(使用window确定),然后确定Toast所需要显示的View,并且绑定到Toast中。到此为止,Toast创建完毕。不过并没有show。那么接下来我们来看看show()方法。
深入show()方法,可以看到:
public void show() {
if (mNextView == null) {
throw new RuntimeException("setView must have been called");
}
INotificationManager service = getService();
String pkg = mContext.getOpPackageName();
TN tn = mTN;
tn.mNextView = mNextView;
try {
service.enqueueToast(pkg, tn, mDuration);
} catch (RemoteException e) {
// Empty
}
}
有了上面的基础,这部分理解上并不存在问题,来看看:
首先判断我们创建的view是否为null,为null直接抛出。
然後获取NotificationManager 的IPC接口,INotificationManager 。
获取到包名,再将我们之前所处理好的view传递给布局的binder对象mTN。
最后,用接口发起跨进程通信。
在服务端,我们来看看这个enqueueToast(pkg, tn, mDuration)的实现,此方法代码量较大,我们挑主要的看:
synchronized (mToastQueue) {
int callingPid = Binder.getCallingPid();
long callingId = Binder.clearCallingIdentity();
try {
ToastRecord record;
int index = indexOfToastLocked(pkg, callback);
// If it's already in the queue, we update it in place, we don't
// move it to the end of the queue.
if (index >= 0) {
record = mToastQueue.get(index);
record.update(duration);
} else {
// Limit the number of toasts that any given package except the android
// package can enqueue. Prevents DOS attacks and deals with leaks.
if (!isSystemToast) {
int count = 0;
final int N = mToastQueue.size();
for (int i=0; i<N; i++) {
final ToastRecord r = mToastQueue.get(i);
if (r.pkg.equals(pkg)) {
count++;
if (count >= MAX_PACKAGE_NOTIFICATIONS) {
Slog.e(TAG, "Package has already posted " + count
+ " toasts. Not showing more. Package=" + pkg);
return;
}
}
}
}
record = new ToastRecord(callingPid, pkg, callback, duration);
mToastQueue.add(record);
index = mToastQueue.size() - 1;
keepProcessAliveLocked(callingPid);
}
// If it's at index 0, it's the current toast. It doesn't matter if it's
// new or just been updated. Call back and tell it to show itself.
// If the callback fails, this will remove it from the list, so don't
// assume that it's valid after this.
if (index == 0) {
showNextToastLocked();
}
在此方法中,首先涉及到同步过程,其所用的锁为一个ArrayList mToastQueue,从名字也可以看出来,它保存的东西是ToastRecord。从这里可以看出一个问题,为啥系统中,应用中如此多的Toast,却能一一显示,并不出现问题,它们是保存在一个队列中的,会一个一个地取出来显示。
接下来的代码,我们慢慢看:
首先,int index = indexOfToastLocked(pkg, callback)
进入此方法:
int indexOfToastLocked(String pkg, ITransientNotification callback)
{
IBinder cbak = callback.asBinder();
ArrayList<ToastRecord> list = mToastQueue;
int len = list.size();
for (int i=0; i<len; i++) {
ToastRecord r = list.get(i);
if (r.pkg.equals(pkg) && r.callback.asBinder() == cbak) {
return i;
}
}
return -1;
}
此方法中主要是遍历了mToastQueue将我们传入的pkg以及mTN与其中的ToastRecord对比,若是存在,那么直接返回这一条记录的index,若是不存在那么返回-1。
看接下来的方法:
if (index >= 0) {
record = mToastQueue.get(index);
record.update(duration);
}
之前说明了,在这个if中处理的就是能在mToastQueue中找到的情况:
首先获取到这条记录,然后更新,其中传入的参数是延时时间,这个函数没有做什么其他的操作,仅仅只是赋值了新的时间。现在来看看没有加入的情况,这部分是重点:
else {
// Limit the number of toasts that any given package except the android
// package can enqueue. Prevents DOS attacks and deals with leaks.
if (!isSystemToast) {
int count = 0;
final int N = mToastQueue.size();
for (int i=0; i<N; i++) {
final ToastRecord r = mToastQueue.get(i);
if (r.pkg.equals(pkg)) {
count++;
if (count >= MAX_PACKAGE_NOTIFICATIONS) {
Slog.e(TAG, "Package has already posted " + count
+ " toasts. Not showing more. Package=" + pkg);
return;
}
}
}
}
从代码中可以看出,此方法使用来限制Toast的数量的,详细看下:
首先若其不是系统应用,那么执行这个方法吗,先获取到enqueue的大小N,设定count计数器,然后遍历,若是遍历项的pkg和我们传入的pkg相同,那么count++。即是说,统计一个package中的Toast的数量,Android只允许最多50个Toast的数量,不过可以知道的是,基本不会有应用能用到那么多,哈哈。
接下来继续解析代码:
record = new ToastRecord(callingPid, pkg, callback, duration);
mToastQueue.add(record);
index = mToastQueue.size() - 1;
keepProcessAliveLocked(callingPid);
接下来的代码,首先新创建了ToastRecord,然后将这条记录添加到mToastQueue,index是用来判断是否是当前Toast的。接下来:
if (index == 0) {
showNextToastLocked();
}
若是当前Toast,那么就显示,看来此方法是真正显示的方法,深入进去看看:
void showNextToastLocked() {
ToastRecord record = mToastQueue.get(0);
while (record != null) {
if (DBG) Slog.d(TAG, "Show pkg=" + record.pkg + " callback=" + record.callback);
try {
record.callback.show();
scheduleTimeoutLocked(record);
return;
} catch (RemoteException e) {
Slog.w(TAG, "Object died trying to show notification " + record.callback
+ " in package " + record.pkg);
// remove it from the list and let the process die
int index = mToastQueue.indexOf(record);
if (index >= 0) {
mToastQueue.remove(index);
}
keepProcessAliveLocked(record.pid);
if (mToastQueue.size() > 0) {
record = mToastQueue.get(0);
} else {
record = null;
}
}
}
}
下面来分析一下这段代码:
因为是当前Toast,那么直接获取到第一个 ToastRecord即可。接下来调用record.callback.show(),看来我们离真相越来越近了,这里的callback就是TN对象,在这里,show()方法又是一次IPC,那么深入这个show()方法:
public void show() {
if (localLOGV) Log.v(TAG, "SHOW: " + this);
mHandler.post(mShow);
}
可以看到,这里就是实际进行显示的地方,Handler的东西就不再多说了,说说这个mShow,其是一个Runnable,实际切换线程之后,运行的是它:
final Runnable mShow = new Runnable() {
@Override
public void run() {
handleShow();
}
};
唉,层层套啊,现在深入看看 handleShow()方法:
public void handleShow() {
if (localLOGV) Log.v(TAG, "HANDLE SHOW: " + this + " mView=" + mView
+ " mNextView=" + mNextView);
if (mView != mNextView) {
// remove the old view if necessary
handleHide();
mView = mNextView;
Context context = mView.getContext().getApplicationContext();
String packageName = mView.getContext().getOpPackageName();
if (context == null) {
context = mView.getContext();
}
mWM = (WindowManager)context.getSystemService(Context.WINDOW_SERVICE);
// We can resolve the Gravity here by using the Locale for getting
// the layout direction
final Configuration config = mView.getContext().getResources().getConfiguration();
final int gravity = Gravity.getAbsoluteGravity(mGravity, config.getLayoutDirection());
mParams.gravity = gravity;
if ((gravity & Gravity.HORIZONTAL_GRAVITY_MASK) == Gravity.FILL_HORIZONTAL) {
mParams.horizontalWeight = 1.0f;
}
if ((gravity & Gravity.VERTICAL_GRAVITY_MASK) == Gravity.FILL_VERTICAL) {
mParams.verticalWeight = 1.0f;
}
mParams.x = mX;
mParams.y = mY;
mParams.verticalMargin = mVerticalMargin;
mParams.horizontalMargin = mHorizontalMargin;
mParams.packageName = packageName;
if (mView.getParent() != null) {
if (localLOGV) Log.v(TAG, "REMOVE! " + mView + " in " + this);
mWM.removeView(mView);
}
if (localLOGV) Log.v(TAG, "ADD! " + mView + " in " + this);
mWM.addView(mView, mParams);
trySendAccessibilityEvent();
}
}
代码量有点大,不过我们需要注意的核心是:
mWM.addView(mView, mParams)
到此,Toast终于在屏幕上显示出来!整个流程结束。关于Toast的cancel方法,分析方法和步骤都差不多,就不赘述了。
整个流程分析完,感觉Android系统真的是相当牛的,深入学习Android,学习大师们的思想是很棒的学习方法,我觉得对于我来说,分析Android这个具体的分析过程,其实是理解软件思想与计算机的知识,计算机相关知识分析到最后都是相通的,所以这部分的付出是值得的。