Java 中文官方教程 2022 版(四十四)(3)

简介: Java 中文官方教程 2022 版(四十四)

Java 中文官方教程 2022 版(四十四)(2)https://developer.aliyun.com/article/1488316

课程:数组和枚举类型

原文:docs.oracle.com/javase/tutorial/reflect/special/index.html

从 Java 虚拟机的角度看,数组和枚举类型(或枚举)只是类。许多 Class 中的方法可以用于它们。反射为数组和枚举提供了一些特定的 API。本课程使用一系列代码示例来描述如何区分这些对象与其他类,并对其进行操作。还将检查各种错误。

数组

数组有一个组件类型和一个长度(长度不是类型的一部分)。数组可以整体操作,也可以逐个组件操作。反射为后者提供了 java.lang.reflect.Array 类。

  • 识别数组类型 描述了如何确定类成员是否是数组类型的字段
  • 创建新数组 演示了如何创建具有简单和复杂组件类型的新数组实例
  • 获取和设置数组及其组件 展示了如何访问数组类型的字段以及单独访问数组元素
  • 故障排除 包括常见错误和编程误解

枚举类型

在反射代码中,枚举类型与普通类非常相似。Class.isEnum() 可以告诉一个 Class 是否表示一个 enumClass.getEnumConstants() 可以检索在枚举中定义的枚举常量。java.lang.reflect.Field.isEnumConstant() 表示一个字段是否是一个枚举类型。

  • 检查枚举 演示了如何检索枚举的常量以及任何其他字段、构造函数和方法
  • 使用枚举类型获取和设置字段 展示了如何设置和获取具有枚举常量值的字段
  • 故障排除 描述了与枚举相关的常见错误

数组

原文:docs.oracle.com/javase/tutorial/reflect/special/array.html

一个数组是引用类型的对象,包含固定数量的相同类型的组件;数组的长度是不可变的。创建数组的实例需要知道长度和组件类型。每个组件可以是原始类型(如byteintdouble),引用类型(如StringObjectjava.nio.CharBuffer),或者是数组。多维数组实际上只是包含数组类型组件的数组。

数组在 Java 虚拟机中实现。数组上的唯一方法是从Object继承的方法。数组的长度不是其类型的一部分;数组有一个length字段,可以通过java.lang.reflect.Array.getLength()访问。

反射提供了访问数组类型和数组组件类型、创建新数组以及检索和设置数组组件值的方法。以下各节包括对数组上常见操作的示例:

  • 识别数组类型描述了如何确定类成员是否是数组类型的字段
  • 创建新数组演示了如何创建具有简单和复杂组件类型的新数组实例
  • 获取和设置数组及其组件展示了如何访问数组类型的字段以及单独访问数组元素
  • 故障排除涵盖了常见错误和编程误解

所有这些操作都通过java.lang.reflect.Array中的static方法支持。

识别数组类型

原文:docs.oracle.com/javase/tutorial/reflect/special/arrayComponents.html

可以通过调用Class.isArray()来识别数组类型。要获取一个Class,请使用本教程中检索类对象部分描述的方法之一。

ArrayFind示例标识了命名类中的数组类型字段,并报告了每个字段的组件类型。

import java.lang.reflect.Field;
import java.lang.reflect.Type;
import static java.lang.System.out;
public class ArrayFind {
    public static void main(String... args) {
  boolean found = false;
  try {
      Class<?> cls = Class.forName(args[0]);
      Field[] flds = cls.getDeclaredFields();
      for (Field f : flds) {
    Class<?> c = f.getType();
    if (c.isArray()) {
        found = true;
        out.format("%s%n"
                               + "           Field: %s%n"
             + "            Type: %s%n"
             + "  Component Type: %s%n",
             f, f.getName(), c, c.getComponentType());
    }
      }
      if (!found) {
    out.format("No array fields%n");
      }
        // production code should handle this exception more gracefully
  } catch (ClassNotFoundException x) {
      x.printStackTrace();
  }
    }
}

Class.get*Type()返回值的语法在Class.getName()中有描述。类型名称开头的’['字符的数量表示数组的维度(即嵌套的深度)。

输出示例如下。用户输入用斜体表示。一个原始类型为byte的数组:

$*java ArrayFind java.nio.ByteBuffer*
final byte[] java.nio.ByteBuffer.hb
           Field: hb
            Type: class [B
  Component Type: byte

一个引用类型为StackTraceElement的数组:

$ *java ArrayFind java.lang.Throwable*
private java.lang.StackTraceElement[] java.lang.Throwable.stackTrace
           Field: stackTrace
            Type: class [Ljava.lang.StackTraceElement;
  Component Type: class java.lang.StackTraceElement

predefined是一个引用类型的一维数组java.awt.Cursor,而cursorProperties是一个引用类型的二维数组String

$ *java ArrayFind java.awt.Cursor*
protected static java.awt.Cursor[] java.awt.Cursor.predefined
           Field: predefined
            Type: class [Ljava.awt.Cursor;
  Component Type: class java.awt.Cursor
static final java.lang.String[][] java.awt.Cursor.cursorProperties
           Field: cursorProperties
            Type: class [[Ljava.lang.String;
  Component Type: class [Ljava.lang.String;

创建新数组

原文:docs.oracle.com/javase/tutorial/reflect/special/arrayInstance.html

就像非反射代码一样,反射支持通过java.lang.reflect.Array.newInstance()动态创建任意类型和维度的数组的能力。考虑ArrayCreator,一个能够动态创建数组的基本解释器。将解析的语法如下:

fully_qualified_class_name variable_name[] = 
     { val1, val2, val3, ... }

假设fully_qualified_class_name代表一个具有接受单个String参数的构造函数的类。数组的维度由提供的值的数量确定。以下示例将构造一个fully_qualified_class_name数组的实例,并用val1val2等给定的实例填充其值。(此示例假定熟悉Class.getConstructor()java.lang.reflect.Constructor.newInstance()。有关Constructor的反射 API 的讨论,请参阅本教程的 Creating New Class Instances 部分。)

import java.lang.reflect.Array;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.util.regex.Pattern;
import java.util.regex.Matcher;
import java.util.Arrays;
import static java.lang.System.out;
public class ArrayCreator {
    private static String s = "java.math.BigInteger bi[] = { 123, 234, 345 }";
    private static Pattern p = Pattern.compile("^\\s*(\\S+)\\s*\\w+\\[\\].*\\{\\s*([^}]+)\\s*\\}");
    public static void main(String... args) {
        Matcher m = p.matcher(s);
        if (m.find()) {
            String cName = m.group(1);
            String[] cVals = m.group(2).split("[\\s,]+");
            int n = cVals.length;
            try {
                Class<?> c = Class.forName(cName);
                Object o = Array.newInstance(c, n);
                for (int i = 0; i < n; i++) {
                    String v = cVals[i];
                    Constructor ctor = c.getConstructor(String.class);
                    Object val = ctor.newInstance(v);
                    Array.set(o, i, val);
                }
                Object[] oo = (Object[])o;
                out.format("%s[] = %s%n", cName, Arrays.toString(oo));
            // production code should handle these exceptions more gracefully
            } catch (ClassNotFoundException x) {
                x.printStackTrace();
            } catch (NoSuchMethodException x) {
                x.printStackTrace();
            } catch (IllegalAccessException x) {
                x.printStackTrace();
            } catch (InstantiationException x) {
                x.printStackTrace();
            } catch (InvocationTargetException x) {
                x.printStackTrace();
            }
        }
    }
}
$ *java ArrayCreator*
java.math.BigInteger [] = [123, 234, 345]

上面的示例展示了可能希望通过反射创建数组的一种情况;即如果组件类型直到运行时才知道。在这种情况下,代码使用Class.forName()获取所需组件类型的类,然后调用特定的构造函数来初始化数组的每个组件,然后设置相应的数组值。

获取和设置数组及其组件

原文:docs.oracle.com/javase/tutorial/reflect/special/arraySetGet.html

就像在非反射代码中一样,可以整体设置或逐个组件设置或检索数组字段。要一次设置整个数组,请使用java.lang.reflect.Field.set(Object obj, Object value)。要检索整个数组,请使用Field.get(Object)。可以使用java.lang.reflect.Array中的方法来设置或检索单个组件。

Array提供了形式为set*Foo*()get*Foo*()的方法,用于设置和获取任何原始类型的组件。例如,可以使用Array.setInt(Object array, int index, int value)设置int数组的组件,并可以使用Array.getInt(Object array, int index)检索它。

这些方法支持自动扩宽数据类型。因此,Array.getShort()可以用于设置int数组的值,因为一个 16 位的short可以被扩宽为 32 位的int而不会丢失数据;另一方面,在int数组上调用Array.setLong()将导致抛出IllegalArgumentException,因为 64 位的long不能被缩小为 32 位的int而不丢失信息。无论传递的实际值是否能够准确表示为目标数据类型,这都是正确的。Java 语言规范,Java SE 7 版,章节Widening Primitive ConversionNarrowing Primitive Conversion包含了对扩宽和缩窄转换的完整讨论。

引用类型数组(包括数组的数组)的组件使用Array.set(Object array, int index, int value)Array.get(Object array, int index)进行设置和检索。

设置类型为数组的字段

GrowBufferedReader示例演示了如何替换类型为数组的字段的值。在这种情况下,代码将java.io.BufferedReader的后备数组替换为更大的数组。(这假设原始BufferedReader的创建在不可修改的代码中;否则,可以简单地使用接受输入缓冲区大小的替代构造函数BufferedReader(java.io.Reader in, int size)。)

import java.io.BufferedReader;
import java.io.CharArrayReader;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.lang.reflect.Field;
import java.util.Arrays;
import static java.lang.System.out;
public class GrowBufferedReader {
    private static final int srcBufSize = 10 * 1024;
    private static char[] src = new char[srcBufSize];
    static {
  src[srcBufSize - 1] = 'x';
    }
    private static CharArrayReader car = new CharArrayReader(src);
    public static void main(String... args) {
  try {
      BufferedReader br = new BufferedReader(car);
      Class<?> c = br.getClass();
      Field f = c.getDeclaredField("cb");
      // cb is a private field
      f.setAccessible(true);
      char[] cbVal = char[].class.cast(f.get(br));
      char[] newVal = Arrays.copyOf(cbVal, cbVal.length * 2);
      if (args.length > 0 && args[0].equals("grow"))
    f.set(br, newVal);
      for (int i = 0; i < srcBufSize; i++)
    br.read();
      // see if the new backing array is being used
      if (newVal[srcBufSize - 1] == src[srcBufSize - 1])
    out.format("Using new backing array, size=%d%n", newVal.length);
      else
    out.format("Using original backing array, size=%d%n", cbVal.length);
        // production code should handle these exceptions more gracefully
  } catch (FileNotFoundException x) {
      x.printStackTrace();
  } catch (NoSuchFieldException x) {
      x.printStackTrace();
  } catch (IllegalAccessException x) {
      x.printStackTrace();
  } catch (IOException x) {
      x.printStackTrace();
  }
    }
}
$ *java GrowBufferedReader grow*
Using new backing array, size=16384
$ *java GrowBufferedReader*
Using original backing array, size=8192

请注意,上述示例使用了数组实用方法java.util.Arrays.copyOf)java.util.Arrays包含许多在操作数组时方便的方法。

访问多维数组的元素

多维数组简单来说就是嵌套数组。二维数组是数组的数组。三维数组是二维数组的数组,依此类推。CreateMatrix示例演示了如何使用反射创建和初始化多维数组。

import java.lang.reflect.Array;
import static java.lang.System.out;
public class CreateMatrix {
    public static void main(String... args) {
        Object matrix = Array.newInstance(int.class, 2, 2);
        Object row0 = Array.get(matrix, 0);
        Object row1 = Array.get(matrix, 1);
        Array.setInt(row0, 0, 1);
        Array.setInt(row0, 1, 2);
        Array.setInt(row1, 0, 3);
        Array.setInt(row1, 1, 4);
        for (int i = 0; i < 2; i++)
            for (int j = 0; j < 2; j++)
                out.format("matrix[%d][%d] = %d%n", i, j, ((int[][])matrix)[i][j]);
    }
}
$ *java CreateMatrix*
matrix[0][0] = 1
matrix[0][1] = 2
matrix[1][0] = 3
matrix[1][1] = 4

通过使用以下代码片段也可以获得相同的结果:

Object matrix = Array.newInstance(int.class, 2);
Object row0 = Array.newInstance(int.class, 2);
Object row1 = Array.newInstance(int.class, 2);
Array.setInt(row0, 0, 1);
Array.setInt(row0, 1, 2);
Array.setInt(row1, 0, 3);
Array.setInt(row1, 1, 4);
Array.set(matrix, 0, row0);
Array.set(matrix, 1, row1);

可变参数Array.newInstance(Class componentType, int... dimensions)提供了一个方便的方式来创建多维数组,但组件仍然需要使用多维数组是嵌套数组的原则进行初始化。(反射不提供用于此目的的多个索引get/set方法。)

故障排除

原文:docs.oracle.com/javase/tutorial/reflect/special/arrayTrouble.html

以下示例展示了在操作数组时可能发生的典型错误。

由于不可转换的类型导致IllegalArgumentException

ArrayTroubleAgain示例将生成一个IllegalArgumentException。调用Array.setInt()来设置一个Integer类型的组件,其值为基本类型int。在非反射等效的ary[0] = 1中,编译器会将值1转换(或装箱)为引用类型new Integer(1),以便其类型检查接受该语句。在使用反射时,类型检查仅在运行时发生,因此没有机会将值装箱。

import java.lang.reflect.Array;
import static java.lang.System.err;
public class ArrayTroubleAgain {
    public static void main(String... args) {
  Integer[] ary = new Integer[2];
  try {
      Array.setInt(ary, 0, 1);  // IllegalArgumentException
        // production code should handle these exceptions more gracefully
  } catch (IllegalArgumentException x) {
      err.format("Unable to box%n");
  } catch (ArrayIndexOutOfBoundsException x) {
      x.printStackTrace();
  }
    }
}
$ *java ArrayTroubleAgain*
Unable to box

要消除此异常,有问题的行应该被以下调用替换Array.set(Object array, int index, Object value)

Array.set(ary, 0, new Integer(1));

提示: 当使用反射设置或获取数组组件时,编译器无法执行装箱。它只能转换与Class.isAssignableFrom()规范描述的相关类型。该示例预计会失败,因为isAssignableFrom()在此测试中将返回false,可以用程序验证特定转换是否可能:

Integer.class.isAssignableFrom(int.class) == false 

同样,在反射中从基本类型到引用类型的自动转换也是不可能的。

int.class.isAssignableFrom(Integer.class) == false

对空数组的ArrayIndexOutOfBoundsException

ArrayTrouble示例说明了如果尝试访问长度为零的数组元素将会发生的错误:

import java.lang.reflect.Array;
import static java.lang.System.out;
public class ArrayTrouble {
    public static void main(String... args) {
        Object o = Array.newInstance(int.class, 0);
        int[] i = (int[])o;
        int[] j = new int[0];
        out.format("i.length = %d, j.length = %d, args.length = %d%n",
                   i.length, j.length, args.length);
        Array.getInt(o, 0);  // ArrayIndexOutOfBoundsException
    }
}
$ *java ArrayTrouble*
i.length = 0, j.length = 0, args.length = 0
Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException
        at java.lang.reflect.Array.getInt(Native Method)
        at ArrayTrouble.main(ArrayTrouble.java:11)

提示: 可以有没有元素的数组(空数组)。在常见代码中只有少数情况下会看到它们,但它们可能会在反射中无意中出现。当然,无法设置/获取空数组的值,因为会抛出ArrayIndexOutOfBoundsException


如果尝试缩小范围会导致IllegalArgumentException

ArrayTroubleToo示例包含的代码会失败,因为它尝试执行一个可能会丢失数据的操作:

import java.lang.reflect.Array;
import static java.lang.System.out;
public class ArrayTroubleToo {
    public static void main(String... args) {
        Object o = new int[2];
        Array.setShort(o, 0, (short)2);  // widening, succeeds
        Array.setLong(o, 1, 2L);         // narrowing, fails
    }
}
$ *java ArrayTroubleToo*
Exception in thread "main" java.lang.IllegalArgumentException: argument type
  mismatch
        at java.lang.reflect.Array.setLong(Native Method)
        at ArrayTroubleToo.main(ArrayTroubleToo.java:9)

提示: Array.set*()Array.get*() 方法将执行自动扩展转换,但如果尝试进行缩小转换,则会抛出 IllegalArgumentException。有关扩展和缩小转换的完整讨论,请参阅Java 语言规范,Java SE 7 版,分别查看Widening Primitive ConversionNarrowing Primitive Conversion部分。


枚举类型

原文:docs.oracle.com/javase/tutorial/reflect/special/enum.html

枚举是一种语言构造,用于定义类型安全的枚举,当需要固定一组命名值时可以使用。所有枚举隐式扩展 java.lang.Enum。枚举可以包含一个或多个枚举常量,这些常量定义了枚举类型的唯一实例。枚举声明定义了一个枚举类型,与类非常相似,可以具有字段、方法和构造函数等成员(有一些限制)。

由于枚举是类,反射不需要定义一个显式的java.lang.reflect.Enum类。枚举特定的反射 API 只有 Class.isEnum()Class.getEnumConstants()java.lang.reflect.Field.isEnumConstant()。涉及枚举的大多数反射操作与任何其他类或成员相同。例如,枚举常量被实现为枚举上的public static final字段。以下部分展示了如何在枚举中使用 Classjava.lang.reflect.Field

  • 检查枚举 说明了如何检索枚举的常量以及任何其他字段、构造函数和方法
  • 使用枚举类型获取和设置字段 展示了如何使用枚举常量值设置和获取字段
  • 故障排除描述了与枚举相关的常见错误

有关枚举的介绍,请参阅 枚举类型 课程。

Java 中文官方教程 2022 版(四十四)(4)https://developer.aliyun.com/article/1488326

相关文章
|
13天前
|
Java 测试技术 Python
《手把手教你》系列技巧篇(二十九)-java+ selenium自动化测试- Actions的相关操作上篇(详解教程)
【4月更文挑战第21天】本文介绍了Selenium中处理特殊测试场景的方法,如鼠标悬停。Selenium的Actions类提供了鼠标悬停功能,用于模拟用户在网页元素上的悬停行为。文中通过实例展示了如何使用Actions悬停并展开下拉菜单,以及在搜索时选择自动补全的字段。代码示例包括了打开百度首页,悬停在“更多”元素上显示下拉菜单并点击“音乐”,以及在搜索框输入关键词并自动补全的过程。
35 0
|
6天前
|
Java 测试技术 Python
《手把手教你》系列技巧篇(三十六)-java+ selenium自动化测试-单选和多选按钮操作-番外篇(详解教程)
【4月更文挑战第28天】本文简要介绍了自动化测试的实战应用,通过一个在线问卷调查(&lt;https://www.sojump.com/m/2792226.aspx/&gt;)为例,展示了如何遍历并点击问卷中的选项。测试思路包括找到单选和多选按钮的共性以定位元素,然后使用for循环进行点击操作。代码设计方面,提供了Java+Selenium的示例代码,通过WebDriver实现自动答题。运行代码后,可以看到控制台输出和浏览器的相应动作。文章最后做了简单的小结,强调了本次实践是对之前单选多选操作的巩固。
18 0
|
6天前
|
Java 测试技术 项目管理
Java基础教程(22)-构建工具Maven的基本使用
【4月更文挑战第22天】Maven是Java项目管理及构建工具,简化构建、测试、打包和部署等任务。遵循约定优于配置原则,核心是`pom.xml`配置文件,用于管理依赖和项目信息。安装涉及下载、解压、配置环境变量。在IDEA中使用Maven创建项目,通过`pom.xml`添加依赖和管理版本。常用命令包括`clean`、`compile`、`test`、`package`、`install`和`deploy`。IDEA支持直接执行这些命令。
|
6天前
|
NoSQL Java 关系型数据库
Java基础教程(21)-Java连接MongoDB
【4月更文挑战第21天】MongoDB是开源的NoSQL数据库,强调高性能和灵活性。Java应用通过MongoDB Java驱动与之交互,涉及MongoClient、MongoDatabase、MongoCollection和Document等组件。连接MongoDB的步骤包括:配置连接字符串、创建MongoClient、选择数据库和集合。伪代码示例展示了如何建立连接、插入和查询数据。
|
7天前
|
存储 前端开发 测试技术
《手把手教你》系列技巧篇(三十五)-java+ selenium自动化测试-单选和多选按钮操作-下篇(详解教程)
【4月更文挑战第27天】本文介绍了使用Java+Selenium进行Web自动化测试时,如何遍历和操作多选按钮的方法。文章分为两个部分,首先是一个本地HTML页面的示例,展示了多选按钮的HTML代码和页面效果,并详细解释了遍历多选按钮的思路:找到所有多选按钮的共同点,通过定位这些元素并放入list容器中,然后使用for循环遍历并操作。 第二部分介绍了在JQueryUI网站上的实战,给出了被测网址,展示了代码设计,同样使用了findElements()方法获取所有多选按钮并存储到list中,然后遍历并进行点击操作。最后,文章对整个过程进行了小结,并推荐了作者的其他自动化测试教程资源。
17 0
|
7天前
|
Java 关系型数据库 MySQL
Java基础教程(20)-Java连接mysql数据库CURD
【4月更文挑战第19天】MySQL是流行的关系型数据库管理系统,支持SQL语法。在IDEA中加载jar包到项目类路径:右击项目,选择“Open Module Settings”,添加库文件。使用JDBC连接MySQL,首先下载JDBC驱动,然后通过`Class.forName()`加载驱动,`DriverManager.getConnection()`建立连接。执行CRUD操作,例如创建表、插入数据和查询,使用`Statement`或`PreparedStatement`,并确保正确关闭数据库资源。
|
8天前
|
设计模式 算法 Java
Java基础教程(19)-设计模式简述
【4月更文挑战第19天】设计模式是软件设计中反复使用的代码设计经验,旨在提升代码的可重用性、可扩展性和可维护性。23种模式分为创建型、结构型和行为型三类。创建型模式如工厂方法、抽象工厂、建造者、原型和单例,关注对象创建与使用的分离。结构型模式涉及对象组合,如适配器、装饰器、外观等,增强结构灵活性。行为型模式专注于对象间职责分配和算法合作,包括责任链、命令、观察者等。设计模式提供标准化解决方案,促进代码交流和复用。
|
8天前
|
前端开发 测试技术 Python
《手把手教你》系列技巧篇(三十三)-java+ selenium自动化测试-单选和多选按钮操作-上篇(详解教程)
【4月更文挑战第25天】本文介绍了自动化测试中如何处理单选和多选按钮的操作,包括它们的定义、HTML代码示例以及如何判断和操作这些元素。文章通过一个简单的HTML页面展示了单选和多选框的示例,并提供了Java+Selenium实现的代码示例,演示了如何检查单选框是否选中以及如何进行全选操作。
15 0
|
9天前
|
网络协议 Java 网络架构
Java基础教程(18)-Java中的网络编程
【4月更文挑战第18天】Java网络编程简化了底层协议处理,利用Java标准库接口进行TCP/IP通信。TCP协议提供可靠传输,常用于HTTP、SMTP等协议;UDP协议则更高效但不保证可靠性。在TCP编程中,ServerSocket用于监听客户端连接,Socket实现双进程间通信。UDP编程中,DatagramSocket处理无连接的数据报文。HTTP编程可以通过HttpURLConnection发送请求并接收响应。
|
10天前
|
前端开发 Java 测试技术
《手把手教你》系列技巧篇(三十二)-java+ selenium自动化测试-select 下拉框(详解教程)
【4月更文挑战第24天】本文介绍了在自动化测试中处理HTML下拉选择(select)的方法。使用Selenium的Select类,可以通过index、value或visible text三种方式选择选项,并提供了相应的取消选择的方法。此外,文章还提供了一个示例HTML页面(select.html)和相关代码实战,演示了如何使用Selenium进行选择和取消选择操作。最后,文章提到了现代网页中类似下拉框的新设计,如12306网站的出发地选择,并给出了相应的代码示例,展示了如何定位并选择特定选项。
18 0