Java 未选中:未选中 varargs 参数的泛型数组创建

2022-08-31 08:21:51

我已将 Netbeans 设置为在我的 Java 代码中显示未经检查的警告,但我无法理解以下行中的错误:

private List<String> cocNumbers;
private List<String> vatNumbers;
private List<String> ibans;
private List<String> banks;
...
List<List<String>> combinations = Utils.createCombinations(cocNumbers, vatNumbers, ibans);

给:

[unchecked] unchecked generic array creation for varargs parameter of type List<String>[]

方法来源:

/**
 * Returns a list of all possible combinations of the entered array of lists.
 *
 * Example: [["A", "B"], ["0", "1", "2"]]
 * Returns: [["A", "0"], ["A", "1"], ["A", "2"], ["B", "0"], ["B", "1"], ["B", "2"]]
 *
 * @param <T> The type parameter
 * @param elements An array of lists
 * @return All possible combinations of the entered lists
 */
public static <T> List<List<T>> createCombinations(List<T>... elements) {
    List<List<T>> returnLists = new ArrayList<>();

    int[] indices = new int[elements.length];
    for (int i = 0; i < indices.length; i++) {
        indices[i] = 0;
    }

    returnLists.add(generateCombination(indices, elements));
    while (returnLists.size() < countCombinations(elements)) {
        gotoNextIndex(indices, elements);
        returnLists.add(generateCombination(indices, elements));
    }

    return returnLists;
}

究竟出了什么问题,我该如何解决它,因为我认为在代码中留下未经检查的警告不是一个好主意?

忘了提,但我使用的是Java 7。

编辑:我现在也看到该方法具有以下功能:

[unchecked] Possible heap pollution from parameterized vararg type List<T>
  where T is a type-variable:
    T extends Object declared in method <T>createCombinations(List<T>...)

答案 1

正如 janoh.janoh 上面提到的,Java 中的 varargs 只是数组的语法糖,加上在调用站点隐式创建数组。所以

List<List<String>> combinations =
    Utils.createCombinations(cocNumbers, vatNumbers, ibans);

其实是

List<List<String>> combinations =
    Utils.createCombinations(new List<String>[]{cocNumbers, vatNumbers, ibans});

但是,您可能知道,由于许多其他问题中已经介绍过的原因,Java中不允许这样做,但主要与数组在运行时知道其组件类型这一事实有关,并在运行时检查添加的元素是否与其组件类型匹配,但是对于参数化类型,此检查是不可能的。new List<String>[]

无论如何,编译器不会失败,而是仍然创建数组。它执行类似于以下内容执行的操作:

List<List<String>> combinations =
    Utils.createCombinations((List<String>[])new List<?>[]{cocNumbers, vatNumbers, ibans});

这可能不安全,但不一定不安全。大多数 varargs 方法只是简单地迭代 varargs 元素并读取它们。在这种情况下,它不关心数组的运行时类型。您的方法就是这种情况。由于您使用的是 Java 7,因此应将注释添加到方法中,并且不会再收到此警告。这个注释基本上说,这种方法只关心元素的类型,而不关心数组的类型。@SafeVarargs

但是,有一些 varargs 方法确实使用数组的运行时类型。在这种情况下,它可能是不安全的。这就是为什么警告在那里。


答案 2

因为java编译器使用隐式数组创建varargs,而java不允许创建泛型数组(因为类型参数不可重用)。

下面的代码是正确的(数组允许这些操作),因此需要未选中的警告:

public static <T> List<List<T>> createCombinations(List<T> ... lists) {
    ((Object[]) lists)[0] = new ArrayList<Integer>();
    // place your code here
}

在此处查看全面的说明