理解Android中Dialog

article/2025/8/27 22:18:55

文章收藏的好句子:你能走多远、爬多高,不仅取决于你自身的力量,还取决于周围人带动的力量。

PS:本文是基于 Android Api 26 来分析源码的。

1、Dialog 的 Window 是在哪里创建的?

Dialog 的 Window 是在什么地方创建的呢?我们来看看 Dialog 的一个构造方法,那就是 Dialog(@NonNull Context context, @StyleRes int themeResId, boolean createContextThemeWrapper) 方法;

f4ac8418e45a00b7c34f99e10bcb6213.png

看到注释1中的代码没有,它创建了一个 Window,而这个 Window 的实现类是 PhoneWindow,它跟 Activity 一样也是用 PhoneWindow 作为自己的 Window;好,那既然有 Window,那就必然有 View 显示出来对不对?那我们看看这个 Dialog 的 PhoneWindow 是如何加载 View 的,我们看一下 Dialog 其中一个加载 View 的方法 setContentView(@LayoutRes int layoutResID);

7146e6fe1f67081f269a361ad89e5775.png

mWindow 就是 Window,而 Window 的实现类是 PhoneWindow,所以我们往下看 PhoneWindow 的 setContentView(@LayoutRes int layoutResID) 方法;

33833314a0b6eb9c36fee057da51c74f.png

看到 PhoneWindow 的 setContentView(@LayoutRes int layoutResID) 方法没有,Activity 的 setContentView(@LayoutRes int layoutResID) 方法解析 View 过程和 Dialog 的 setContentView(@LayoutRes int layoutResID) 方法解析 View 过程是一样的,所以就不在对 PhoneWindow 的 setContentView(@LayoutRes int layoutResID) 方法进行分析了,可以看Android中AppCompatActivity的setContentView方法分析这篇文章进行理解它。

2、我们在平时开发的时候是否遇到这样的一个问题:如果 Dialog 使用的 Context 不是 Activity 的而是 Application 的,那么就会报错。

好,为了更好的理解,我们先上一段简单的代码;

467eceef293c767dde42f0aa6c01e43d.png

把 app 运行一下,发现报了如下错误;

2f3b5aefc58479c8a465befa880f6f9d.png

这里报错的原因是没有应用 token 所导致的,应用 token 一般只有 Activity 才拥有,所以这里只用用 Activity 的 Context 来作为 Dialog 的就可以了;这里因为 Appliation 的 token 是空的,而 Dialog 一开始创建 Winow 的时候 token 要是空的;我们来看一下 Dialog 的一个构造方法,那就是 Dialog(@NonNull Context context, @StyleRes int themeResId, boolean createContextThemeWrapper);

de33e37b9eaf10ae67fc9cd219570269.png

看注释3的代码,第二个参数是不是 null,那就说明 Dialog 的 token 为 null 了;Dialog 的窗口类型为 TYPE_APPLICATION 类型的,要求必需是 Activity 的 Token,不是的话系统会抛出 BadTokenException 异常;Dialog 是应用窗口类型,Token 必须是 Activity 的 Token;我们看一下注释2的代码,假设 context 是 Activity,看 getSystemService 方法的传入的是 Context.WINDOW_SERVICE,好,我们看一下 Activity 的 getSystemService方法;

8e70ae433c9d174f6387864b662f8d69.png

看注释4的代码,nam 为 Context.WINDOW_SERVICE,所以返回的是 mWindowManager ,而 mWindowManager 的实现类是 WindowManagerImpl;那 Activity 是在什么时候设置 token 的呢?答案是在 Activity 的 attach方法;

346695c455cdd33b8af48ef8130b826e.png

看到注释5的代码没,第二个参数就是 Activity 的 token ,好,我们往下看 Window 的 setWindowManager方法;

7afc02d40dc326789524ed2f3d8be50a.png

注释6的代码,只是将 token 保存到 Activity 的 Window 中;看注释7的代码,这里的 this 指的是 Activity 的 PhoneWindow;好,我们往下看 WindowManagerImpl 的 createLocalWindowManager方法;

9055fee69bb3b36939546121eb5cf71f.png

WindowManagerImpl 的 createLocalWindowManager 方法又调用了 WindowManagerImpl(Context context, Window parentWindow)方法;

d8b10ae8c860740853505ee61ce37fc9.png

mParentWindow是什么?一看这名字就知道肯定是某个 Window 的父 Window,也就是说 Activity 的 Window 是某个 Window 的父 Window;那什么时候 Activity 的 token 给了 Dialog 呢?我们看一下 Dialog 的 show 方法;

404832d0e9ae95697b509314b179d472.png

注释9代码其实就是显示 Dialog 的 View 过程,并将 mShowing 置为 true,表示 View 正在显示;好,我们看看注释9代码的实现,mWindowManager 是 WindowManagerImpl,我们看看 addView(@NonNull View view, @NonNull ViewGroup.LayoutParams params)方法;

b25e246989873c03c075fc58204152d1.png

注释10中的 mParentWindow是 Activity 的 PhoneWindow(如果 mContext 是 Activity 的话),mGlobal 是 WindowManagerGlobal,我们看一下 WindowManagerGlobal 的 addView(View view, ViewGroup.LayoutParams params, Display display, Window parent-Window)方法;

2483df6bb06ecf79ec990c1967138fab.png

看注释11的代码,parentWindow 就是 Activity 的 PhoneWindow,所以执行到 Window 的 adjustLayoutParamsForSubWindow(WindowMana-ger.LayoutParams wp)方法;

0335ded631d4e670260c9f87c875241c.png

看到注释12的代码没有,decor本质上是 Activity 中 PhoneWindow 的 DecorView,decor 拿到的是 mAttachInfo.mWindowToken ,而 mAttachInfo.mWindowToken正是 Activity 中 PhoneWindow 的 token,所以在 Dialog 的 show 过程其实也将 Activity 的 token 赋值给了 Dialog。

我们上面不是提到过 Dialog 的 Window 是 TYPE_APPLICATION 类型的窗口吗?怎么证明它是?好,我们回过头来看注释8的代码,也就是 Dialog 的 PhoneWindow 的 getAttributes方法,它在 Window 中实现的;

678d05c5d673cd64faca025ab8047dd7.png

mWindowAttributes 是什么,我们看看 mWindowAttributes的声明;

a322c3c49449b8e05b0e6af275e12e94.png

我们看看 WindowManager.LayoutParams 的无参构造方法;

1c92e49bb9c3b0439a78d97a5a6d910f.png

看到注释13的代码没有,Window 默认使用的 type 是 TYPE_APPLICATION,而 Dialog 的 Window 没有改变 type,所以 Dialog 的 Window 是 TYPE_APPLICATION 类型的窗口。

最后我们看一下 Dialog 的 dismiss 方法的实现;

e817255e40d1c8111af51000ce163ad5.png

看注释14,如果当前线程是主线程,那么直接执行 dismissDialog 方法;看注释15,如果不是主线程,那么就切换到主线程去执行;好,我们继续看 Dialog 的 dismissDialog方法;

8f4100f93d877aef4b1bebaa449cbeee.png

看注释16代码,最终将 DecorView 从 PhoneWindow 中删除就完事了。


http://chatgpt.dhexx.cn/article/7IC33EJS.shtml

相关文章

Android之Dialog分析

Android之Dialog分析 以Dialog为引导,Android的弹出式消息一共是三种(据我所知):Dialog,tocast,notification 其三种弹出式消息各有所长。今天重点是分析其中的Dialog。Android的Dialog是android界面编程的…

Android开发dialog内存泄露,Dialog引发的内存泄漏

8种机械键盘轴体对比 本人程序员,要买一个写代码的键盘,请问红轴和茶轴怎么选? 本文是本人对于 LeakCanary 团队的一篇分析内存泄漏的文章的意译。水平有限,如有不够准确之处,敬请包涵。 主旨:在Lollipop之前的版本,Dialog可能导致内存泄漏。 引言 LeakCanary 提示存在内…

android源码分析-Dialog

今天给大家介绍android -Dialog源码分析 Dialog 是所有对话框的基类,例如AlertDialog,我们要深入了解指导Dialog的用法,逻辑,必须要把Dilaog弄清楚,下面首先我们来看下Google对Dialog的类描述:/**Base clas…

Dialog源码分析

1, 概述 Dialog(对话框)不仅可以显示信息,还可以和Activity界面进行交互,这种交互是阻塞式的. 继承Dialog的类有好几种,主要以AlertDialog为例来分析一下具体的原理。 2 实现 Dialog依附于Activity来实现,一般在acitivty中显示,因为Dialog的交互是阻塞式的,所以最好另开一线…

爬虫为什么会使用到HTTP代理?

在进行网页爬虫的时候使用HTTP代理,可以进行匿名抓取网页信息,爬取大数据等使用方向。HTTP代理我们很了解,但是你有了解过HTTP协议是什么吗? HTTP协议即超文本传输协议,是Internet上信息传输时使用最为广泛的一种简单…

爬虫笔记——全局代理下的requests写法

当PC挂了全局代理的时候,如果你的requests还像下面写的话,大概率会报错: 正确的写法是:

抓包:Android不走代理的请求

测试用例 测试应用有两个按钮,分别用 HttpURLConnection 和 Okhttp3 请求 https://www.baidu.com/。注意:两个请求都加入了 Proxy.NO_PROXY。 //HttpURLConnection请求https://tcc.taobao.com/cc/json/mobile_tel_segment.htm?tel13812371237 public …

python爬虫——selenium+chrome使用代理

先看下本文中的知识点: python selenium库安装chrome webdirver的下载安装seleniumchrome使用代理进阶学习 搭建开发环境: selenium库chrome webdirver谷歌浏览器 >7.9 PS:安装了的同学可以跳过了接着下一步,没安装的同学跟…

python 爬虫 客户端_python爬虫

爬虫简介 什么是爬虫? 爬虫:就是抓取网页数据的程序。 HTTP和HTTPS HTTP协议(HyperText Transfer Protocol,超文本传输协议):是一种发布和接收 HTML页面的方法。 HTTPS(Hypertext Transfer Protocol over Secure Socket Layer)简单…

爬虫从入门到精通(7) | 常见反爬-代理IP的使用

目录 一、 为什么要使用代理IP? 二、代理IP的原理 三、代理IP的作用 四、代理IP的分类 ​ 1.根据代理的协议区分 2.根据匿名程度区分 五、在requests模块中如何设置代理 一、 为什么要使用代理IP? 使用自己本地的IP 利用爬虫技术获取某个网站信息的时…

爬虫手册06 代理池和代理服务器搭建

免费代理池的搭建 参考资料:Python3 网络爬虫开发实战(第二版) 一. 学习目标 : 参考《第9章 代理的使用》9.2节搭建自己私有的代理IP池。 二. 实验环境: 阿里云或腾讯云的服务器,我是在打折和新用户优…

我裂开了,教给他如何搭建和使用代理服务器,他居然用来做这么不正经的事(爬虫,代理ip)

代码是正经代码,但是程序员正不正经就不知道了。 ​ 前言 在使用爬虫对某些网站进行爬取时,为了不让网站发现我们的ip,模拟其他用户ip地址去访问网站。也就相当于间接的去访问网站,流程如图: 我们使用到代理服务器&…

Python爬虫进阶 - win和linux下selenium使用代理

目录 Windows selenium配置 下载地址 Chrome Chromedriver 版本对应关系 实践测试 操作元素 浏览器操作 获取元素信息 鼠标操作 实战demo selenium添加代理 Linux selenium配置 检查服务器环境 下载安装第三方库(最简单版) 实践测试 代码…

Python自助爬虫代理ip模块

短小无比的前言: 代理对于爬虫来说可是很重要的一环,尤其在对于大量数据的时候,一不小心自己ip挂了,要么你换网,要么你等个几小时恢复 之后你上网查阅了种种办法,跨越种种艰难险阻,数以堆计的bu…

python爬虫之requests库使用代理方式

在看这篇文章之前,需要大家掌握的知识技能: python基础html基础http状态码 让我们看看这篇文章中有哪些知识点: get方法post方法header参数,模拟用户data参数,提交数据proxies参数,使用代理进阶学习 安装…

爬虫代理IP哪家好?

前言 随着大数据时代的到来,爬虫已经成了获取数据的必不可少的方式,做过爬虫的想必都深有体会,爬取的时候莫名其妙 IP 就被网站封掉了,毕竟各大网站也不想自己的数据被轻易地爬走。 对于爬虫来说,为了解决封禁 IP 的问题,一个有效的方式就是使用代理,使用代理之后可以让…

Python爬虫基础-使用代理

为什么需要代理? 我们爬取数据的时候,开始可以正常爬取,但是过了一段时间,网站可能就会提示“您的IP访问频率过高”,然后就无法正常访问网站。 这是因为网站采取了反爬策略,某个ip访问频率超过一个阈值后&…

Python爬取代理IP

在一些网页的内容爬取过程中,有时候在单位时间内如果我们发送的请求次数过多,网站就可能会封掉我们的IP地址,这时候为了保证我们的爬虫的正常运行,我们就要使用代理IP。 下面来介绍如何构建自己的IP池。 我们用快代理来获取代理i…

流量数据分析的方法学习

1、看数字和趋势(以电商网站为例) 2、维度分解 3、用户分群(又叫用户画像) 4、转化漏斗 5、行为轨迹 关注行为轨迹,是为了真实了解用户行为。通过大数据手段,还原用户的行为轨迹,有助于增长团队…

恶意流量分析(一)

在分析恶意样本时,样本可能包含网络行为,比如样本从C2服务器上请求下载后续病毒文件。所以对于在病毒分析的角度,对恶意流量的分析也是不可避免的。在这里通过恶意流量习题(malware-traffic-analysis)对恶意流量进行入…