I. What is the reflex mechanism?

The JAVA reflection mechanism allows you to know all the properties and methods of any class in the running state. For any object, you can call any of its methods and properties; This ability to dynamically retrieve information and invoke object methods is called the Reflection mechanism of the Java language.


What do we need to know?

The most common classes we encounter in reflection are as follows:

  • Class: A Class object is a special object that is used to create other objects (by which I mean Java classes). A Class object is a.class file generated after a Java Class is compiled, which contains information about the Class.
  • Field: A single Field of a class or interface that provides information about and dynamic access. The reflected fields may be class (static) fields or instance fields.
  • Method: Method methods provide information about and access to individual methods on a class or interface. The reflected method may be a class method or an instance method (including an abstract method).
  • Constructor: Constructor provides the required information about the Constructor of a class.

Three. What methods should we master?

  • (1) Class name. Class; (2) class.forname (Class path); (3) object. GetClass (); The difference between the three methods is that method one does not execute static blocks and dynamic constructors, method two does static blocks and does not execute dynamic constructors, and method three needs to create an object, and both static and dynamic constructors execute.
  • Methods to get class attributes:
    (1) PersionClass.getField (method name); (2) persionClass.getFields(); Method. (3) persionClass getDeclaredField (name); (4) persionClass.getDeclaredFields();Copy the code

    GetField and getFields can only obtain public fields,getDeclaredField and getDeclaredFields can obtain all fields. The one ending in s gets all the fields and returns an array of fields.

  • Methods to get a class method:
    (1) PersionClass.getMethod (method name); (2) persionClass.getMethods(); Method. (3) persionClass getDeclaredMethod (name); (4) persionClass.getDeclaredMethods();Copy the code

    GetMethod and getMethods can only get public modified method names,getDeclaredMethod and getDeclaredMethods can get all methods.

  • There are several ways to get a constructor:
    (1) persionClass. GetConstructor (variable parameter set); (2) persionClass.getConstructors(); Method. (3) persionClass getDeclaredConstructor (name); (4) persionClass.getDeclaredConstructors();Copy the code

    GetConstructor and getConstructors get only the name of a public modified method, while getDeclaredConstructor and getDeclaredConstructors get all methods.

  • Methods for creating objects:

    (1) Constructor. NewInstance; eg:Persion p = (Persion)constructor.newInstance("1");Copy the code
  • Set the property value:
    (1) field.set(Objkect obj,Object value); (2) field.setInt(Objkect obj,int value); . (n) file.setLong(Objkect obj,long value);Copy the code
  • Get the property object:

    (1) field.get(Object); Eg :Person p = (Persion)filed. Get (object);Copy the code

    Notice that if the field modifier is Static you can pass in any value including NULL;

  • Call method: It is important to note that method.setaccessible (true) is used to make the object accessible when the operation is private. Method.invoke (Object obj, parameter) is then called, and the Object of this Object must be an Object of the class. It’s not a class object.


4. Where Android can be used

  • Change the length of the underline in TabLayout.

    For TabLayout, you can change the height and color of the underline, but not the length. The reflection area is used to get the internal control length of the TabLayout. Here only by setting the Margin of each Tab to control the width of the underline, there may be Tab text squeezed situation, can only be used. The code is as follows:
    Class<? extends TabLayout> tabClass = tabLayout.getClass();
         try {
             Field mTabStrip = tabClass.getDeclaredField("mTabStrip");
             mTabStrip.setAccessible(true);
            LinearLayout linearLayout = (LinearLayout) mTabStrip.get(tabLayout);
             for(int i = 0; i < linearLayout.getChildCount(); i++) { View child = linearLayout.getChildAt(i); Child. SetPadding (0,0,0,0); LinearLayout.LayoutParams layoutParams = (LinearLayout.LayoutParams) child.getLayoutParams(); // layoutParams.width = 300; layoutParams.leftMargin = 150; layoutParams.rightMargin = 150; child.setLayoutParams(layoutParams); child.invalidate(); } } catch (NoSuchFieldException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); }Copy the code
  • Control the display time of Toast. Toast internal class TN set display time code:

    mParams.hideTimeoutMilliseconds = mDuration == Toast.LENGTH_LONG ? LONG_DURATION_TIMEOUT : SHORT_DURATION_TIMEOUT; . /** * schedule handleHide into the right thread */ @Override public voidhide() {
             if (localLOGV) Log.v(TAG, "HIDE: " + this);
             mHandler.obtainMessage(HIDE).sendToTarget();
         }Copy the code

    The system Toast only gives us two time choices: SHORT_DURATION_TIMEOUT and LONG_DURATION_TIMEOUT. Fortunately, the system provides a hide method, which is not accessible from the outside. Here we can also use reflection. Most of the operation is done by the internal TN class of Toast. First, the Toast class object contains the final TN mTN field of the inner class, so that we can get the inner class object, and then use the inner class object to get the hide() method in the inner class. The code is as follows:

    try{
                     Class<Toast> toastClass = Toast.class;
                     Field mTN = toastClass.getDeclaredField("mTN"); // Get modifier type toastclass.getModifiers (); mTN.setAccessible(true); Object o = mTN.get(toast); Class<? > aClass = o.getClass(); Method hide = aClass.getDeclaredMethod("hide");
                     hide.setAccessible(true);
                     hide.invoke(o);
                 } catch (NoSuchFieldException e) {
                     e.printStackTrace();
                 } catch (IllegalAccessException e) {
                     e.printStackTrace();
                 } catch (InvocationTargetException e) {
                     e.printStackTrace();
                 } catch (NoSuchMethodException e) {
                     e.printStackTrace();
                 }Copy the code

    V. Concluding remarks

    Reflection is important to us, we don’t have to be very proficient at it, we just have to be very familiar with it. It is relatively easy for students with Java foundation to learn.

    Welcome to watch and put forward valuable comments