【发布时间】:2011-03-05 02:42:45
【问题描述】:
我想在 Java 中将数组转换为 Set。有一些明显的方法可以做到这一点(即使用循环),但我想要一些更整洁的方法,比如:
java.util.Arrays.asList(Object[] a);
有什么想法吗?
【问题讨论】:
标签: java collections arrays set
我想在 Java 中将数组转换为 Set。有一些明显的方法可以做到这一点(即使用循环),但我想要一些更整洁的方法,比如:
java.util.Arrays.asList(Object[] a);
有什么想法吗?
【问题讨论】:
标签: java collections arrays set
在你执行Arrays.asList(array)之后你可以执行Set set = new HashSet(list);
这是一个示例方法,你可以这样写:
public <T> Set<T> GetSetFromArray(T[] array) {
return new HashSet<T>(Arrays.asList(array));
}
【讨论】:
像这样:
Set<T> mySet = new HashSet<>(Arrays.asList(someArray));
在 Java 9+ 中,如果不可修改的集合是可以的:
Set<T> mySet = Set.of(someArray);
在 Java 10+ 中,泛型类型参数可以从数组组件类型中推断出来:
var mySet = Set.of(someArray);
小心
Set.of 抛出 IllegalArgumentException - 如果有任何重复 someArray 中的元素。 查看更多详情:https://docs.oracle.com/en/java/javase/11/docs/api/java.base/java/util/Set.html#of(E...)
【讨论】:
Arrays.asList 是 O(1)。
new HashSet<Object>(Arrays.asList(Object[] a));
但我认为这样会更有效率:
final Set s = new HashSet<Object>();
for (Object o : a) { s.add(o); }
【讨论】:
HashSet的初始容量是根据数组的大小来设置的,例如
使用Guava,您可以:
T[] array = ...
Set<T> set = Sets.newHashSet(array);
【讨论】:
new HashSet<T>(Arrays.asList(someArray))。见google.github.io/guava/releases/19.0/api/docs/com/google/common/…
Set<T> mySet = new HashSet<T>();
Collections.addAll(mySet, myArray);
这是来自 JDK 6 的 Collections.addAll(java.util.Collection, T...)。
另外:如果我们的数组充满了原语怎么办?
对于 for 循环即可一次性完成包装和添加设置。
对于 JDK >= 8,一个有吸引力的选项是:
Arrays.stream(intArray).boxed().collect(Collectors.toSet());
【讨论】:
java.util.Collections.addAll 做到这一点。另外,我不会再推荐 Commons Collections,因为它没有被泛化并且 Guava 存在。
addAll 将是 O(n)。
快点:你可以做到:
// Fixed-size list
List list = Arrays.asList(array);
// Growable list
list = new LinkedList(Arrays.asList(array));
// Duplicate elements are discarded
Set set = new HashSet(Arrays.asList(array));
倒转
// Create an array containing the elements in a list
Object[] objectArray = list.toArray();
MyClass[] array = (MyClass[])list.toArray(new MyClass[list.size()]);
// Create an array containing the elements in a set
objectArray = set.toArray();
array = (MyClass[])set.toArray(new MyClass[set.size()]);
【讨论】:
有时使用一些标准库会有很大帮助。尝试查看Apache Commons Collections。在这种情况下,您的问题只是简单地转化为这样的事情
String[] keys = {"blah", "blahblah"}
Set<String> myEmptySet = new HashSet<String>();
CollectionUtils.addAll(pythonKeywordSet, keys);
【讨论】:
java.util.Collections.addAll(myEmptySet, keys);??
在Eclipse Collections 中,以下将起作用:
Set<Integer> set1 = Sets.mutable.of(1, 2, 3, 4, 5);
Set<Integer> set2 = Sets.mutable.of(new Integer[]{1, 2, 3, 4, 5});
MutableSet<Integer> mutableSet = Sets.mutable.of(1, 2, 3, 4, 5);
ImmutableSet<Integer> immutableSet = Sets.immutable.of(1, 2, 3, 4, 5);
Set<Integer> unmodifiableSet = Sets.mutable.of(1, 2, 3, 4, 5).asUnmodifiable();
Set<Integer> synchronizedSet = Sets.mutable.of(1, 2, 3, 4, 5).asSynchronized();
ImmutableSet<Integer> immutableSet = Sets.mutable.of(1, 2, 3, 4, 5).toImmutable();
注意:我是 Eclipse Collections 的提交者
【讨论】:
Java 8:
String[] strArray = {"eins", "zwei", "drei", "vier"};
Set<String> strSet = Arrays.stream(strArray).collect(Collectors.toSet());
System.out.println(strSet);
// [eins, vier, zwei, drei]
【讨论】:
我们也可以选择使用Stream。我们可以通过多种方式获取流:
Set<String> set = Stream.of("A", "B", "C", "D").collect(Collectors.toCollection(HashSet::new));
System.out.println(set);
String[] stringArray = {"A", "B", "C", "D"};
Set<String> strSet1 = Arrays.stream(stringArray).collect(Collectors.toSet());
System.out.println(strSet1);
// if you need HashSet then use below option.
Set<String> strSet2 = Arrays.stream(stringArray).collect(Collectors.toCollection(HashSet::new));
System.out.println(strSet2);
Collectors.toSet() 的源代码显示元素是一个一个地添加到HashSet,但规范不保证它将是HashSet。
“类型、可变性、可序列化性或 返回的 Set 的线程安全性。”
所以最好使用后面的选项。输出是:
[A, B, C, D]
[A, B, C, D]
[A, B, C, D]
Java 9 引入了Set.of 静态工厂方法,它为提供的元素或数组返回不可变集合。
@SafeVarargs
static <E> Set<E> of(E... elements)
查看Immutable Set Static Factory Methods了解详情。
我们还可以通过两种方式获得不可变集合:
Set.copyOf(Arrays.asList(array))Arrays.stream(array).collect(Collectors.toUnmodifiableList());Collectors.toUnmodifiableList() 方法在内部使用了 Java 9 中引入的Set.of。更多信息请查看我的answer。
【讨论】:
Stream.of() - 我不知道那个。关于Collectors.toSet() 的一个小问题:你说规范不保证一个一个地添加元素,但这就是它的意思:“积累......到一个新的Set”。而且它更具可读性 - 如果您不需要具体类型、可变性、可序列化和线程安全性的保证,那么对我来说更可取。
HashSet。它只保证它将是Set,这就是我的意思。希望我澄清了。
可变参数也可以!
Stream.of(T... values).collect(Collectors.toSet());
【讨论】:
Set<T> b = new HashSet<>(Arrays.asList(requiredArray));
【讨论】:
我已经根据上面的建议写了下面的内容 - 偷它......这很好!
/**
* Handy conversion to set
*/
public class SetUtil {
/**
* Convert some items to a set
* @param items items
* @param <T> works on any type
* @return a hash set of the input items
*/
public static <T> Set<T> asSet(T ... items) {
return Stream.of(items).collect(Collectors.toSet());
}
}
【讨论】:
private Map<Integer, Set<Integer>> nobreaks = new HashMap();
nobreaks.put(1, new HashSet(Arrays.asList(new int[]{2, 4, 5})));
System.out.println("expected size is 3: " +nobreaks.get(1).size());
输出是
expected size is 3: 1
改成
nobreaks.put(1, new HashSet(Arrays.asList( 2, 4, 5 )));
输出是
expected size is 3: 3
【讨论】:
在 Java 10 中:
String[] strs = {"A", "B"};
Set<String> set = Set.copyOf(Arrays.asList(strs));
Set.copyOf 返回一个不可修改的Set,其中包含给定Collection 的元素。
给定的Collection 不得为null,且不得包含任何null 元素。
【讨论】:
使用CollectionUtils 或ArrayUtils 来自stanford-postagger-3.0.jar
import static edu.stanford.nlp.util.ArrayUtils.asSet;
or
import static edu.stanford.nlp.util.CollectionUtils.asSet;
...
String [] array = {"1", "q"};
Set<String> trackIds = asSet(array);
【讨论】:
已经有很多很好的答案,但其中大多数不适用于基元数组(如int[]、long[]、char[]、byte[] 等)
在 Java 8 及更高版本中,您可以将数组装箱:
Integer[] boxedArr = Arrays.stream(arr).boxed().toArray(Integer[]::new);
然后转换成使用流设置:
Stream.of(boxedArr).collect(Collectors.toSet());
【讨论】:
对于任何解决 Android 问题的人:
星号* 是spread 运算符。它单独应用集合中的所有元素,每个元素都按顺序传递给vararg 方法参数。相当于:
val myArray = arrayOf("data", "foo")
val mySet = setOf(*myArray)
// Equivalent to
val mySet = setOf("data", "foo")
// Multiple spreads ["data", "foo", "bar", "data", "foo"]
val mySet = setOf(*myArray, "bar", *myArray)
不传递参数setOf() 会导致一个空集。
除了setOf,您还可以将其中任何一个用于特定的哈希类型:
hashSetOf()
linkedSetOf()
mutableSetOf()
sortableSetOf()
这是明确定义集合项类型的方法。
setOf<String>()
hashSetOf<MyClass>()
【讨论】:
如果您需要构建一个内部只有一个元素的不可变集合,您可以使用Collections.singleton(...)。这是一个例子:
Set<String> mySet = Collections.singleton("Have a good day :-)");
这不能回答最初的问题,但可能对某人有用(至少对我来说)。如果您认为此答案不合适,请告诉我,我将删除它。
【讨论】: