Я хотів би перетворити масив у набір на Java. Є кілька очевидних способів зробити це (тобто з циклом), але мені хотілося б дещо акуратніше, щось на кшталт:
java.util.Arrays.asList(Object[] a);
Будь-які ідеї?
Я хотів би перетворити масив у набір на Java. Є кілька очевидних способів зробити це (тобто з циклом), але мені хотілося б дещо акуратніше, щось на кшталт:
java.util.Arrays.asList(Object[] a);
Будь-які ідеї?
Відповіді:
Подобається це:
Set<T> mySet = new HashSet<>(Arrays.asList(someArray));
У Java 9+, якщо неможливо змінити набір:
Set<T> mySet = Set.of(someArray);
У Java 10+ параметр загального типу можна зробити з типу компонента масивів:
var mySet = Set.of(someArray);
Arrays.asList
є O (1).
Set<T> mySet = new HashSet<T>();
Collections.addAll(mySet, myArray);
Це Collections.addAll (java.util.Collection, T ...) від JDK 6.
Додатково: що робити, якщо наш масив сповнений примітивів?
Для JDK <8 я просто напишу очевидний for
цикл, щоб зробити обгортання та додавання до набору за один прохід.
Для JDK> = 8 привабливий варіант - це щось на зразок:
Arrays.stream(intArray).boxed().collect(Collectors.toSet());
java.util.Collections.addAll
. Плюс до цього, я б більше не рекомендував колекції Commons, що при цьому не узагальнюється, а Guava існує.
addAll
буде O ( n ).
З 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/…
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
але специфікація не гарантує, що це буде a HashSet
.
"Жодних гарантій щодо типу, змінності, серіалізаційності чи безпечності потоку повернутого набору немає."
Тож краще скористатися пізнішим варіантом. Вихід:
[A, B, C, D]
[A, B, C, D]
[A, B, C, D]
Java 9 представила Set.of
статичний заводський метод, який повертає незмінний набір для наданих елементів або масиву.
@SafeVarargs
static <E> Set<E> of(E... elements)
Перевірте непорушний Набір статичних фабричні методи для деталей.
Ми також можемо отримати незмінний набір двома способами:
Set.copyOf(Arrays.asList(array))
Arrays.stream(array).collect(Collectors.toUnmodifiableList());
Метод Collectors.toUnmodifiableList()
внутрішньо використовує Set.of
впроваджений на Java 9. Також перевірте цю відповідь моєї для отримання додаткової інформації.
Stream.of()
- я цього не знав. Невелика приказка про те, що Collectors.toSet()
ви говорите: специфікація не гарантує додавання елементів по черзі, але це означає: "накопичується ... в нове Set
". І це легше читати - настільки краще, на мій погляд, якщо вам не потрібні гарантії конкретного типу, змінності, серіалізаційності та безпеки ниток.
HashSet
. Це лише гарантує, що це буде Set
і саме про це я. Сподіваюся, я це уточнив.
Після цього Arrays.asList(array)
ви можете виконатиSet set = new HashSet(list);
Ось зразок методу, ви можете написати:
public <T> Set<T> GetSetFromArray(T[] array) {
return new HashSet<T>(Arrays.asList(array));
}
У колекціях Eclipse працює наступне:
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
Швидко: ви можете:
// 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()]);
Нижче я написав вищезгадану пораду - вкрасти ... це приємно!
/**
* 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());
}
}
Там було багато хороших відповідей вже, але більшість з них не буде працювати з масивом примітивів (як int[]
, long[]
, char[]
, byte[]
і т.д.)
У Java 8 та вище, ви можете встановити вікно масиву за допомогою:
Integer[] boxedArr = Arrays.stream(arr).boxed().toArray(Integer[]::new);
Потім перетворіть у набір за допомогою потоку:
Stream.of(boxedArr).collect(Collectors.toSet());
Колись використання деяких стандартних бібліотек дуже допомагає. Спробуйте переглянути колекції Apache Commons . У цьому випадку ваші проблеми просто перетворюються на щось подібне
String[] keys = {"blah", "blahblah"}
Set<String> myEmptySet = new HashSet<String>();
CollectionUtils.addAll(pythonKeywordSet, keys);
А ось колекціїUtils javadoc
java.util.Collections.addAll(myEmptySet, keys);
??
Використовувати 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);
На Java 10 :
String[] strs = {"A", "B"};
Set<String> set = Set.copyOf(Arrays.asList(strs));
Set.copyOf
повертає немодифікується, Set
що містить елементи заданого Collection
.
Дане Collection
не повинно бути null
, і воно не повинно містити null
елементів.
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
Для всіх, хто вирішує для 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>()
new HashSet<Object>(Arrays.asList(Object[] a));
Але я думаю, що це було б більш ефективно:
final Set s = new HashSet<Object>();
for (Object o : a) { s.add(o); }
HashSet
встановлена, наприклад, залежно від розміру масиву.
Set<T> b = new HashSet<>(Arrays.asList(requiredArray));