Java通用varargs方法参数

时间:2013-03-12 09:25:43

标签: java arrays variadic-functions

我想知道是否有一种简单的,优雅的和可重用的方法将字符串和字符串数组传递给期望varargs的方法。

/**
 * The entry point with a clearly separated list of parameters.
 */
public void separated(String p1, String ... p2) {
    merged(p1, p2, "another string", new String[]{"and", "those", "one"});
}

/**
 * For instance, this method outputs all the parameters.
 */
public void merged(String ... p) {
    // magic trick
}

即使所有类型都是一致的(String),我找不到告诉JVM 展平 p2并将其注入合并参数列表的方法吗?

此时唯一的方法是创建一个新数组,将所有数据复制到该数组中并将其传递给该函数。

有什么想法吗?


修改

根据您的建议,我将使用的通用方法:

/**
 * Merge the T and T[] parameters into a new array.
 *
 * @param type       the destination array type
 * @param parameters the parameters to merge
 * @param <T>        Any type
 * @return the new holder
 */
@SuppressWarnings("unchecked")
public static <T> T[] normalize(Class<T> type, Object... parameters) {
    List<T> flatten = new ArrayList<>();
    for (Object p : parameters) {
        if (p == null) {
            // hum... assume it is a single element
            flatten.add(null);
            continue;
        }
        if (type.isInstance(p)) {
            flatten.add((T) p);
            continue;
        }
        if (p.getClass().isArray() && 
            p.getClass().getComponentType().equals(type)) {
            Collections.addAll(flatten, (T[]) p);
        } else {
            throw new RuntimeException("should be " + type.getName() + 
                                             " or " + type.getName() + 
                                             "[] but was " + p.getClass());
        }
    }
    return flatten.toArray((T[]) Array.newInstance(type, flatten.size()));
}

normalize(String.class, "1", "2", new String[]{"3", "4", null}, null, "7", "8");

6 个答案:

答案 0 :(得分:6)

我认为有一种方法可以将标量和数组隐式展平为单个数组。

我能想到的最干净的解决方案是使用辅助函数:

// generic helper function
public static<T> T[] join(T scalar, T[] arr) {
    T[] ret = Arrays.copyOfRange(arr, 0, arr.length + 1);
    System.arraycopy(ret, 0, ret, 1, arr.length);
    ret[0] = scalar;
    return ret;
}

public void separated(String p1, String ... p2) {
    merged(join(p1, p2));
}

答案 1 :(得分:2)

merge的签名更改为:

public<T> void merged(T ... p) 

至少可以让你毫无问题地调用merge。但是,您必须将字符串数组作为参数处理。

答案 2 :(得分:0)

我认为创建新阵列是最佳选择。

例如,您可以使用Apache Commons中的ArrayUtils来实现此目标。

ArrayUtils.addAll(p2, p1);

祝你好运, 托马斯

答案 3 :(得分:0)

separated内,参数p2String[]merged方法可以 单个String[]参数一系列字符串但不是两者的混合,所以我不知道认为除了构建一个新数组之外你还有很多选择(无论如何这都是varargs在幕后做的事情)。

答案 4 :(得分:0)

来自varargs doc

  

最终参数类型后的三个句点表示   final参数可以作为数组或序列传递   参数。

它明确指出an array OR a sequence of arguments,因此您必须自己展平数组,例如NPE建议的那样。

答案 5 :(得分:0)

我怀疑你能做到这一点,Java编译器应该将merged(p1,p2)转换为merged(p1,p2 [0],p2 [1] ... p2 [p2.length]),但我不知道我认为他们曾经想要创建这样一个智能编译器(如果合并(p1,p2)也会产生问题。)

你可以这样做(使用伪语法):

import com.google.common.collect.Lists
import java.util.Arrays

separated ( String p1, String... p2 ) {
  merged ( Lists.asList ( p1, p2 ) )
}

merged ( List<String> p ) {
...
}

merged ( String ... p ) {
  merged ( Arrays.asList ( p )
}

Arrays来自标准JDK,Lists来自Guava。两个asList()方法都创建了下划线数组的视图,而不是副本,因此速度/内存问题并不多。

我认为不存在更简单的方法。是的,按照其他人的建议,复制到新阵列更简单,但耗费时间/内存。如果您的阵列很小,可能会发生创建和访问列表比复制慢,但对于较大的阵列,复制将成为更高的开销。

我喜欢Sebastian的建议:你定义了一个merge()的包装器,它可以平滑飞行中的数组。只要merge()不是第三方方法并且您可以更改它,这种方法很有效。另外,我在这里做类似的事情。