Educoder–Java高级特性(第八章)- Java反射【笔记+参考代码】

Educoder–Java高级特性(第八章)- Java反射【笔记+参考代码】


第一关


编程要求
请仔细阅读右侧代码,结合相关知识,在Begin-End
区域内进行代码补充,完成三个方法getPersonClass1()、getPersonClass2()、getPersonClass3()的代码编写,要求分别使用三种方式获取Person类的Class对象并返回。
注意:无需修改main()方法的输出内容。

测试说明 平台会对你编写的代码进行测试:

预期输出:

通过Object 类中的 getClass() 获取的 Class 对象为:class step1.Person 通过静态方法
Class.forName() 获取的 Class 对象为:class step1.Person 通过类字面常量获取 Class
的对象为:class step1.Person



参考代码

package step1;
/**
 * 学员任务文件
 */
public class Reflect_stu {

    public static void main(String[] args) {
        System.out.println("通过Object 类中的 getClass() 获取的 Class 对象为:" + getPersonClass1());
        System.out.println("通过静态方法 Class.forName() 获取的 Class 对象为:" + getPersonClass2());
        System.out.println("通过类字面常量获取 Class 的对象为:" + getPersonClass3());
    }
    /**
     * 通过 Object 类中的 getClass() 获取的 Class 对象
     *
     * @return
     */
    public static Class getPersonClass1() {
        /********** Begin *********/
		Person person = new Person();
		Class c = person.getClass();
        return c;
        /********** End *********/
    }
    /**
     * 通过静态方法 Class.forName() 获取的 Class 对象
     * <p>
     * 注意:Person 类的全路径为: step1.Person
     *
     * @return
     */
    public static Class getPersonClass2() {
        /********** Begin *********/
		Class c = null;
		String classname = "step1.Person";
		try{
			c = Class.forName(classname);一个类的完整路径加名称
		}catch(ClassNotFoundException e){
		}
        return c;
        /********** End *********/
    }
    /**
     * 通过类字面常量获取 Class 的对象
     *
     * @return
     */
    public static Class getPersonClass3() {
        /********** Begin *********/
		Class c = Person.class;
        return c;
        /********** End *********/
    }
}



第二关


编程要求
请仔细阅读右侧代码,结合相关知识,在Begin-End 区域内进行代码补充,打印Apple类的所有public
域、方法和构造器。已分别提供了方法声明printConstructors、printFields、printMethods,请将代码补充完整,且按照打印格式要求输出。

提示:

Method.getReturnType()可以获得方法的返回类型。

打印方法或域的修饰符可以调用提供的printModifiers()方法

打印方法的参数可以调用提供的printParamTypes()方法

Field的getType方法可以获得域类型、getName方法可以获得域的名称

测试说明
预期输出: private java.lang.String name; public step2.Apple(); public
step2.Apple(java.lang.String); public void setName(java.lang.String);

平台会对你编写的代码进行测试。



参考代码

package step2;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;

class Apple {
    private String name;
    public Apple(){}
    public Apple(String name){}
    public void setName(String name) {
        this.name = name;
    }
}
public class Reflect_stu {
    public static void main(String[] args) {
        // 请根据提供的 classPath 获取 step2.Apple 的 Class 对象, 请使用 Class.forName() 方法, 注意捕获异常
        // 通关之后,你也可以修改 clasapath 为其他类路径,分析某个类的能力, 例如: java.util.Date
        String classPath = "step2.Apple";
        Class clazz = null;
        /********** Begin *********/
		try{
			clazz = Class.forName(classPath);
		}catch(ClassNotFoundException e){	
		}
        /********** End *********/
        printFields(clazz);
        printConstructors(clazz);
        printMethods(clazz);
    }
  	/**
     * 请打印类的每个域,输出格式为:修饰符 类型 变量名;
     * @param clazz
     */
    public static void printFields(Class clazz) {
        /********** Begin *********/
		Field[] f = clazz.getDeclaredFields();
		for(Field ff:f){
			Class type = ff.getType();
			int mod = ff.getModifiers();
			System.out.print(Modifier.toString(mod)+" ");
			System.out.print(type.getName()+" ");
			System.out.println(ff.getName()+";");
		}
        /********** End *********/
    }
    /**
     * 打印构造函数,输出格式为:修饰符 方法名称(参数)
     * @param clazz
     */
    public static void printConstructors(Class clazz) {
 		Constructor[] constructors = clazz.getDeclaredConstructors();
        for (Constructor constructor : constructors) {
            Class[] paramTypes = constructor.getParameterTypes();
            /********** Begin *********/
            String name = constructor.getName();
            String modifiers = Modifier.toString(constructor.getModifiers());
            if (modifiers.length() > 0) {
                System.out.print(modifiers + " ");
            }
            System.out.print(name + "(");
            /********** End *********/
            printParamTypes(paramTypes);
        }
    }
    /**
     * 请针对每个方法打印其签名,格式为:修饰符 返回值类型 方法名称(参数);
     * @param clazz
     */
    public static void printMethods(Class clazz) {
        Method[] methos = clazz.getDeclaredMethods();
        for (Method method : methos) {
            Class[] paramTypes = null;
            /********** Begin *********/
			String name = method.getName();
            Class returnType = method.getReturnType();
            String modifiers = Modifier.toString(method.getModifiers());
            System.out.print(modifiers+" "+returnType.getName() + " " + name + "(");
            paramTypes = method.getParameterTypes();
            /********** End *********/
            printParamTypes(paramTypes);
        }
    }
    /**
     * 打印方法参数
     * @param paramTypes
     */
    private static void printParamTypes(Class[] paramTypes) {
        for (int j = 0; j < paramTypes.length; ++j) {
            if (j > 0) {
                System.out.print(",");
            }
            System.out.print(paramTypes[j].getName());
        }
        System.out.println(");");
    }
}



第三关


编程要求
请仔细阅读右侧代码,结合相关知识,在Begin-End 区域内进行代码补充,完成通用toString()方法。

提示:

快速设置访问权限: AccessibleObject.setAccessible(fields, true);
获得所有域:Class.getDeclaredFields()
测试说明
平台会对你编写的代码进行测试。

示例:

public static void toString(Object obj) {
// 请完成代码 } public static void main(String[] args) {
Person person = new Person(123, 19, 175);
toString(person); } 预期输出: [weight=[value=123],age=[value=19],height=[value=175.0]]



参考代码

package step3;

import java.lang.reflect.AccessibleObject;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;

public class Reflect_stu {
    public static String toString(Object obj) {
        Class cl = obj.getClass();
        String r = "";
        r += "[";
        // 请获取所有 Field 并设置访问权限为 true
        /********** Begin *********/
        Field[] fields = cl.getDeclaredFields();
        AccessibleObject.setAccessible(fields, true);
        /********** End *********/
        for (Field f : fields) {
            // 此处 if,逻辑为判断 Field 域是否为非静态域
            if (!Modifier.isStatic(f.getModifiers())) {
                if (!r.endsWith("[")) r += ",";
                r += f.getName() + "=";
                try {
                    // 请获取域的类型及值
                    /********** Begin *********/
                    Class t = f.getType();
                    Object val = f.get(obj);
                    /********** End *********/
                    // isPrimitive() 用于判断是否为基本数据类型,若为基础数据类型直接拼接,否则递归调用 toString 方法
                    if (t.isPrimitive()) r += val;
                    else r += toString(val);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }
        r += "]";
        return r;
    }
    public static void main(String[] args) {
        Person person = new Person(88, 19, 175);
        System.out.println(toString(person));
    }
}
class Person {
    public Integer weight;
    private Integer age;
    private Double height;
    public Person(Integer weight, Integer age, double height) {
        this.weight = weight;
        this.age = age;
        this.height = height;
    }
}



第四关


编程要求
请仔细阅读右侧代码,结合相关知识,在Begin-End 区域内进行代码补充,使用反射调用 Apple 类的
setPrice()方法,设置苹果价格为 14,并打印价格。接着还要用反射去调用getTotal方法获取单价为 20,数量 24
的总金额并打印。

测试说明
预期输出:
14.0
480.0

平台会对你编写的代码进行测试。



参考代码

package step4;

import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;


public class Reflect_stu {
    public static void main(String[] args) throws InvocationTargetException {
        //使用反射调用
        Class clazz = null;
        try {
            clazz = Class.forName("step4.Apple");
            /********** Begin *********/
            Method setPriceMethod = clazz.getMethod("setPrice", double.class);
            Constructor appleConstructor = clazz.getConstructor();
            Object apple = appleConstructor.newInstance();
            setPriceMethod.invoke(apple, 14);
            Method getPriceMethod = clazz.getMethod("getPrice");
            System.out.println(getPriceMethod.invoke(apple));
            Method getTotal = clazz.getMethod("getTotal", double.class, int.class);
            System.out.println(getTotal.invoke(apple, 20, 24));
            /********** End *********/
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

}
class Apple {
    private double price;
    private int count;
    public Apple() {
    }
    public double getPrice() {
        return price;
    }

    public void setPrice(double price) {
        this.price = price;
    }

    public int getCount() {
        return count;
    }

    public void setCount(int count) {
        this.count = count;
    }

    public double getTotal(double price, int count) {
        return price * count;
    }
}

猜你喜欢

转载自blog.csdn.net/weixin_44177494/article/details/106202778