(2)Class类的基本使用

思考一下:原生类和引用类型如何获取一个class对象,获取了class对象之后,怎么获取这个类有关的成员,比如方法,字段,构造函数,如何获取类中的泛型信息以及修饰符?

于是接下来class内容会按照下面的几个方面来讲解:

  • 如何获取一个class类对象
  • 获取类的修饰符和类型
  • 获取类中的成员信息(比如方法,字段,构造函数)
  • 常见关于class反射错误

1.如何获取一个class类对象

方式1 :Object.getClass()

这种方法只适用于引用类型:

        Class c ="foo".getClass();
        System.out.println("引用类型"+c);
        //字节数据类型
        byte[] bytes =new byte[1024];
        System.out.println("字节数组类型"+bytes.getClass());

        System.out.println("字符串数组:"+new String[]{}.getClass());

输出结果如下:

引用类型class java.lang.String
枚举类型:class com.java.reflect.classes.E
字节数组类型class [B
字符串数组:class [Ljava.lang.String;
方式2 :The .class Syntax

这种方式适用于原生类型和引用类型:

        Class b =boolean.class;
        System.out.println("原型布尔类型:"+b);

        System.out.println("打印流:"+java.io.PrintStream.class);

        System.out.println("int数组:"+int[][].class);
        System.out.println("String数组:"+new String[][]{}.getClass());

输出结果如下:

原型布尔类型:boolean
打印流:class java.io.PrintStream
int数组:class [[I
String数组:class [[Ljava.lang.String;
方式3: Class.forName()

只适合在引用类型,需要给出一个全限定的类的名称

       Class sc = Class.forName("com.java.reflect.classes.RetrievClassTest");
        System.out.println("RetrievClassTest:"+sc.getName());
        //double数组
        Class cDoubleArray = Class.forName("[D");
        System.out.println("dubbo数组:"+cDoubleArray.getName());
        //String数组
        Class cStringArray = Class.forName("[[Ljava.lang.String;");
        System.out.println("String数组"+cStringArray.getName());

输出结果如下:

RetrievClassTest:com.java.reflect.classes.RetrievClassTest
dubbo数组:[D
String数组[[Ljava.lang.String;
方式4: 获取包装类型的原生类型
       System.out.println("Double Type:"+Double.TYPE);
        //跟void.class 一模一样
        System.out.println("Void Type:"+ Void.TYPE);

输出结果如下:

double Type:double
void Type:void
方式5:Methods that Return Classes(返回类的方法)
  • Class.getSuperclass() 返回直接上级父类
        Class superClass = javax.swing.JButton.class.getSuperclass();
        System.out.println("JButton的父类:"+superClass.getName());

输出结果如下:

JButton的父类:javax.swing.AbstractButton
  • Class.getClasses() 返回该类中(包括继承的成员)公开的内部类,内部接口类,内部枚举类
        Class<?>[] publicClass = Character.class.getClasses();
        for(Class merber : publicClass){
            if(merber.isEnum()){
                System.out.println("内部枚举类:"+merber);
            }else if(merber.isInterface()){
                System.out.println("内部接口类:"+merber);
            }else{
                System.out.println("普通内部类:"+merber);
            }
        }

输出结果如下:

普通内部类:class java.lang.Character$Subset
普通内部类:class java.lang.Character$UnicodeBlock
内部枚举类:class java.lang.Character$UnicodeScript
  • Class.getDeclaredClasses() 得到该类中所有声明的类,包括内部类,内部接口,内部枚举,私有的内部类也会存在(不包括继承的内部类)
Class<?>[] allClass = Character.class.getDeclaredClasses();

输出结果如下:这里的输出结果比上面的输出结果多了一个私有的java.lang.Character$CharacterCache内部类

java.lang.Character$CharacterCache
java.lang.Character$Subset
java.lang.Character$UnicodeBlock
java.lang.Character$UnicodeScript
  • java.lang.reflect.Field.getDeclaringClass()
    java.lang.reflect.Method.getDeclaringClass()
    java.lang.reflect.Constructor.getDeclaringClass()都是返回成员所在声明的类,(也就是说该类包含了该成员)
        Field f = System.class.getField("out");
        Class declaringClass = f.getDeclaringClass();
        System.out.println("out字段声明的类:"+declaringClass.getName())

结果如下:

out字段声明的类:java.lang.System
  • Class.getEnclosingClass() 返回一个包围的类(也就是一个内部类调用该方法返回一个外部的类)
        Class stateClass = Thread.State.class;
        System.out.println("Thread.State的声明的类为:"+stateClass);
        System.out.println("Thread.State该类的包围类:"+stateClass.getEnclosingClass());
        System.out.println("对比内部类调用getClass方法:"+stateClass.getClass());

结果为: State为Thread的一个内部类,需要先得到

Thread.State的声明的类为:class java.lang.Thread$State
Thread.State该类的包围类: class java.lang.Thread
对比内部类调用getClass方法:class java.lang.Class

2.获取类的修饰符和类型

一个类可以有一个或多个修饰符,这些修饰符会影响运行时的行为,从下面几个方面讲解:

  • 访问修饰: public, protected, and private
  • 是否需要覆盖的修饰符: abstract
  • 限制只有一个实例的修饰符: static
  • 禁止值修改的修饰符: final
  • 强制执行严格的浮点行为的修饰符: strictfp
  • 注解:Annotations

这里引用官方的例子,该类的作用是是实现了类中的修饰符的扫描

import java.lang.annotation.Annotation;
import java.lang.reflect.Modifier;
import java.lang.reflect.Type;
import java.lang.reflect.TypeVariable;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.ConcurrentNavigableMap;

import static java.lang.System.out;

/**
 * @Project: jdk
 * @description: 类的修饰符扫描
 * @author: sunkang
 * @create: 2018-09-27 09:25
 * @ModificationHistory who      when       What
 **/
public class ExaminingModifiersTypes {

    public static void main(String... args) {

//        args = new String[]{GenericBean.class.getName()};
        args = new String[]{ConcurrentNavigableMap.class.getName()};

//        args = new String[]{"[Ljava.lang.String;"};
        try {
            Class<?> c = Class.forName(args[0]);
            out.format("Class:%n  %s%n%n", c.getCanonicalName());
            out.format("Modifiers:%n  %s%n%n",
                    Modifier.toString(c.getModifiers()));

            out.format("Type Parameters:%n");
            TypeVariable[] tv = c.getTypeParameters();
            if (tv.length != 0) {
                out.format("  ");
                for (TypeVariable t : tv){
                    out.format("%s%n", t.getName());
//                   out.println(Arrays.asList(t.getAnnotatedBounds()));
//                   out.println(Arrays.asList(t.getBounds()));
//                    out.println(t.getGenericDeclaration());
                }

                out.format("%n%n");

            } else {
                out.format("  -- No Type Parameters --%n%n");
            }

            out.format("Implemented Interfaces:%n");
            Type[] intfs = c.getGenericInterfaces();
            if (intfs.length != 0) {
                for (Type intf : intfs)
                    out.format("  %s%n", intf.getTypeName()());
                out.format("%n");
            } else {
                out.format("  -- No Implemented Interfaces --%n%n");
            }

            out.format("Inheritance Path:%n");
            List<Class> l = new ArrayList<Class>();
            printAncestor(c, l);
            if (l.size() != 0) {
                for (Class<?> cl : l)
                    out.format("  %s%n", cl.getCanonicalName());
                out.format("%n");
            } else {
                out.format("  -- No Super Classes --%n%n");
            }

            out.format("Annotations:%n");
            Annotation[] ann = c.getAnnotations();
            if (ann.length != 0) {
                for (Annotation a : ann)
                    out.format("  %s%n", a.toString());
                out.format("%n");
            } else {
                out.format("  -- No Annotations --%n%n");
            }

            // production code should handle this exception more gracefully
        } catch (ClassNotFoundException x) {
            x.printStackTrace();
        }
    }

    private static void printAncestor(Class<?> c, List<Class> l) {
        Class<?> ancestor = c.getSuperclass();
        if (ancestor != null) {
            l.add(ancestor);
            printAncestor(ancestor, l);
        }
    }
}

输出结果如下:

Class:
  java.util.concurrent.ConcurrentNavigableMap

Modifiers:
  public abstract interface

Type Parameters:
  K
V


Implemented Interfaces:
  java.util.concurrent.ConcurrentMap<K, V>
  java.util.NavigableMap<K, V>

Inheritance Path:
  -- No Super Classes --

Annotations:
  -- No Annotations --

下面是几个方法的简介:
Classs.getCanonicalName()是得到基于java规范的类的名称,后面章节会详细讲解该源码的解析部分
Classs.getModifiers()得到该类的修饰符,不同的修饰符的有具体的值进行代表,然后进行累计得到一个总的值,该值返回的是一个int
Modifier.toString(Classs.getModifiers())得到修饰符的名称
Class.getTypeParameters()返回一个有泛型声明的数组对象
TypeVariable.getName()返回泛型的信息
Class.getGenericInterfaces()得到一个泛型描述的数组接口
Type.getTypeName得到该类型的字符串信息,包括泛型的信息
Class.getAnnotations()得到该类的所有的注解类

3.获取类中的成员信息(比如方法,字段,构造函数)

思考:如何得到一个类的信心,第一节已经介绍了 ,那如何得到成员的信息呢,比如得到类中的字段的信息,可以有不同的方法去得到字段的信息,而这些方法有什么区别呢?

  • 从类中加载字段的不同方法的区别
ClassAPI List of members? Inherited members? Private members?
getDeclaredField() no no yes
getField() no yes no
getDeclaredFields() yes no yes
getFields() yes yes no
  • 从类中获取方法的不同方法调用之间的区别
ClassAPI List of members? Inherited members? Private members?
getDeclaredMethod() no no yes
getMethod() no yes no
getDeclaredMethods() yes no yes
getMethods() yes yes no
  • 从类中获取构造函数的不同方法调用之间的区别 (构造函数不存在继承)
ClassAPI List of members? Inherited members? Private members?
getDeclaredConstructor() no N/A1 yes
getConstructor() no N/A1 no
getDeclaredConstructors() yes N/A1 yes
getConstructors() yes N/A1 no

这里在引用官方的例子:


/**
 * @Project: jdk
 * @description:   类的成员扫描
 * @author: sunkang
 * @create: 2018-09-27 10:50
 * @ModificationHistory who      when       What
 **/
public class DiscoverClassMerbers {
    public   class de{
    }

    public static void main(String... args) {
//        args=new String[]{ConcurrentNavigableMap.class.getName(),"CONSTRUCTOR","METHOD","FIELD","CLASS"};
        args=new String[]{String.class.getName(),"CONSTRUCTOR","METHOD","FIELD","CLASS"};
//        args=new String[]{DiscoverClassMerbers.class.getName(),"CONSTRUCTOR","METHOD","FIELD","CLASS"};
//        System.out.println(Arrays.asList(DiscoverClassMerbers.class.getClasses()));
        try {
            Class<?> c = Class.forName(args[0]);
            out.format("Class:%n  %s%n%n", c.getCanonicalName());
            //根据类型的名称截取包的名称
            Package p = c.getPackage();
            out.format("Package:%n  %s%n%n",
                    (p != null ? p.getName() : "-- No Package --"));
            for (int i = 1; i < args.length; i++) {
                switch (ClassMember.valueOf(args[i])) {
                    case CONSTRUCTOR:
                        printMembers(c.getConstructors(), "Constructor");
                        break;
                    case FIELD:
                        printMembers(c.getFields(), "Fields");
                        break;
                    case METHOD:
                        printMembers(c.getMethods(), "Methods");
                        break;
                    case CLASS:
                        printClasses(c);
                        break;
                    case ALL:
                        printMembers(c.getConstructors(), "Constuctors");
                        printMembers(c.getFields(), "Fields");
                        printMembers(c.getMethods(), "Methods");
                        printClasses(c);
                        break;
                    default:
                        assert false;
                }
            }
            // production code should handle these exceptions more gracefully
        } catch (ClassNotFoundException x) {
            x.printStackTrace();
        }
    }

    private static void printMembers(Member[] mbrs, String s) {
        out.format("%s:%n", s);
        for (Member mbr : mbrs) {
            if (mbr instanceof Field){
                out.format("  %s%n", ((Field)mbr).toGenericString());
//                out.format("  %s%n", ((Field)mbr).getDeclaringClass());
            }
            else if (mbr instanceof Constructor)
                out.format("  %s%n", ((Constructor)mbr).toGenericString());
            else if (mbr instanceof Method)
                out.format("  %s%n", ((Method)mbr).toGenericString());
        }
        if (mbrs.length == 0)
            out.format("  -- No %s --%n", s);
        out.format("%n");
    }

    private static void printClasses(Class<?> c) {
        out.format("Classes:%n");
        Class<?>[] clss = c.getClasses();
        for (Class<?> cls : clss)
            out.format("  %s%n", cls.getCanonicalName());
        if (clss.length == 0)
            out.format("  -- No member interfaces, classes, or enums --%n");
        out.format("%n");
    }
}
enum ClassMember { CONSTRUCTOR, FIELD, METHOD, CLASS, ALL }

输出结果如下: 可以发现String 的构造函数重载的有很多,方法中Methods输出结果中,默认隐式包含了Object对象中的方法如果是接口的话,方法是不会默认继承Object对象,这些Object对象的方法并不会输出。这些成员默认继承了Member这个对象

Class:
  java.lang.String

Package:
  java.lang

Constructor:
  public java.lang.String(byte[],int,int)
  public java.lang.String(byte[],java.nio.charset.Charset)
  public java.lang.String(byte[],java.lang.String) throws java.io.UnsupportedEncodingException
  public java.lang.String(byte[],int,int,java.nio.charset.Charset)
  public java.lang.String(byte[],int,int,java.lang.String) throws java.io.UnsupportedEncodingException
  public java.lang.String(java.lang.StringBuilder)
  public java.lang.String(java.lang.StringBuffer)
  public java.lang.String(byte[])
  public java.lang.String(int[],int,int)
  public java.lang.String()
  public java.lang.String(char[])
  public java.lang.String(java.lang.String)
  public java.lang.String(char[],int,int)
  public java.lang.String(byte[],int)
  public java.lang.String(byte[],int,int,int)

Methods:
  public boolean java.lang.String.equals(java.lang.Object)
  public java.lang.String java.lang.String.toString()
  public int java.lang.String.hashCode()
  public int java.lang.String.compareTo(java.lang.String)
  public int java.lang.String.compareTo(java.lang.Object)
  public int java.lang.String.indexOf(java.lang.String,int)
  public int java.lang.String.indexOf(java.lang.String)
  public int java.lang.String.indexOf(int,int)
  public int java.lang.String.indexOf(int)
  public static java.lang.String java.lang.String.valueOf(int)
  public static java.lang.String java.lang.String.valueOf(long)
  public static java.lang.String java.lang.String.valueOf(float)
  public static java.lang.String java.lang.String.valueOf(boolean)
  public static java.lang.String java.lang.String.valueOf(char[])
  public static java.lang.String java.lang.String.valueOf(char[],int,int)
  public static java.lang.String java.lang.String.valueOf(java.lang.Object)
  public static java.lang.String java.lang.String.valueOf(char)
  public static java.lang.String java.lang.String.valueOf(double)
  public char java.lang.String.charAt(int)
  public int java.lang.String.codePointAt(int)
  public int java.lang.String.codePointBefore(int)
  public int java.lang.String.codePointCount(int,int)
  public int java.lang.String.compareToIgnoreCase(java.lang.String)
  public java.lang.String java.lang.String.concat(java.lang.String)
  public boolean java.lang.String.contains(java.lang.CharSequence)
  public boolean java.lang.String.contentEquals(java.lang.CharSequence)
  public boolean java.lang.String.contentEquals(java.lang.StringBuffer)
  public static java.lang.String java.lang.String.copyValueOf(char[])
  public static java.lang.String java.lang.String.copyValueOf(char[],int,int)
  public boolean java.lang.String.endsWith(java.lang.String)
  public boolean java.lang.String.equalsIgnoreCase(java.lang.String)
  public static java.lang.String java.lang.String.format(java.util.Locale,java.lang.String,java.lang.Object...)
  public static java.lang.String java.lang.String.format(java.lang.String,java.lang.Object...)
  public void java.lang.String.getBytes(int,int,byte[],int)
  public byte[] java.lang.String.getBytes(java.nio.charset.Charset)
  public byte[] java.lang.String.getBytes(java.lang.String) throws java.io.UnsupportedEncodingException
  public byte[] java.lang.String.getBytes()
  public void java.lang.String.getChars(int,int,char[],int)
  public native java.lang.String java.lang.String.intern()
  public boolean java.lang.String.isEmpty()
  public static java.lang.String java.lang.String.join(java.lang.CharSequence,java.lang.CharSequence...)
  public static java.lang.String java.lang.String.join(java.lang.CharSequence,java.lang.Iterable<? extends java.lang.CharSequence>)
  public int java.lang.String.lastIndexOf(int)
  public int java.lang.String.lastIndexOf(java.lang.String)
  public int java.lang.String.lastIndexOf(java.lang.String,int)
  public int java.lang.String.lastIndexOf(int,int)
  public int java.lang.String.length()
  public boolean java.lang.String.matches(java.lang.String)
  public int java.lang.String.offsetByCodePoints(int,int)
  public boolean java.lang.String.regionMatches(int,java.lang.String,int,int)
  public boolean java.lang.String.regionMatches(boolean,int,java.lang.String,int,int)
  public java.lang.String java.lang.String.replace(char,char)
  public java.lang.String java.lang.String.replace(java.lang.CharSequence,java.lang.CharSequence)
  public java.lang.String java.lang.String.replaceAll(java.lang.String,java.lang.String)
  public java.lang.String java.lang.String.replaceFirst(java.lang.String,java.lang.String)
  public java.lang.String[] java.lang.String.split(java.lang.String)
  public java.lang.String[] java.lang.String.split(java.lang.String,int)
  public boolean java.lang.String.startsWith(java.lang.String,int)
  public boolean java.lang.String.startsWith(java.lang.String)
  public java.lang.CharSequence java.lang.String.subSequence(int,int)
  public java.lang.String java.lang.String.substring(int)
  public java.lang.String java.lang.String.substring(int,int)
  public char[] java.lang.String.toCharArray()
  public java.lang.String java.lang.String.toLowerCase(java.util.Locale)
  public java.lang.String java.lang.String.toLowerCase()
  public java.lang.String java.lang.String.toUpperCase()
  public java.lang.String java.lang.String.toUpperCase(java.util.Locale)
  public java.lang.String java.lang.String.trim()
  public final void java.lang.Object.wait() throws java.lang.InterruptedException
  public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException
  public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException
  public final native java.lang.Class<?> java.lang.Object.getClass()
  public final native void java.lang.Object.notify()
  public final native void java.lang.Object.notifyAll()
  public default java.util.stream.IntStream java.lang.CharSequence.chars()
  public default java.util.stream.IntStream java.lang.CharSequence.codePoints()

Fields:
  public static final java.util.Comparator<java.lang.String> java.lang.String.CASE_INSENSITIVE_ORDER

Classes:
  -- No member interfaces, classes, or enums --

4.常见关于class反射错误

构造函数是私有时会抛出InstantiationException

public class ClassTrouble {
    public static void main(String... args) {
    try {
        Class<?> c = Class.forName("com.java.reflect.classes.Cls");
        c.newInstance();  // InstantiationException

        // production code should handle these exceptions more gracefully
    } catch (InstantiationException x) {
        x.printStackTrace();
    } catch (IllegalAccessException x) {
        x.printStackTrace();
    } catch (ClassNotFoundException x) {
        x.printStackTrace();
    }
    }
}

class Cls  {
    private Cls() {}
}

由于class.newInstance(),只能调用public的构造函数,但是Cls类中的构造函数私有化,会导致实例失败
java.lang.reflect.AccessibleObject提供了一种控制访问检查,Class这个类并没有继承这个类,所以不能正常的工作,解决方法可以使用 Constructor.newInstance(),Constructor这个对象继承了AccessibleObject这个类,解决方法如下:

public class ClassTrouble {
    public static void main(String... args) {
    try {
        Class<?> c = Class.forName("com.java.reflect.classes.Cls");
//      c.newInstance();  // InstantiationException
        Constructor  constructor = c.getDeclaredConstructor();
        constructor.setAccessible(true);
        constructor.newInstance();
    } catch (InstantiationException x) {
        x.printStackTrace();
    } catch (IllegalAccessException x) {
        x.printStackTrace();
    } catch (ClassNotFoundException x) {
        x.printStackTrace();
    } catch (NoSuchMethodException e) {
        e.printStackTrace();
    } catch (InvocationTargetException e) {
        e.printStackTrace();
    }
    }
}
class Cls  {
    private Cls() {}
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,711评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,079评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,194评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,089评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,197评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,306评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,338评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,119评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,541评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,846评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,014评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,694评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,322评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,026评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,257评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,863评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,895评论 2 351

推荐阅读更多精彩内容