屬性動畫給target的setXXX回撥

pdog發表於2017-12-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 = 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); 
}
複製程式碼

相關文章