美文网首页
ButterKnife原理分析

ButterKnife原理分析

作者: 梦星夜雨 | 来源:发表于2021-02-18 09:11 被阅读0次

    前言

    ButterKnife又名黄油刀,是一款知名的Andorid框架,通过注解绑定,省去初始化控件等重复工作,简化代码,极大提高工作效率。

    dependencies {
        ......
        implementation 'com.jakewharton:butterknife:8.8.1'
        annotationProcessor 'com.jakewharton:butterknife-compiler:8.8.1'
    }
    
     @BindView(R.id.tv)
        TextView tv;
    
     @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
    
            ButterKnife.bind(this);
        }
    
    @OnClick(R.id.tv)
        public void click(View view) {
            Toast.makeText(this, tv.getText().toString(), Toast.LENGTH_SHORT).show();
        }
    

    使用非常简单,但这里需要注意,使用的时候不能使用加private和static,并且生成的文件必须同包。

    public static Unbinder bind(@NonNull Activity target) {
        View sourceView = target.getWindow().getDecorView();
        return bind(target, sourceView);
      }
     public static Unbinder bind(@NonNull Object target, @NonNull View source) {
        Class<?> targetClass = target.getClass();
        if (debug) Log.d(TAG, "Looking up binding for " + targetClass.getName());
        Constructor<? extends Unbinder> constructor = findBindingConstructorForClass(targetClass);
    
        if (constructor == null) {
          return Unbinder.EMPTY;
        }
    
        //noinspection TryWithIdenticalCatches Resolves to API 19+ only type.
        try {
          return constructor.newInstance(target, source);
        } catch (IllegalAccessException e) {
          throw new RuntimeException("Unable to invoke " + constructor, e);
        } catch (InstantiationException e) {
          throw new RuntimeException("Unable to invoke " + constructor, e);
        } catch (InvocationTargetException e) {
          Throwable cause = e.getCause();
          if (cause instanceof RuntimeException) {
            throw (RuntimeException) cause;
          }
          if (cause instanceof Error) {
            throw (Error) cause;
          }
          throw new RuntimeException("Unable to create binding instance.", cause);
        }
      }
    

    我们从 ButterKnife.bind(this)入手,点进去调用了findBindingConstructorForClass()发现一个构造方法,构造方法不为空,就把这个方法初始化。

    private static Constructor<? extends Unbinder> findBindingConstructorForClass(Class<?> cls) {
        Constructor<? extends Unbinder> bindingCtor = BINDINGS.get(cls);
        if (bindingCtor != null) {
          if (debug) Log.d(TAG, "HIT: Cached in binding map.");
          return bindingCtor;
        }
        String clsName = cls.getName();
        if (clsName.startsWith("android.") || clsName.startsWith("java.")
            || clsName.startsWith("androidx.")) {
          if (debug) Log.d(TAG, "MISS: Reached framework class. Abandoning search.");
          return null;
        }
        try {
          Class<?> bindingClass = cls.getClassLoader().loadClass(clsName + "_ViewBinding");
          //noinspection unchecked
          bindingCtor = (Constructor<? extends Unbinder>) bindingClass.getConstructor(cls, View.class);
          if (debug) Log.d(TAG, "HIT: Loaded binding class and constructor.");
        } catch (ClassNotFoundException e) {
          if (debug) Log.d(TAG, "Not found. Trying superclass " + cls.getSuperclass().getName());
          bindingCtor = findBindingConstructorForClass(cls.getSuperclass());
        } catch (NoSuchMethodException e) {
          throw new RuntimeException("Unable to find binding constructor for " + clsName, e);
        }
        BINDINGS.put(cls, bindingCtor);
        return bindingCtor;
      }
    

    在findBindingConstructorForClass()方法中,我们可以看到,如果这个类是以android、java、androidx开始,那么都会直接返回null,否则就会调用cls.getClassLoader().loadClass(clsName + "_ViewBinding")生成这个类的对象。我们上述代码对应的应该是MainActivity_ViewBinding.class。

     public MainActivity_ViewBinding(final MainActivity target, View source) {
        this.target = target;
    
        View view;
        view = Utils.findRequiredView(source, R.id.tv, "field 'tv' and method 'click'");
        target.tv = Utils.castView(view, R.id.tv, "field 'tv'", TextView.class);
        view7f07008d = view;
        view.setOnClickListener(new DebouncingOnClickListener() {
          @Override
          public void doClick(View p0) {
            target.click(p0);
          }
        });
      }
    
     public static View findRequiredView(View source, @IdRes int id, String who) {
        View view = source.findViewById(id);
        if (view != null) {
          return view;
        }
        String name = getResourceEntryName(source, id);
        throw new IllegalStateException("Required view '"
            + name
            + "' with ID "
            + id
            + " for "
            + who
            + " was not found. If this view is optional add '@Nullable' (fields) or '@Optional'"
            + " (methods) annotation.");
      }
    

    然后调用里面的构造方法,在findRequiredView中调用了findViewById()方法。

    public abstract class DebouncingOnClickListener implements View.OnClickListener {
      static boolean enabled = true;
    
      private static final Runnable ENABLE_AGAIN = new Runnable() {
        @Override public void run() {
          enabled = true;
        }
      };
    
      @Override public final void onClick(View v) {
        if (enabled) {
          enabled = false;
          v.post(ENABLE_AGAIN);
          doClick(v);
        }
      }
    
      public abstract void doClick(View v);
    }
    

    下面同步调用了setOnClickListener也初始化了点击事件,不同的是DebouncingOnClickListener是抽象类,实现了View.OnClickListener接口,在onClick事件中调用了抽象方法doClick(),然后doClick()在外面重写了,这里实际上是模板方法模式,感兴趣的同学可以去这篇文章看看。

    至此,ButterKnife原理解析我们已经分析完毕,至于MainActivity_ViewBinding.class代码的生成则是使用的APT技术。

    相关文章

      网友评论

          本文标题:ButterKnife原理分析

          本文链接:https://www.haomeiwen.com/subject/nuhazktx.html