Android hook技术实现一键换肤,Android面试你必须要知道的那些知识

} else {//权限定名走这里

view = createView(name, null, attrs);

}

} finally {

mConstructorArgs[0] = lastContext;

}

}

这里的两个方法onCreateView(parent, name, attrs)createView(name, null, attrs);都最终索引到:

public final View createView(String name, String prefix, AttributeSet attrs)

throws ClassNotFoundException, InflateException {

Constructor<? extends View> constructor = sConstructorMap.get(name);

if (constructor != null && !verifyClassLoader(constructor)) {

constructor = null;

sConstructorMap.remove(name);

}

Class<? extends View> clazz = null;

try {

Trace.traceBegin(Trace.TRACE_TAG_VIEW, name);

if (constructor == null) {

// Class not found in the cache, see if it’s real, and try to add it

clazz = mContext.getClassLoader().loadClass(

prefix != null ? (prefix + name) : name).asSubclass(View.class);

if (mFilter != null && clazz != null) {

boolean allowed = mFilter.onLoadClass(clazz);

if (!allowed) {

failNotAllowed(name, prefix, attrs);

}

}

constructor = clazz.getConstructor(mConstructorSignature);

constructor.setAccessible(true);

sConstructorMap.put(name, constructor);

} else {

// If we have a filter, apply it to cached constructor

if (mFilter != null) {

// Have we seen this name before?

Boolean allowedState = mFilterMap.get(name);

if (allowedState == null) {

// New class – remember whether it is allowed

clazz = mContext.getClassLoader().loadClass(

prefix != null ? (prefix + name) : name).asSubclass(View.class);

boolean allowed = clazz != null && mFilter.onLoadClass(clazz);

mFilterMap.put(name, allowed);

if (!allowed) {

failNotAllowed(name, prefix, attrs);

}

} else if (allowedState.equals(Boolean.FALSE)) {

failNotAllowed(name, prefix, attrs);

}

}

}

Object lastContext = mConstructorArgs[0];

if (mConstructorArgs[0] == null) {

// Fill in the context if not already within inflation.

mConstructorArgs[0] = mContext;

}

Object[] args = mConstructorArgs;

args[1] = attrs;

final View view = constructor.newInstance(args); // 真正需要关注的关键代码,就是这一行,执行了构造函数,返回了一个View对象

if (view instanceof ViewStub) {

// Use the same context when inflating ViewStub later.

final ViewStub viewStub = (ViewStub) view;

viewStub.setLayoutInflater(cloneInContext((Context) args[0]));

}

mConstructorArgs[0] = lastContext;

return view;

} catch (NoSuchMethodException e) {

·····

}

}

这么一大段好像有点让人害怕。其实真正需要关注的,就是反射的代码,最后的 newInstance().

OK,Activity上那些丰富多彩的View的来源,就说到这里, 如果有看不懂的,欢迎留言探讨. ( ̄▽ ̄) !

  • app中资源文件大管家 Resources / AssetManager 是怎么工作的

从我们的终极目的出发:我们要做的是“换肤”,如果我们拿到了要换肤的View,可以对他们进行setXXX属性来改变UI,那么属性值从哪里来?

界面元素丰富多彩,但是这些View,都是用资源文件来进行 "装扮"出来的,资源文件大致可以分为:

图片,文字,颜色,声音视频,字体等。如果我们控制了资源文件,那么是不是有能力对界面元素进行set某某属性来进行“再装扮”呢? 当然,这是可行的。因为,我们平时拿到一个TextView,就能对它进行setTextColor,这种操作,在view还存活的时候,都可以进行操作,并且这种操作,并不会造成Activity的重启。

这些资源文件,有一个统一的大管家。可能有人说是R.java文件,它里面统筹了所有的资源文件int值.没错,但是这个R文件是如何产生作用的呢? 答案:Resources.

本来这里应该写上源码追踪记录的,但是由于 源码无法追踪,原因暂时还没找到,之前追查setContentView(R.layout.xxxx)的时候还可以debug,现在居然不行了,很诡异!

**答案找到了:因为我使用的是 真机,一般手机厂商都会对原生系统进行修改,然后将系统写到到真机里面。

而,我们debug,用的是原生SDK。 用实例来说,我本地是SDK 27的源码,真机也是27的系统,但是真机的运行起来的系统的代码,是被厂家修改了的,和我本地的必然有所差别,所以,有些代码报红,就很正常了,无法debug也很正常。**

既然如此,那我就直接写结论了,一张图说明一切:


5. “全app一键换肤” Demo源码详解(戳这里获得源码)

  • 项目工程结构:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

  • 关键类 SkinFactory
`SkinFactory`类, 继承LayoutInflater.Factory2 ,它的实例,会负责创建View,收集 支持换肤的view

import android.content.Context;

import android.content.res.TypedArray;

import android.support.v7.app.AppCompatDelegate;

import android.text.TextUtils;

import android.util.AttributeSet;

import android.view.LayoutInflater;

import android.view.View;

import android.widget.TextView;

import com.enjoy02.skindemo.R;

import com.enjoy02.skindemo.view.ZeroView;

import java.lang.reflect.Constructor;

import java.util.ArrayList;

import java.util.HashMap;

import java.util.List;

public class SkinFactory implements LayoutInflater.Factory2 {

private AppCompatDelegate mDelegate;//预定义一个委托类,它负责按照系统的原有逻辑来创建view

private List listCacheSkinView = new ArrayList<>();//我自定义的list,缓存所有可以换肤的View对象

/**

  • 给外部提供一个set方法

  • @param mDelegate

*/

public void setDelegate(AppCompatDelegate mDelegate) {

this.mDelegate = mDelegate;

}

/**

  • Factory2 是继承Factory的,所以,我们这次是主要重写Factory的onCreateView逻辑,就不必理会Factory的重写方法了

  • @param name

  • @param context

  • @param attrs

  • @return

*/

@Override

public View onCreateView(String name, Context context, AttributeSet attrs) {

return null;

}

/**

  • @param parent

  • @param name

  • @param context

  • @param attrs

  • @return

*/

@Override

public View onCreateView(View parent, String name, Context context, AttributeSet attrs) {

// TODO: 关键点1:执行系统代码里的创建View的过程,我们只是想加入自己的思想,并不是要全盘接管

View view = mDelegate.createView(parent, name, context, attrs);//系统创建出来的时候有可能为空,你问为啥?请全文搜索 “标记标记,因为” 你会找到你要的答案

if (view == null) {//万一系统创建出来是空,那么我们来补救

try {

if (-1 == name.indexOf(‘.’)) {//不包含. 说明不带包名,那么我们帮他加上包名

view = createViewByPrefix(context, name, prefixs, attrs);

} else {//包含. 说明 是权限定名的view name,

view = createViewByPrefix(context, name, null, attrs);

}

} catch (Exception e) {

e.printStackTrace();

}

}

//TODO: 关键点2 收集需要换肤的View

collectSkinView(context, attrs, view);

return view;

}

/**

  • TODO: 收集需要换肤的控件

  • 收集的方式是:通过自定义属性isSupport,从创建出来的很多View中,找到支持换肤的那些,保存到map中

*/

private void collectSkinView(Context context, AttributeSet attrs, View view) {

// 获取我们自己定义的属性

TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.Skinable);

boolean isSupport = a.getBoolean(R.styleable.Skinable_isSupport, false);

if (isSupport) {//找到支持换肤的view

final int Len = attrs.getAttributeCount();

HashMap<String, String> attrMap = new HashMap<>();

for (int i = 0; i < Len; i++) {//遍历所有属性

String attrName = attrs.getAttributeName(i);

String attrValue = attrs.getAttributeValue(i);

attrMap.put(attrName, attrValue);//全部存起来

}

SkinView skinView = new SkinView();

skinView.view = view;

skinView.attrsMap = attrMap;

listCacheSkinView.add(skinView);//将可换肤的view,放到listCacheSkinView中

}

}

/**

  • 公开给外界的换肤入口

*/

public void changeSkin() {

for (SkinView skinView : listCacheSkinView) {

skinView.changeSkin();

}

}

static class SkinView {

View view;

HashMap<String, String> attrsMap;

/**

  • 真正的换肤操作

*/

public void changeSkin() {

if (!TextUtils.isEmpty(attrsMap.get(“background”))) {//属性名,例如,这个background,text,textColor…

int bgId = Integer.parseInt(attrsMap.get(“background”).substring(1));//属性值,R.id.XXX ,int类型,

// 这个值,在app的一次运行中,不会发生变化

String attrType = view.getResources().getResourceTypeName(bgId); // 属性类别:比如 drawable ,color

if (TextUtils.equals(attrType, “drawable”)) {//区分drawable和color

view.setBackgroundDrawable(SkinEngine.getInstance().getDrawable(bgId));//加载外部资源管理器,拿到外部资源的drawable

} else if (TextUtils.equals(attrType, “color”)) {

view.setBackgroundColor(SkinEngine.getInstance().getColor(bgId));

}

}

if (view instanceof TextView) {

if (!TextUtils.isEmpty(attrsMap.get(“textColor”))) {

int textColorId = Integer.parseInt(attrsMap.get(“textColor”).substring(1));

((TextView) view).setTextColor(SkinEngine.getInstance().getColor(textColorId));

}

}

//那么如果是自定义组件呢

if (view instanceof ZeroView) {

//那么这样一个对象,要换肤,就要写针对性的方法了,每一个控件需要用什么样的方式去换,尤其是那种,自定义的属性,怎么去set,

// 这就对开发人员要求比较高了,而且这个换肤接口还要暴露给 自定义View的开发人员,他们去定义

// …

}

}

}

/**

  • 所谓hook,要懂源码,懂了之后再劫持系统逻辑,加入自己的逻辑。

  • 那么,既然懂了,系统的有些代码,直接拿过来用,也无可厚非。

*/

//*下面一大片,都是从源码里面抄过来的,并不是我自主设计

// 你问我抄的哪里的?到 AppCompatViewInflater类源码里面去搜索:view = createViewFromTag(context, name, attrs);

static final Class<?>[] mConstructorSignature = new Class[]{Context.class, AttributeSet.class};//

final Object[] mConstructorArgs = new Object[2];//View的构造函数的2个"实"参对象

private static final HashMap<String, Constructor<? extends View>> sConstructorMap = new HashMap<String, Constructor<? extends View>>();//用映射,将View的反射构造函数都存起来

static final String[] prefixs = new String[]{//安卓里面控件的包名,就这么3种,这个变量是为了下面代码里,反射创建类的class而预备的

“android.widget.”,

“android.view.”,

“android.webkit.”

};

/**

  • 反射创建View

  • @param context

  • @param name

  • @param prefixs

  • @param attrs

  • @return

*/

private final View createViewByPrefix(Context context, String name, String[] prefixs, AttributeSet attrs) {

Constructor<? extends View> constructor = sConstructorMap.get(name);

Class<? extends View> clazz = null;

if (constructor == null) {

try {

if (prefixs != null && prefixs.length > 0) {

for (String prefix : prefixs) {

clazz = context.getClassLoader().loadClass(

prefix != null ? (prefix + name) : name).asSubclass(View.class);//控件

if (clazz != null) break;

}

} else {

if (clazz == null) {

clazz = context.getClassLoader().loadClass(name).asSubclass(View.class);

}

}

if (clazz == null) {

return null;

}

constructor = clazz.getConstructor(mConstructorSignature);//拿到 构造方法,

} catch (Exception e) {

e.printStackTrace();

return null;

}

constructor.setAccessible(true);//

sConstructorMap.put(name, constructor);//然后缓存起来,下次再用,就直接从内存中去取

}

Object[] args = mConstructorArgs;

args[1] = attrs;

try {

//通过反射创建View对象

final View view = constructor.newInstance(args);//执行构造函数,拿到View对象

return view;

} catch (Exception e) {

e.printStackTrace();

}

return null;

}

//**********************************************************************************************

}

关键类 SkinEngine

import android.content.Context;

import android.content.pm.PackageInfo;

import android.content.pm.PackageManager;

import android.content.res.AssetManager;

import android.content.res.Resources;

import android.graphics.drawable.Drawable;

import android.support.v4.content.ContextCompat;

import android.util.Log;

import java.io.File;

import java.lang.reflect.Method;

public class SkinEngine {

//单例

private final static SkinEngine instance = new SkinEngine();

public static SkinEngine getInstance() {

return instance;

}

private SkinEngine() {

}

public void init(Context context) {

mContext = context.getApplicationContext();

//使用application的目的是,如果万一传进来的是Activity对象

//那么它被静态对象instance所持有,这个Activity就无法释放了

}

private Resources mOutResource;// TODO: 资源管理器

private Context mContext;//上下文

private String mOutPkgName;// TODO: 外部资源包的packageName

/**

  • TODO: 加载外部资源包

*/

public void load(final String path) {//path 是外部传入的apk文件名

File file = new File(path);

if (!file.exists()) {

return;

}

//取得PackageManager引用

PackageManager mPm = mContext.getPackageManager();

//“检索在包归档文件中定义的应用程序包的总体信息”,说人话,外界传入了一个apk的文件路径,这个方法,拿到这个apk的包信息,这个包信息包含什么?

PackageInfo mInfo = mPm.getPackageArchiveInfo(path, PackageManager.GET_ACTIVITIES);

mOutPkgName = mInfo.packageName;//先把包名存起来

AssetManager assetManager;//资源管理器

try {

//TODO: 关键技术点3 通过反射获取AssetManager 用来加载外面的资源包

assetManager = AssetManager.class.newInstance();//反射创建AssetManager对象,为何要反射?使用反射,是因为他这个类内部的addAssetPath方法是hide状态

//addAssetPath方法可以加载外部的资源包

Method addAssetPath = assetManager.getClass().getMethod(“addAssetPath”, String.class);//为什么要反射执行这个方法?因为它是hide的,不直接对外开放,只能反射调用

addAssetPath.invoke(assetManager, path);//反射执行方法

mOutResource = new Resources(assetManager,//参数1,资源管理器

mContext.getResources().getDisplayMetrics(),//这个好像是屏幕参数

mContext.getResources().getConfiguration());//资源配置

//最终创建出一个 "外部资源包"mOutResource ,它的存在,就是要让我们的app有能力加载外部的资源文件

} catch (Exception e) {

e.printStackTrace();

}

}

/**

  • 提供外部资源包里面的颜色

  • @param resId

  • @return

*/

public int getColor(int resId) {

if (mOutResource == null) {

return resId;

}

String resName = mOutResource.getResourceEntryName(resId);

int outResId = mOutResource.getIdentifier(resName, “color”, mOutPkgName);

if (outResId == 0) {

return resId;

}

return mOutResource.getColor(outResId);

}

/**

  • 提供外部资源包里的图片资源

  • @param resId

  • @return

*/

public Drawable getDrawable(int resId) {//获取图片

if (mOutResource == null) {

return ContextCompat.getDrawable(mContext, resId);

}

String resName = mOutResource.getResourceEntryName(resId);

int outResId = mOutResource.getIdentifier(resName, “drawable”, mOutPkgName);

if (outResId == 0) {

return ContextCompat.getDrawable(mContext, resId);

}

return mOutResource.getDrawable(outResId);

}

//… 这里还可以提供外部资源包里的String,font等等等,只不过要手动写代码来实现getXX方法

}

  • 关键类的调用方式

1. 初始化"换肤引擎"

public class MyApp extends Application {

@Override

public void onCreate() {

super.onCreate();

//初始化换肤引擎

SkinEngine.getInstance().init(this);

}

}

2. 劫持 系统创建view的过程

public class BaseActivity extends AppCompatActivity {

@Override

protected void onCreate(Bundle savedInstanceState) {

// TODO: 关键点1:hook(劫持)系统创建view的过程

if (ifAllowChangeSkin) {

mSkinFactory = new SkinFactory();

mSkinFactory.setDelegate(getDelegate());

LayoutInflater layoutInflater = LayoutInflater.from(this);

layoutInflater.setFactory2(mSkinFactory);//劫持系统源码逻辑

}

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
img

最后

总之啊,家里没矿的同学们,如果你们想以后的日子过得好一些,多想想你们的业余时间怎么安排吧;

技术方面的提升肯定是重中之重,但是技术外的一些“软实力”也不能完全忽视,很多时候升职确实是因为你的技术足够强,但也与你的“软实力”密切相关

在这我也分享一份大佬自己收录整理的 Android学习PDF+架构视频+面试文档+源码笔记 ,还有高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料这些都是我闲暇还会反复翻阅并给下属员工学习的精品资料。在脑图中,每个知识点专题都配有相对应的实战项目,可以有效的帮助大家掌握知识点。

总之也是在这里帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习

相信自己,没有做不到的,只有想不到的

图片转存中…(img-N2UisteF-1711726463879)]
[外链图片转存中…(img-6DqGO2rs-1711726463879)]
[外链图片转存中…(img-xvseGjPZ-1711726463879)]
[外链图片转存中…(img-V4MQKdW1-1711726463880)]
[外链图片转存中…(img-78gqtzi8-1711726463880)]
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
[外链图片转存中…(img-PCullwiT-1711726463880)]

最后

总之啊,家里没矿的同学们,如果你们想以后的日子过得好一些,多想想你们的业余时间怎么安排吧;

技术方面的提升肯定是重中之重,但是技术外的一些“软实力”也不能完全忽视,很多时候升职确实是因为你的技术足够强,但也与你的“软实力”密切相关

在这我也分享一份大佬自己收录整理的 Android学习PDF+架构视频+面试文档+源码笔记 ,还有高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料这些都是我闲暇还会反复翻阅并给下属员工学习的精品资料。在脑图中,每个知识点专题都配有相对应的实战项目,可以有效的帮助大家掌握知识点。

总之也是在这里帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习

[外链图片转存中…(img-QTsBjhZ1-1711726463881)]

[外链图片转存中…(img-F7mARC7A-1711726463881)]

相信自己,没有做不到的,只有想不到的

本文已被CODING开源项目:《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》收录

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值