在写凯哥作业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
搜索 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 = 547212489000
mSetter = null
mIntAnimatedValue = 87
mProperty = null
mTmpValueArray[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<jlong>(mid);
}