源码编译——Xposed源码编译详解
本文深入解析了基于Android 6.0源码环境,实现Xposed框架的码安码源码编译至定制化全过程,提供一套清晰、制源卓绘制源系统的码安码操作指南。实验环境选取了Android 6.0系统,制源卓绘制源旨在探索并解决源码编译过程中遇到的码安码polsarpro源码难点,同时也借助于社区中其他大神的制源卓绘制源宝贵资源,让编译过程更加高效且精准。码安码
致谢部分,制源卓绘制源首先对定制Xposed框架的码安码世界美景大佬致以诚挚的感谢,其提供框架的制源卓绘制源特征修改思路和代码实例给予了深度学习的基础,虽然个人能力有限,码安码未能完整复现所有的制源卓绘制源细节,但通过对比和实践,码安码逐步解决了遇到的制源卓绘制源问题。特别提及的是肉丝大佬的两篇文章,《来自高纬的对抗:魔改XPOSED过框架检测(上)》和《来自高纬的对抗:魔改XPOSED过框架检测(下)》,这两篇文章是本文深入定制Xposed框架的基础指引,通过它们的学习,许多技术细节和解决方案得以明确。
关于Xposed框架编译和配置的技术细节,参考文章《xposed源码编译与集成》提供了清晰的理论框架,而在《学习篇-xposed框架及高版本替代方案》中,能够找到关于Xposed安装、功能验证以及遇到问题时的解决策略,这两篇文档对理解Xposed框架运行机制、安装流程以及后续的调试工作大有裨益。
在编译流程中,我们首先对Xposed框架中的各个核心组件进行详细的解析和功能定位,包括XposedInstaller、XposedBridge、Xposed、kettle网页版源码android_art、以及XposedTools。每一步都精心设计,确保实现模块与Android系统环境的无缝对接。接下来,我们进行具体的编译步骤。
首先是XposedBridge源码的下载,直接从GitHub上获取最新且与Android 6.0版本相适配的代码,这里选择下载Xposed_art。其次,通过Android.mk文件,我们可以配置编译环境,明确哪些源文件需要编译、生成的目标文件类型以及依赖的其他库文件。在Android.mk文件中,要确保针对特定的XposedBridge版本进行参数的调整,避免不必要的错误。
后续的编译过程可通过mmm或Android Studio完成。mmm编译更倾向于手动操作,适合熟悉CMakebuild系统的开发者,而Android Studio提供了一站式的IDE解决方案,操作流程更为便捷且直观。无论是采用哪种编译方式,最终的目标是生成XposedBridge.jar文件,这个文件将成为Xposed框架的核心组件,用于在Android系统上运行模块化的功能。
探索Android开源框架 - 8. Gson使用及源码解析
深入解析Android开源框架中的Gson使用及其源码解析,Gson作为Java语言的高效JSON转换库,以其简洁的API和高性能的特点,成为Android开发中进行数据序列化和反序列化的首选工具。本文将详细介绍Gson的伴奏人声分离源码使用方法,包括基本的解析与生成、属性重命名、POJO与JSON的字段映射规则、泛型的封装、序列化与反序列化过程,以及如何进行字段过滤与自定义序列化器和反序列化器的实现。
1. **基本的解析与生成
**使用Gson进行JSON字符串解析时,可以通过Gson对象的fromJson方法将JSON字符串转换为Java对象,反之,使用toJson方法将Java对象转换为JSON字符串。
2. **属性重命名
**通过使用@SerializedName注解,可以方便地在POJO类中重命名JSON字段,以匹配服务端返回的数据结构,从而避免硬编码的字符串匹配。
3. **POJO与JSON的字段映射规则
**Gson通过构建一个映射规则来匹配JSON字段到POJO类的属性,确保数据的正确解析与生成。这主要通过类型适配器(TypeAdapter)来实现,使得Gson能够理解如何处理复杂数据类型,如日期、集合等。
4. **泛型的封装
**在使用Gson进行序列化和反序列化时,可以通过泛型来保证类型安全,确保不会出现类型转换错误。GsonBuilder提供了一系列方法来实现泛型的封装,使得API调用更为清晰和明确。
5. **Gson的序列化、反序列化
**通过Gson的API,可以轻松实现Java对象到JSON字符串的序列化,以及从JSON字符串反序列化到Java对象的过程。这使得数据在不同系统间传输变得简单高效。
6. **字段过滤
**提供了多种方法进行字段过滤,如何编写公式源码如使用@Expose注解、基于版本的过滤、访问修饰符、以及基于策略的自定义过滤规则,以满足不同的数据处理需求。
7. **TypeAdapter、JsonSerializer与JsonDeserializer
**实战中,可能需要处理一些特殊的数据类型或复杂逻辑,这时可以通过实现JsonSerializer或JsonDeserializer来自定义序列化和反序列化过程。TypeAdapter则可以用于处理复杂类型的序列化。
8. **实战TypeAdapterFactory
**在某些场景下,可以通过实现TypeAdapterFactory来创建自定义的TypeAdapter,从而实现更为灵活的数据处理逻辑。
9. **@JsonAdapter注解
**用于指定自定义的序列化器或反序列化器,简化了实现自定义序列化逻辑的步骤,使得代码更为简洁和易读。
. **源码解析
**通过深入源码分析,可以更好地理解Gson内部的实现机制,如fromJson方法如何获取适配器、getAdapter方法如何选择适配器、Gson的构造方法如何初始化适配器列表,以及如何通过反射机制创建绑定字段等关键逻辑。这不仅有助于解决实际开发中的问题,还能加深对Gson工作的理解,为后续的优化和定制提供基础。
Android-Fragment源码分析
Fragment是Android系统为了提高应用性能和降低资源消耗而引入的一种更轻量级的组件,它允许开发者在同一个Activity中加载多个UI组件,实现页面的切换与回退。Fragment可以看作是Activity的一个子部分,它有自己的生命周期和内容视图。
在实际应用中,全球质量溯源码Fragment可以用于构建动态、可复用的UI组件,例如聊天应用中,左右两边的布局(联系人列表和聊天框)可以分别通过Fragment来实现,通过动态地更换Fragment,达到页面的切换效果,而无需整个页面的刷新或重新加载。
在实现上,v4.Fragment与app.Fragment主要区别在于兼容性。app.Fragment主要面向Android 3.0及以上版本,而v4.Fragment(即支持包Fragment)则旨在提供向下兼容性,支持Android 1.6及更高版本。使用v4.Fragment时,需要继承FragmentActivity并使用getSupportFragmentManager()方法获取FragmentManager对象。尽管从API层面看,两者差异不大,但官方倾向于推荐使用v4.Fragment,以确保更好的兼容性和性能优化。
下面的示例展示了如何使用v4.Fragment实现页面的加载与切换。通过创建Fragment和FragmentActivity,我们可以加载特定的Fragment,并在不同Fragment间进行切换。
在FragmentDemo的布局文件中,定义了Fragment容器。
在Fragment代码中,定义了具体的业务逻辑和视图渲染,如初始化界面数据、响应用户事件等。
在Activity代码中,通过FragmentManager的beginTransaction方法,加载指定的Fragment实例,并在需要时切换到不同Fragment,实现页面的动态更新。
从官方的建议来看,v4.Fragment已经成为推荐的使用方式,因为它在兼容性、性能和功能方面都更优于app.Fragment。随着Android系统的迭代,使用v4.Fragment能确保应用在不同版本的Android设备上均能获得良好的运行效果。
在Fragment的生命周期管理中,Fragment与Activity的生命周期紧密关联。通过FragmentManager的操作,如commit、replace等,可以将Fragment加入到Activity的堆栈中,实现页面的加载与切换。当用户需要返回时,系统会自动将当前Fragment从堆栈中移除,从而实现页面的回退。
深入Fragment源码分析,我们可以了解其如何在底层实现这些功能。Fragment的初始化、加载、切换等过程涉及到多个关键类和方法,如FragmentManager、FragmentTransaction、BackStackRecord等。通过这些组件的协作,Fragment能够实现与Activity的生命周期同步,确保用户界面的流畅性和高效性。
在实际开发中,使用Fragment可以显著提高应用的响应速度和用户体验。通过动态加载和切换不同的Fragment,开发者可以构建出更加灵活、高效的应用架构,同时减少资源的消耗,提高应用的性能。
Android全品类源码大全
以下是Android源码大全的相关内容整理,包含了多个方面的代码资源:
1. Android源码类型丰富多样,包括:
- TextView
- UI布局
- UPnP
- Widget小组件
- WiFi蓝牙
- Win8风格
- XMPP
- 安装与卸载
- 编程知识
- 标签云
- 抽屉效果
- 串口、Socket通讯与USB驱动
- 窗口抖动
- 代码安全
- 导航菜单分类
- 登录与注册
- 地图、导航、定位等
- 电量管理
- 动画效果
- 动态布局
- 短信彩信
- 短信验证
- 对讲机与录音
- 多点触控与手势控制
- 飞行模式
- 富文本编辑器
- 工具与文档
- 刮刮乐
- 广告展示
- 后台服务
- 换肤功能
- 机顶盒应用
- 计算器
- 记事本与备忘录
- 键盘输入
- 截屏功能
- 进度条
- 开发框架
- 开关效果
- 课程表
- 聊天通讯
- 浏览器与相关技术
- 闹钟
- 拍照与录像
- 相关功能扩展
- 跑马灯
- 瀑布流
- 其他功能
- 启动与网络判断
- 切换动画
- 人脸识别
- 日志分析
- 闪光灯
- 社交分享与第三方登录
- 声波通讯与耳机
- 市县联动与多级联动
- 时间轴
- 视频播放与流媒体
- 视图效果与库
- 搜索相关
- 锁屏与安全功能
- 天气日历
- 条码扫描与二维码
- 通讯录与联系人
- 图表报表
- 处理
- 选择与管理
- 加载与缓存
- 编辑功能
- 网站交互与数据传输
- 文档操作
- 文件管理
- 文件下载与上传
- 下拉刷新与上拉加载
- 消息推送
- 悬浮窗
- 验证码
- 摇一摇与重力传感器
- 夜间模式
- 医疗相关功能
- 仪表盘效果
- 音乐播放器与相关
- 引导页面
- 应用更新与管理
- 应用信息
- 邮件相关
- 游戏源码
- 语音识别与文本朗读
- 运营商相关
- 支付示例
- 字母索引
- 自定义控件
- 自适应布局
- 对话框
- DLAN功能
- EditText输入框
- Emoji表情
- Fragment与Tab选项卡
- GIF支持
- GridView相关
- HOME键处理
- iOS风格
- IPCamera应用
- JBox2D相关
- Launcher桌面
- ListView相关
- NFC功能
- OAuth授权
- OCR图像识别
- P2P通信
- PopupWindow
- SD卡管理
- SQLite数据库
- SQL Server与安卓集成
以上资源均提供了下载链接,访问密码为,可以根据需求选择下载。这些代码涵盖了Android开发中的各个模块,对开发者来说是非常宝贵的参考资料。
安卓源代码怎么用?
1. 如何使用网上提供的Android源代码?
首先,确保你的开发环境中安装了Git。在Eclipse中,导航到"File"菜单,选择"Import",然后浏览到包含library的目录并导入。接着,找到samples目录并导入其中的项目。这个过程大约只需要两分钟,包括下载、构建和截图等步骤。
2. 如何用Eclipse运行Android源代码?
在Eclipse中,通过"File"菜单选择"Import",输入"android"并选择相应的项目目录进行导入。
3. Android源码如何使用?
使用Eclipse的"Import"功能将源码导入,就可以打开并开始使用了。
4. 如何读懂Android源代码?
刚开始接触Android源代码时可能会感到困惑,因为网络上或书本上的解释往往不够清晰。这可能是因为人们往往不愿意分享自己的经验和心得。Android软件实际上是用Java语言编写的,加上许多现成的第三方库。它的界面主要是由XML文件构成,这些XML文件使用标准的标签来定义界面元素和功能。
5. 如何运行Android源代码?
如果你已经安装了Eclipse,可以配置Android SDK环境,然后创建一个新的Android项目,并将你的源代码放入其中。选择在手机上运行,即可自动安装到你的设备上。如果觉得麻烦,可以发送给我,我可以帮你运行并获取app文件。
6. 如何打开Android源代码?
在Eclipse中,通过"File"菜单选择"Import",然后在导入面板中选择已存在的项目,指定项目的文件夹。Eclipse会自动扫描并导入工程。
7. 如何着手研究Android源代码?
首先,需要导入整个Android源码库,不能单独导入一个工程。其次,使用git和repo来管理Android源代码,具体步骤如下:
1. 安装Git:`sudo apt-get install git-core`。
2. 安装curl:`sudo apt-get install git-core curl`。
3. 安装Repo,可以直接通过curl将其安装到用户根目录中:`curl | sh`。
8. Android游戏源代码的用途、编写和解析方式是什么?
如果你熟悉Java语言,理解Android游戏的源代码编写和解析将更容易。如果不熟悉,解释可能对你来说并不容易理解。
简述android源代码的编译过程
编译Android源代码是一个相对复杂的过程,涉及多个步骤和工具。下面我将首先简要概括编译过程,然后详细解释每个步骤。
简要
Android源代码的编译过程主要包括获取源代码、设置编译环境、选择编译目标、开始编译以及处理编译结果等步骤。
1. 获取源代码:编译Android源代码的第一步是从官方渠道获取源代码。通常,这可以通过使用Git工具从Android Open Source Project(AOSP)的官方仓库克隆代码来完成。命令示例:`git clone /platform/manifest`。
2. 设置编译环境:在编译之前,需要配置合适的编译环境。这通常涉及安装特定的操作系统(如Ubuntu的某些版本),安装必要的依赖项(如Java开发工具包和Android Debug Bridge),以及配置特定的环境变量等。
3. 选择编译目标:Android支持多种设备和配置,因此编译时需要指定目标。这可以通过选择特定的设备配置文件(如针对Pixel手机的`aosp_arm-eng`)或使用通用配置来完成。选择目标后,编译系统将知道需要构建哪些组件和变种。
4. 开始编译:设置好环境并选择了编译目标后,就可以开始编译过程了。在源代码的根目录下,可以使用命令`make -jN`来启动编译,其中`N`通常设置为系统核心数的1~2倍,以并行处理编译任务,加快编译速度。编译过程中,系统将根据Makefile文件和其他构建脚本,自动下载所需的预构建二进制文件,并编译源代码。
5. 处理编译结果:编译完成后,将在输出目录(通常是`out/`目录)中生成编译结果。这包括可用于模拟器的系统镜像、可用于实际设备的OTA包或完整的系统镜像等。根据需要,可以进一步处理这些输出文件,如打包、签名等。
在整个编译过程中,还可能遇到各种依赖问题和编译错误,需要根据错误信息进行调试和解决。由于Android源代码庞大且复杂,完整的编译可能需要数小时甚至更长时间,因此耐心和合适的硬件配置也是成功编译的重要因素。
Android UIç»å¶ä¹Viewç»å¶çå·¥ä½åç
è¿æ¯AndroidUIç»å¶æµç¨åæç第äºç¯æç« ï¼ä¸»è¦åæçé¢ä¸Viewæ¯å¦ä½ç»å¶å°çé¢ä¸çå ·ä½è¿ç¨ãViewRoot 对åºäº ViewRootImpl ç±»ï¼å®æ¯è¿æ¥ WindowManager å DecorView ç纽带ï¼Viewçä¸å¤§æµç¨åæ¯éè¿ ViewRoot æ¥å®æçãå¨ ActivityThread ä¸ï¼å½ Activity 对象被å建å®æ¯åï¼ä¼å° DecorView æ·»å å° Window ä¸,åæ¶ä¼å建 ViewRootImpl 对象ï¼å¹¶å° ViewRootImpl 对象å DecorView 建ç«å ³èã
measure è¿ç¨å³å®äº View ç宽/é«ï¼ Measure å®æ以åï¼å¯ä»¥éè¿ getMeasuredWidth å getMeasuredHeight æ¹æ³æ¥è·å View æµéåç宽/é«ï¼å¨å ä¹ææçæ åµä¸ï¼å®çåäºViewçæç»ç宽/é«ï¼ä½æ¯ç¹æ®æ åµé¤å¤ã Layout è¿ç¨å³å®äº View çå个顶ç¹çåæ åå®é ç宽/é«ï¼å®æ以åï¼å¯ä»¥éè¿ getTopãgetBottomãgetLeft å getRight æ¥æ¿å°Viewçå个顶ç¹çä½ç½®ï¼å¯ä»¥éè¿ getWidth å getHeight æ¹æ³æ¿å°Viewçæç»å®½/é«ã Draw è¿ç¨å³å®äº View çæ¾ç¤ºï¼åªæ draw æ¹æ³å®æå View çå 容æè½åç°å¨å±å¹ä¸ã
DecorView ä½ä¸ºé¡¶çº§ View ï¼ä¸è¬æ åµä¸ï¼å®å é¨ä¼å å«ä¸ä¸ªç«ç´æ¹åç LinearLayout ï¼å¨è¿ä¸ª LinearLayout éé¢æä¸ä¸ä¸¤ä¸ªé¨åï¼ä¸é¢æ¯æ é¢æ ï¼ä¸é¢æ¯å 容æ ãå¨Activityä¸ï¼æ们éè¿ setContentView æ设置çå¸å±æä»¶å ¶å®å°±æ¯è¢«å å°å 容æ ä¸çï¼èå 容æ id为 content ãå¯ä»¥éè¿ä¸é¢æ¹æ³å¾å° content:ViewGroup content = findViewById(R.android.id.content) ãéè¿ content.getChildAt(0) å¯ä»¥å¾å°è®¾ç½®ç view ã DecorView å ¶å®æ¯ä¸ä¸ª FrameLayout , View å±çäºä»¶é½å ç»è¿ DecorView ï¼ç¶åæä¼ éç»æ们ç View ã
MeasureSpec 代表ä¸ä¸ªä½çintå¼ï¼é«2ä½ä»£è¡¨ SpecMode ,ä½ä½ä»£è¡¨ SpecSize , SpecMode æ¯ææµé模å¼ï¼è SpecSize æ¯æå¨æç§æµé模å¼ä¸çè§æ ¼å¤§å°ã
SpecMode æä¸ç±»ï¼å¦ä¸æ示ï¼
UNSPECIFIED
EXACTLY
AT_MOST
LayoutParamséè¦åç¶å®¹å¨ä¸èµ·æè½å³å®ViewçMeasureSpecï¼ä»èè¿ä¸æ¥å³å®Viewç宽/é«ã
对äºé¡¶çº§Viewï¼å³DecorViewåæ®éViewæ¥è¯´ï¼MeasureSpecç转æ¢è¿ç¨ç¥æä¸åã对äºDecorViewï¼å ¶MeasureSpecç±çªå£ç尺寸åå ¶èªèº«çLayoutParamså ±åç¡®å®ï¼
对äºæ®éViewï¼å ¶MeasureSpecç±ç¶å®¹å¨çMeasureSpecåèªèº«çLayoutparamså ±åå³å®ï¼
MeasureSpecä¸æ¦ç¡®å®ï¼onMeasureå°±å¯ä»¥ç¡®å®Viewçæµé宽/é«ã
å°ç»ä¸ä¸
å½å View ç宽é«éç¨ wrap_content æ¶ï¼ä¸ç®¡ç¶å®¹å¨ç模å¼æ¯ç²¾ç¡®æ¨¡å¼è¿æ¯æ大模å¼ï¼å View ç模å¼æ»æ¯æ大模å¼+ç¶å®¹å¨çå©ä½ç©ºé´ã
View çå·¥ä½æµç¨ä¸»è¦æ¯æ measure ã layout ã draw ä¸å¤§æµç¨ï¼å³æµéãå¸å±ãç»å¶ãå ¶ä¸ measure ç¡®å® View çæµé宽/é«ï¼ layout ç¡®å® view çæç»å®½/é«åå个顶ç¹çä½ç½®ï¼è draw åå° View ç»å¶å¨å±å¹ä¸ã
measure è¿ç¨è¦åæ åµï¼å¦æåªæ¯ä¸ä¸ªåå§ç view ï¼åéè¿ measure æ¹æ³å°±å®æäºå ¶æµéè¿ç¨ï¼å¦ææ¯ä¸ä¸ª ViewGroup ï¼é¤äºå®æèªå·±çæµéè¿ç¨å¤ï¼è¿ä¼éåè°ç¨ææåå ç´ ç measure æ¹æ³ï¼å个åå ç´ åéå½å»æ§è¡è¿ä¸ªæµç¨ã
å¦ææ¯ä¸ä¸ªåå§ç Viewï¼é£ä¹éè¿ measure æ¹æ³å°±å®æäºæµéè¿ç¨ï¼å¨ measure æ¹æ³ä¸ä¼å»è°ç¨ View ç onMeasure æ¹æ³ï¼View ç±»éé¢å®ä¹äº onMeasure æ¹æ³çé»è®¤å®ç°:
å çä¸ä¸ getSuggestedMinimumWidth å getSuggestedMinimumHeight æ¹æ³çæºç ï¼
å¯ä»¥çå°ï¼ getMinimumWidth æ¹æ³è·åçæ¯ Drawable çåå§å®½åº¦ãå¦æåå¨åå§å®½åº¦ï¼å³æ»¡è¶³ intrinsicWidth > 0ï¼ï¼é£ä¹ç´æ¥è¿ååå§å®½åº¦å³å¯ï¼å¦æä¸åå¨åå§å®½åº¦ï¼å³ä¸æ»¡è¶³ intrinsicWidth > 0ï¼ï¼é£ä¹å°±è¿å 0ã
æ¥ççæéè¦ç getDefaultSize æ¹æ³ï¼
å¦æ specMode 为 MeasureSpec.UNSPECIFIED å³æªæå®æ¨¡å¼ï¼é£ä¹è¿åç±æ¹æ³åæ°ä¼ éè¿æ¥ç尺寸ä½ä¸º View çæµé宽度åé«åº¦ï¼
å¦æ specMode ä¸æ¯ MeasureSpec.UNSPECIFIED å³æ¯æ大模å¼æè 精确模å¼ï¼é£ä¹è¿åä» measureSpec ä¸ååºç specSize ä½ä¸º View æµéåç宽度åé«åº¦ã
çä¸ä¸åæçè¡¨æ ¼ï¼
å½ specMode 为 EXACTLY æè AT_MOST æ¶ï¼View çå¸å±åæ°ä¸º wrap_content æè match_parent æ¶ï¼ç» View ç specSize é½æ¯ parentSize ãè¿ä¼æ¯å»ºè®®çæå°å®½é«è¦å¤§ãè¿æ¯ä¸ç¬¦åæ们çé¢æçãå 为æä»¬ç» View 设置 wrap_content æ¯å¸æViewç大å°å好å¯ä»¥å 裹å®çå 容ã
å æ¤ï¼
å¦ææ¯ä¸ä¸ª ViewGroupï¼é¤äºå®æèªå·±ç measure è¿ç¨ä»¥å¤ï¼è¿ä¼éåå»è°ç¨ææåå ç´ ç measure æ¹æ³ï¼å个åå ç´ åéå½å»æ§è¡ measure è¿ç¨ã
ViewGroup 并没æéå View ç onMeasure æ¹æ³ï¼ä½æ¯å®æä¾äº measureChildrenãmeasureChildãmeasureChildWithMargins è¿å 个æ¹æ³ä¸é¨ç¨äºæµéåå ç´ ã
å¦ææ¯ View çè¯ï¼é£ä¹å¨å®ç layout æ¹æ³ä¸å°±ç¡®å®äºèªèº«çä½ç½®ï¼å ·ä½æ¥è¯´æ¯éè¿ setFrame æ¹æ³æ¥è®¾å® View çå个顶ç¹çä½ç½®ï¼å³åå§å mLeft ï¼ mRight ï¼ mTop ï¼ mBottom è¿å个å¼ï¼ï¼ layout è¿ç¨å°±ç»æäºã
å¦ææ¯ ViewGroup çè¯ï¼é£ä¹å¨å®ç layout æ¹æ³ä¸åªæ¯ç¡®å®äº ViewGroup èªèº«çä½ç½®ï¼è¦ç¡®å®åå ç´ çä½ç½®ï¼å°±éè¦éå onLayout æ¹æ³ï¼å¨ onLayout æ¹æ³ä¸ï¼ä¼è°ç¨åå ç´ ç layout æ¹æ³ï¼åå ç´ å¨å®ç layout æ¹æ³ä¸ç¡®å®èªå·±çä½ç½®ï¼è¿æ ·ä¸å±ä¸å±å°ä¼ éä¸å»å®ææ´ä¸ª View æ ç layout è¿ç¨ã
layout æ¹æ³çä½ç¨æ¯ç¡®å® View æ¬èº«çä½ç½®ï¼å³è®¾å® View çå个顶ç¹çä½ç½®ï¼è¿æ ·å°±ç¡®å®äº View å¨ç¶å®¹å¨ä¸çä½ç½®ï¼
onLayout æ¹æ³çä½ç¨æ¯ç¶å®¹å¨ç¡®å®åå ç´ çä½ç½®ï¼è¿ä¸ªæ¹æ³å¨ View ä¸æ¯ç©ºå®ç°ï¼å 为 View 没æåå ç´ äºï¼å¨ ViewGroup ä¸åè¿è¡æ½è±¡åï¼å®çåç±»å¿ é¡»å®ç°è¿ä¸ªæ¹æ³ã
1.ç»å¶èæ¯ï¼ background.draw(canvas); ï¼ï¼
2.ç»å¶èªå·±ï¼ onDraw ï¼ï¼
3.ç»å¶ childrenï¼ dispatchDraw(canvas) ï¼ï¼
4.ç»å¶è£ é¥°ï¼ onDrawScrollBars ï¼ã
dispatchDraw æ¹æ³çè°ç¨æ¯å¨ onDraw æ¹æ³ä¹åï¼ä¹å°±æ¯è¯´ï¼æ»æ¯å ç»å¶èªå·±åç»å¶å View ã
å¯¹äº View ç±»æ¥è¯´ï¼ dispatchDraw æ¹æ³æ¯ç©ºå®ç°çï¼å¯¹äº ViewGroup ç±»æ¥è¯´ï¼ dispatchDraw æ¹æ³æ¯æå ·ä½å®ç°çã
éè¿ dispatchDraw æ¥ä¼ éçã dispatchDraw ä¼éåè°ç¨åå ç´ ç draw æ¹æ³ï¼å¦æ¤ draw äºä»¶å°±ä¸å±ä¸å±ä¼ éäºä¸å»ãdispatchDraw å¨ View ç±»ä¸æ¯ç©ºå®ç°çï¼å¨ ViewGroup ç±»ä¸æ¯çæ£å®ç°çã
å¦æä¸ä¸ª View ä¸éè¦ç»å¶ä»»ä½å 容ï¼é£ä¹å°±è®¾ç½®è¿ä¸ªæ 记为 trueï¼ç³»ç»ä¼è¿è¡è¿ä¸æ¥çä¼åã
å½å建çèªå®ä¹æ§ä»¶ç»§æ¿äº ViewGroup 并ä¸ä¸å ·å¤ç»å¶åè½æ¶ï¼å°±å¯ä»¥å¼å¯è¿ä¸ªæ è®°ï¼ä¾¿äºç³»ç»è¿è¡åç»çä¼åï¼å½æç¡®ç¥éä¸ä¸ª ViewGroup éè¦éè¿ onDraw ç»å¶å 容æ¶ï¼éè¦å ³éè¿ä¸ªæ è®°ã
åèï¼ãAndroidå¼åèºæ¯æ¢ç´¢ã
2024-12-28 15:42
2024-12-28 15:35
2024-12-28 14:22
2024-12-28 14:17
2024-12-28 14:13