博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
属性动画给target的setXXX回调
阅读量:5825 次
发布时间:2019-06-18

本文共 4449 字,大约阅读时间需要 14 分钟。

在写凯哥作业HenCoder的时候的,发现有个方法setDegree(int degree),并没有被调用啊?那么这个degree永远是0?怎么可能呢?debug一下,发现这个方法是会被调用的。我了个擦,涨见识了。

@SuppressWarnings("unused")    public void setDegree(int degree) {        this.degree = degree;        invalidate();    }复制代码

####HenCoder ?

还没关注凯哥的去关注一下哇http://hencoder.com/,真的挺棒,

freshxu 这个大佬在评论里有回复

@freshxu

@千不许 ObjectAnimator内部的工作机制并不是直接对我们传入的属性名进行操作的,而是会去寻找这个属性名对应的get和set方法,所以你传入degree时,它会自动去寻找你的setDegree方法

虽然知道怎么回事了,但是不亲眼看下代码就不爽,掐指一算,肯定是内部用了反射了呗。走,看下反射的源码去。首先打印一下调用栈,能省很多事。

com.hencoder.hencoderpracticedraw4.practice.Practice13CameraRotateHittingFaceView.setDegree(Practice13CameraRotateHittingFaceView.java:67) android.animation.PropertyValuesHolder.nCallIntMethod(Native Method) android.animation.PropertyValuesHolder.-wrap7(PropertyValuesHolder.java) android.animation.PropertyValuesHolder$IntPropertyValuesHolder.setAnimatedValue(PropertyValuesHolder.java:1241) android.animation.ObjectAnimator.animateValue(ObjectAnimator.java:990) android.animation.ValueAnimator.animateBasedOnTime(ValueAnimator.java:1262) android.animation.ValueAnimator.doAnimationFrame(ValueAnimator.java:1310) android.animation.AnimationHandler.doAnimationFrame(AnimationHandler.java:146) android.animation.AnimationHandler.-wrap2(AnimationHandler.java) android.animation.AnimationHandler$1.doFrame(AnimationHandler.java:54) android.view.Choreographer$CallbackRecord.run(Choreographer.java:874) android.view.Choreographer.doCallbacks(Choreographer.java:688) android.view.Choreographer.doFrame(Choreographer.java:620) android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:862) android.os.Handler.handleCallback(Handler.java:751) android.os.Handler.dispatchMessage(Handler.java:95) android.os.Looper.loop(Looper.java:154) android.app.ActivityThread.main(ActivityThread.java:6222) java.lang.reflect.Method.invoke(Native Method) com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:895) com.android.internal.os.ZygoteInit.main(ZygoteInit.java:785)复制代码

发现走的是一个Native方法的回调

native static private void nCallIntMethod(Object target, long methodID, int arg);复制代码

androidxref搜索源码,这个是属性动画 那么2.3.3肯定不行,选择Android 版本4.0.4

http://androidxref.com/4.0.4/

搜索 nCallIntMethod 找到对应的cpp文件 frameworks/base/core/jni/android_animation_PropertyValuesHolder.cpp

找到对应函数

static void android_animation_PropertyValuesHolder_callIntMethod(    JNIEnv* env, jclass pvhObject, jobject target, jmethodID methodID, int arg) {  env->CallVoidMethod(target, methodID, arg); }复制代码

调用了CallVoidMethod(target, methodID, arg); 也就是调用了target里面的方法,方法标识为methodID,然后将参数arg原封不动的返回了,我去,前面说反射的瞬间打脸。

那么就看看传入这个native method的参数,

this = {PropertyValuesHolder$IntPropertyValuesHolder@4925} "degree:  0  360  "target = {Sample13CameraRotateHittingFaceView@4928} "com.hencoder.hencoderpracticedraw4.sample.Sample13CameraRotateHittingFaceView{36e150e V.ED..... ......ID 0,0-1080,772 #7f0d0089 app:id/model}"mTmpValueArray = {Object[1]@4929} mJniSetter = 547212489000mSetter = nullmIntAnimatedValue = 87mProperty = nullmTmpValueArray[0] = null复制代码

那么这里

  • target就是Sample13CameraRotateHittingFaceView对象,就是创建属性动画对象时传入的this
  • mJniSetter就是方法的标识,setDegree(int degree)
  • mIntAnimatedValue 就是,上面方法setDegree(int degree)中的参数(int degree)

这里知道了这个setDegree(int degree)方法是由native method回调的。至于mJniSetter的标识是如何找到setDegree(int degree)那么就要有一定JNI水平才能理解了,但是很抱歉现在我太菜了,分析就差不多到这吧,起码知道不是用反射,这波不亏。

上层的都是在framework层的java代码调用,仔细一点都可以找到调用的路径就不花时间在这里了。


顺便看下方法的标识mJniSetter是如何产生的。

mJniSetter = nGetMultipleFloatMethod(targetClass, methodName, numParams);复制代码

找到android_animation_PropertyValuesHolder.cpp

static jlong android_animation_PropertyValuesHolder_getMultipleFloatMethod(   JNIEnv* env, jclass pvhClass, jclass targetClass, jstring methodName, jint parameterCount) {     return getMultiparameterMethod(env, targetClass, methodName, parameterCount, 'F'); }复制代码
static jlong getMultiparameterMethod(JNIEnv* env, jclass targetClass, jstring methodName, jint parameterCount, char parameterType) {   const char *nativeString = env->GetStringUTFChars(methodName, 0);   char *signature = new char[parameterCount + 4];   signature[0] = '(';   memset(&(signature[1]), parameterType, parameterCount);   strcpy(&(signature[parameterCount + 1]), ")V");   jmethodID mid = env->GetMethodID(targetClass, nativeString, signature);   delete[] signature;   env->ReleaseStringUTFChars(methodName, nativeString);      return reinterpret_cast
(mid); }复制代码

转载于:https://juejin.im/post/5a31ee415188257a3e4eac0c

你可能感兴趣的文章
find的命令的使用和文件名的后缀
查看>>
Android的Aidl安装方法
查看>>
Linux中rc的含义
查看>>
曾鸣:区块链的春天还没有到来| 阿里内部干货
查看>>
如何通过Dataworks禁止MaxCompute 子账号跨Project访问
查看>>
js之无缝滚动
查看>>
Django 多表联合查询
查看>>
logging模块学习:basicConfig配置文件
查看>>
Golang 使用 Beego 与 Mgo 开发的示例程序
查看>>
+++++++子域授权与编译安装(一)
查看>>
asp.net怎样在URL中使用中文、空格、特殊字符
查看>>
路由器发布服务器
查看>>
实现跨交换机VLAN间的通信
查看>>
jquery中的data-icon和data-role
查看>>
python例子
查看>>
环境变量(总结)
查看>>
ios之UILabel
查看>>
Java基础之String,StringBuilder,StringBuffer
查看>>
1月9日学习内容整理:爬虫基本原理
查看>>
安卓中数据库的搭建与使用
查看>>