Toast源码分析

Toast作为向用户展示提示信息的一种方式, 既不会像Dialog一样打断用户的操作体验, 也不会响应任何点击事件, 并且会在短暂的显示之后自动消失. 因此常常会用来显示一些不是非常重要的提示.

Toast的用法

最简单的用法, 只需一行代码就能展示一段提示信息:

Toast.makeText(this, "toast", Toast.LENGTH_SHORT).show();

不过这种方式不能对默认的样式进行修改, 并且在不同的ROM上, 可能会表现出不一样的结果. 所以Toast也提供了一些方法可以让我们做一些自定义的设置.

// 设置要显示的 view
public void setView(View view) 
// 设置左右边距
public void setMargin(float horizontalMargin, float verticalMargin) 
// 设置对齐方式, 以及 水平 和 垂直方向的偏移
public void setGravity(int gravity, int xOffset, int yOffset) 

如果觉得以上这些方法还不够用的话, 甚至可以拿到它的布局属性LayoutParams, 想怎么改就怎么改.(不过是隐藏方法, 需要反射获取)

/**
 * Gets the LayoutParams for the Toast window.
 * @hide
 */
public WindowManager.LayoutParams getWindowParams()

Toast的用法非常简单, 下面通过简单的源码分析了解下它的工作原理

Toast的makeText方法

public static Toast makeText(Context context, @StringRes int resId, @Duration int duration)throws Resources.NotFoundException {
    return makeText(context, context.getResources().getText(resId), duration);
}
public static Toast makeText(Context context, CharSequence text, @Duration int duration) {
    return makeText(context, null, text, duration);
}
 /** 
  * @param looper         用来循环处理消息, 后面有用到
  * @param text           提示的文案
  * @param duration       显示时间
  * @hide 隐藏方法
  */
public static Toast makeText(@NonNull Context context, @Nullable Looper looper, @NonNull CharSequence text, @Duration int duration) {
    // 通过构造方法 new 了一个 Toast 实例
    Toast result = new Toast(context, looper);
    // 初始化布局
    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);
    // 设置显示的view
    result.mNextView = v;
    // 设置时间
    result.mDuration = duration;
    return result;
}

可以看出 makeText 方法中只是创建了一个 Toast, 然后设置了要显示的view, 还有时间. 下面看看 Toast 的构造方法

Toast的构造方法

public Toast(Context context) {
    this(context, null);
}
public Toast(@NonNull Context context, @Nullable Looper looper) {
    mContext = context;
    // new 了一个 TN
    mTN = new TN(context.getPackageName(), looper);
    // 设置垂直偏移
    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);
}

TN是什么? 这个暂时先放一放, 继续看完 Toast的 show 方法再说:

Toast的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;
    // 将要显示的view也设置给了 TN
    tn.mNextView = mNextView;
    try {
        // 调用 service 的 enqueueToast方法, 应该就是用来显示Toast
        // 传入了包名, TN对象的实例,  时长
        service.enqueueToast(pkg, tn, mDuration);
    } catch (RemoteException e) {
        // Empty
    }
}

继续看看 getService 获得的是个什么鬼?

static private INotificationManager getService() {
    if (sService != null) {
        return sService;
    }
    // 典型的aidl写法: 通过 ServiceManager 找到一个名字叫 "notification" 的Binder, 然后创建一个本地的代理对象
    // 关于Binder的原理这里就不展开了, 不太了解的可以查查资料
    sService = INotificationManager.Stub.asInterface(ServiceManager.getService("notification"));
    return sService;
}

既然看到了这里, 下面肯定要找到这个Binder才能继续跟下去了. 根据系统一惯的命名风格, 这个aidl接口的名字叫 INotificationManager. 那我们就全局搜一下 NotificationManagerService, 发现果然有; 然后再搜一下 enqueueToast 方法, 好巧又搜到了; 看看这个方法归属的类:

private final IBinder mService = new INotificationManager.Stub() {
    ...
    @Override
    public void enqueueToast(String pkg, ITransientNotification callback, int duration){
        ...
    }
    ...
}

果然实现了 INotificationManager.Stub, 不用怀疑, 就是这里了. 在研究enqueueToast这个方法之前, 我们先回去搞清楚 TN到底是个什么东西.

private static class TN extends ITransientNotification.Stub {
    ...
    TN(String packageName, @Nullable Looper looper) {
         final WindowManager.LayoutParams params = mParams;
         ...
         // 类型是 TYPE_TOAST
         params.type = WindowManager.LayoutParams.TYPE_TOAST;
         // 保持屏幕常亮, 不获得焦点, 不处理触摸事件
         params.flags = WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON
                    | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
                    | WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE;
        ... (设置一些布局属性)省略 ...
        if (looper == null) {
            looper = Looper.myLooper();
            if (looper == null) {
                // 需要 Looper 来处理消息
                throw new RuntimeException("Can't toast on a thread that has not called Looper.prepare()");
            }
        }
        // Handler将在Looper所在的线程处理消息
        mHandler = new Handler(looper, null) {
            ... 
        };
    }
}

是不是很熟悉, 原来TN也是一个Binder. 其实TN的作用, 就类似于接口回调的功能. 我们在show一个Toast时, 会通过Binder机制访问处于系统进程的NotificationManagerService, 同时也在app进程向 SystemService注册了一个名字叫 “android.app.ITransientNotification” 的Binder. NotificationManagerService在处理完相关逻辑后, 会通过这个Binder回调到app进程, 让Toast处理后续逻辑.

弄清楚这些类的关系和各自的作用后, 再继续研究enqueueToast方法.

NotificationManagerService里面的 enqueueToast

public void enqueueToast(String pkg, ITransientNotification callback, int duration){
    ... 
    // 包名和TN不能为空
    if (pkg == null || callback == null) {
        Slog.e(TAG, "Not doing toast. pkg=" + pkg + " callback=" + callback);
        return ;
    }
    // 是否系统Toast
    final boolean isSystemToast = isCallerSystemOrPhone() || ("android".equals(pkg));
    // 用户是否禁用了 Toast权限
    final boolean isPackageSuspended = isPackageSuspendedForUser(pkg, Binder.getCallingUid());
    // 如果不是系统应用, 且用户禁用了Toast, 则不处理后续逻辑
    if (ENABLE_BLOCKED_TOASTS && !isSystemToast && (!areNotificationsEnabledForPackage(pkg, Binder.getCallingUid()) || isPackageSuspended)) {
        Slog.e(TAG, "Suppressing toast from package " + pkg + (isPackageSuspended
                        ? " due to package suspended by administrator."
                        : " by user request."));
        return;
    }
    // 同步锁
    synchronized (mToastQueue) {
        int callingPid = Binder.getCallingPid();
        long callingId = Binder.clearCallingIdentity();
        try {
            ToastRecord record;
            int index;
            // 是否已经处于 Toast队列中了
            if (!isSystemToast) {
                index = indexOfToastPackageLocked(pkg);
            } else {
                index = indexOfToastLocked(pkg, callback);
            }
            if (index >= 0) {
                // 如果已经处于队列中, 则更新下时间还和TN
                record = mToastQueue.get(index);
                record.update(duration);
                record.update(callback);
            } else {
                // 创建一个新的ToastRecord, 添加到序列中. (注意设置了一个 token, 为什么要用 Binder ?)
                Binder token = new Binder();
                mWindowManagerInternal.addWindowToken(token, TYPE_TOAST, DEFAULT_DISPLAY);
                record = new ToastRecord(callingPid, pkg, callback, duration, token);
                mToastQueue.add(record);
                index = mToastQueue.size() - 1;
            }
            keepProcessAliveIfNeededLocked(callingPid);
            // 如果处于Toast队列的最前端, 则直接展示
            if (index == 0) {
                showNextToastLocked();
            }
        } finally {
            Binder.restoreCallingIdentity(callingId);
        }
    }
}

继续看 showNextToastLocked 方法:

void showNextToastLocked() {
    // 取出队列最前面的 ToastRecord
    ToastRecord record = mToastQueue.get(0);
    while (record != null) {
        try {
            // 回调到 TN, 并传入了 token  (因为Binder可以跨进程通信)
            record.callback.show(record.token);
            // 延迟一段时间后, 隐藏 Toast
            scheduleTimeoutLocked(record);
            return;
        } catch (RemoteException e) {
            // remove it from the list and let the process die
            int index = mToastQueue.indexOf(record);
            if (index >= 0) {
                mToastQueue.remove(index);
            }
            keepProcessAliveIfNeededLocked(record.pid);
            if (mToastQueue.size() > 0) {
                record = mToastQueue.get(0);
            } else {
                record = null;
            }
        }
    }
}

发现 showNextToastLocked 中先是回调到了Toast.TN去显示Toast, 紧接着又开始处理Toast的隐藏逻辑. 关于Toast.TN的部分等下再看, 先看下 scheduleTimeoutLocked:

private void scheduleTimeoutLocked(ToastRecord r){
    mHandler.removeCallbacksAndMessages(r);
    Message m = Message.obtain(mHandler, MESSAGE_TIMEOUT, r);
    long delay = r.duration == Toast.LENGTH_LONG ? LONG_DELAY : SHORT_DELAY;
    // 通过Handler发送一个编码为 MESSAGE_TIMEOUT 的延迟消息, 实现Toast的自动隐藏隐藏
    mHandler.sendMessageDelayed(m, delay);
}

搜索关键字 MESSAGE_TIMEOUT, 发现mHandler在处理该消息时, 调用了下面的方法:

private void handleTimeout(ToastRecord record){
    synchronized (mToastQueue) {
        // 找出 ToastRecord 在队列中的索引
        int index = indexOfToastLocked(record.pkg, record.callback);
        if (index >= 0) {
            // 真正的隐藏逻辑在这里
            cancelToastLocked(index);
        }
    }
}

关于 NotificationManagerService 部分的快结束了, 最后看下 cancelToastLocked 方法:

void cancelToastLocked(int index) {
    ToastRecord record = mToastQueue.get(index);
    try {
        // 回调 Toast.TN, 隐藏 toast
        record.callback.hide();
    } catch (RemoteException e) {}
    // 移除 ToastRecord
    ToastRecord lastToast = mToastQueue.remove(index);
    // 移除 token
    mWindowManagerInternal.removeWindowToken(lastToast.token, true, DEFAULT_DISPLAY);
    keepProcessAliveIfNeededLocked(record.pid);
    // 继续显示下一个Toast
    if (mToastQueue.size() > 0) {
        showNextToastLocked();
    }
}

回到app进程, 看看TN最后是怎么处理Toast的.

TN的show方法和hide方法

public void show(IBinder windowToken) {
    mHandler.obtainMessage(SHOW, windowToken).sendToTarget();
}
...
public void hide() {
    mHandler.obtainMessage(HIDE).sendToTarget();
}

去Handler中转一圈, 发现最后分别调用了下面的两个方法.

// 显示toast
public void handleShow(IBinder windowToken) {
    // 如果此时发现又调用了隐藏或取消, 则直接不显示
    if (mHandler.hasMessages(CANCEL) || mHandler.hasMessages(HIDE)) {
        return;
    }
    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);
        ... 省略的都是设置一些属性的步骤 ...
        // 设置token
        mParams.token = windowToken;
        if (mView.getParent() != null) {
            mWM.removeView(mView);
        }
        try {
            // 显示toast. 通过 WindowManager 添加 view 实现
            mWM.addView(mView, mParams);
            trySendAccessibilityEvent();
        } catch (WindowManager.BadTokenException e) {
            // 在主线程卡顿时, 可能由于消息不能及时处理, 导致在向窗口addView时, NotificationManagerService已经移除了token, 此时会出现此异常
        }
    }
}
...
// 隐藏toast
public void handleHide() {
    if (mView != null) {
        if (mView.getParent() != null) {
            // 移除 view
            mWM.removeViewImmediate(mView);
        }
        mView = null;
    }
}

关于Toast的分析就到这了, 下面补充一个BadTokenException的异常问题.

BadTokenException

通过上面的分析, 我们已经知道了产生该异常的原因, 并且发现源码中已经对异常进行了捕获, 但是在Android7.1上,还是需要我们自己手动解决一下. 直接捕获Toast的show方法是没用的, 因为其内部经过两次IPC通信之后, 真正显示Toast的逻辑并不在show方法中. 我们需要通过反射拿到mTN里面的mHandler对象, 直接在其处理消息的过程中进行捕获.

if (isSdk25()) {
    Field tnField = Toast.class.getDeclaredField("mTN");
    tnField.setAccessible(true);
    Object mTn = tnField.get(mToast);
    Field handlerField = mTn.getClass().getDeclaredField("mHandler");
    handlerField.setAccessible(true);
    Handler handlerOfTn = (Handler) handlerField.get(mTn);
    handlerField.set(mTn, new SafeHandler(handlerOfTn));
}
...
class SafeHandler extends Handler {
    // 用来保存Tn原有handler
    private Handler mNestedHandler;
    public SafeHandler(Handler nestedHandler) {
        // 构造方法里将Tn原有Handler传入
        mNestedHandler = nestedHandler;
    }
    // 在这里捕获
    @Override
    public void dispatchMessage(Message msg) {
        try {
            super.dispatchMessage(msg);
        } catch (WindowManager.BadTokenException e) {}
    }
    @Override
    public void handleMessage(Message msg) {
        // 交由原有Handler处理
        mNestedHandler.handleMessage(msg);
}

  转载请注明: 四月一号 Toast源码分析

 上一篇
AsyncTask AsyncTask
AsyncTask 是一个轻量级的异步任务类. AsyncTask内部封装了线程池和Handler, 让我们在执行异步任务时, 可以比较容易的将任务进度以及执行结果回调到UI线程. AsyncTask的使用方式打开AsyncTask.jav
2019-03-27
下一篇 
HandlerThread&IntentService HandlerThread&IntentService
在处理多个异步任务时, 有两种选择. 一种是并行, 这种情况可以使用线程池; 另一种是串行, 可以使用 HandlerThread来实现. 顾名思意, HandlerThread继承自Thread, 本质上也是一个线程. HandlerT
2019-03-24
  目录