1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180 1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214 1215 1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234 1235 1236 1237 1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258 1259 1260 1261 1262 1263 1264 1265 1266 1267 1268 1269 1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283 1284 1285 1286 1287 1288 1289 1290 1291 1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307 1308 1309 1310 1311 1312 1313 1314 1315 1316 1317 1318 1319 1320 1321 1322 1323 1324 1325 1326 1327 1328 1329 1330 1331 1332 1333 1334 1335 1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346 1347 1348 1349 1350 1351 1352 1353 1354 1355 1356 1357 1358 1359 1360 1361 1362 1363 1364 1365 1366 1367 1368 1369 1370 1371 1372 1373 1374 1375 1376 1377 1378 1379 1380 1381 1382 1383 1384 1385 1386 1387 1388 1389 1390 1391 1392 1393 1394 1395 1396 1397 1398 1399 1400 1401 1402 1403 1404 1405 1406 1407 1408 1409 1410 1411 1412 1413 1414 1415 1416 1417 1418 1419 1420 1421 1422 1423 1424 1425 1426 1427 1428 1429 1430 1431 1432 1433 1434
|
////
Copyright (c) 2022 Goldman Sachs and others.
All rights reserved.
This program and the accompanying materials are made available
under the terms of the Eclipse Public License v1.0 and
Eclipse Distribution License v.1.0 which accompany this distribution.
The Eclipse Public License is available at
http://www.eclipse.org/legal/epl-v10.html.
The Eclipse Distribution License is available at
http://www.eclipse.org/org/documents/edl-v10.php.
////
= Collections and containers
:sectanchors:
:toc: left
:toc-title:
:toclevels: 3
// Javadoc links
:api-url: https://www.eclipse.org/collections/javadoc/11.0.0/org/eclipse/collections
//
:ArrayStack: {api-url}/impl/stack/mutable/ArrayStack.html[ArrayStack]
:Bag: {api-url}/api/bag/Bag.html[Bag]
:Bags: {api-url}/impl/factory/Bags.html[Bags]
:BiMap: {api-url}/api/bimap/BiMap.html[BiMap]
:BiMaps: {api-url}/impl/factory/BiMaps.html[BiMaps]
:BooleanArrayList: {api-url}/impl/list/mutable/primitive/BooleanArrayList.html[BooleanArrayList]
:BooleanHashSet: {api-url}/impl/set/mutable/primitive/BooleanHashSet.html[BooleanHashSet]
:DoubleSets: {api-url}/impl/factory/primitive/DoubleSets.html[DoubleSets]
:FastList: {api-url}/impl/list/mutable/FastList.html[FastList]
:FastListMultimap: {api-url}/impl/multimap/list/FastListMultimap.html[FastListMultimap]
:FixedSizeList: {api-url}/api/list/FixedSizeList.html[FixedSizeList]
:FloatArrayList: {api-url}/impl/list/mutable/primitive/FloatArrayList.html[FloatArrayList]
:FloatHashSet: {api-url}/impl/set/mutable/primitive/FloatHashSet.html[FloatHashSet]
:Function: {api-url}/api/block/function/Function.html[Function]
:HashBag: {api-url}/impl/bag/mutable/HashBag.html[HashBag]
:HashBiMap: {api-url}/impl/bimap/mutable/HashBiMap.html[HashBiMap]
:ImmutableBag: {api-url}/api/bag/ImmutableBag.html[ImmutableBag]
:ImmutableList: {api-url}/api/list/ImmutableList.html[ImmutableList]
:ImmutableSortedBag: {api-url}/api/bag/sorted/ImmutableSortedBag.html[ImmutableSortedBag]
:ImmutableStack: {api-url}/api/stack/ImmutableStack.html[ImmutableStack]
:IntArrayList: {api-url}/impl/list/mutable/primitive/IntArrayList.html[IntArrayList]
:IntHashSet: {api-url}/impl/set/mutable/primitive/IntHashSet.html[IntHashSet]
:IntLists: {api-url}/impl/factory/primitive/IntLists.html[IntLists]
:IntIntHashMap: {api-url}/impl/map/mutable/primitive/IntIntHashMap.html[IntIntHashMap]
:IntObjectHashMap: {api-url}/impl/map/mutable/primitive/IntObjectHashMap.html[IntObjectHashMap]
:ListIterable: {api-url}/api/list/ListIterable.html[ListIterable]
:Lists: {api-url}/impl/factory/Lists.html[Lists]
:Maps: {api-url}/impl/factory/Maps.html[Maps]
:Multimap: {api-url}/api/multimap/Multimap.html[Multimap]
:Multimaps: {api-url}/impl/factory/Multimaps.html[Multimaps]
:MutableBag: {api-url}/api/bag/MutableBag.html[MutableBag]
:MutableBiMap: {api-url}/api/bimap/MutableBiMap.html[MutableBiMap]
:MutableList: {api-url}/api/list/MutableList.html[MutableList]
:MutableListMultimap: {api-url}/api/multimap/list/MutableListMultimap.html[MutableListMultimap]
:MutableMap: {api-url}/api/map/MutableMap.html[MutableMap]
:MutableSetMultimap: {api-url}/api/multimap/set/MutableSetMultimap.html[MutableSetMultimap]
:MutableSortedBag: {api-url}/api/bag/sorted/MutableSortedBag.html[MutableSortedBag]
:MutableSortedSet: {api-url}/api/set/sorted/MutableSortedSet.html[MutableSortedSet]
:MutableSortedMap: {api-url}/api/map/sorted/MutableSortedSet.html[MutableSortedMap]
:MutableStack: {api-url}/api/stack/MutableStack.html[MutableStack]
:ObjectIntHashMap: {api-url}/impl/map/mutable/primitive/ObjectIntHashMap.html[ObjectIntHashMap]
:RichIterable: {api-url}/api/RichIterable.html[RichIterable]
:Sets: {api-url}/impl/factory/Sets.html[Sets]
:SortedBag: {api-url}/api/bag/sorted/SortedBag.html[SortedBag]
:SortedSetIterable: {api-url}/api/set/sorted/SortedSetIterable.html[SortedSetIterable]
:StackIterable: {api-url}//api/stack/StackIterable.html[StackIterable]
:Stacks: {api-url}/impl/factory/Stacks.html[Stacks]
:StringFunctions: {api-url}/impl/block/factory/StringFunctions.html[StringFunctions]
:TreeBag: {api-url}/impl/bag/sorted/mutable/TreeBag.html[TreeBag]
:TreeSortedMap: {api-url}/impl/map/sorted/mutable/TreeSortedMap.html[TreeSortedMap]
:UnifiedMap: {api-url}/impl/map/mutable/UnifiedMap.html[UnifiedMap]
:UnifiedSet: {api-url}/impl/set/mutable/UnifiedSet.html[UnifiedSet]
:UnsortedBag: {api-url}/api/bag/UnsortedBag.html[UnsortedBag]
:UnsortedSetIterable: {api-url}/api/set/UnsortedSetIterable.html[UnsortedSetIterable]
//
//cross-reference links
:richiterable-xref: link:1-Iteration_Patterns.adoc#richiterable-interface[RichIterable, title="The superinterface that specifies the iteration patterns in Eclipse Collections container types."]
:lazy-iterable-xref: link:1-Iteration_Patterns.adoc#lazy-iteration[lazy iterable]
:groupby-xref: link:1-Iteration_Patterns.adoc#groupby-pattern[groupBy, title="Create a Multimap from a collection by grouping on a selected or generated key value."]
// end links; begin body
What is perhaps most distinctive about the Eclipse Collections collection classes is what (quite properly) is hidden: their implementation of iteration patterns.
Through this encapsulation, Eclipse Collections is able to provide optimized versions of each method on each container.
For example, the first of the classes we'll discuss here, *FastList*, is array-based; it iterates using indexed access directly against its internal array.
We'll begin with the Eclipse Collections implementations of types having analogs in the Java Collections Framework (JCF).
We'll then discuss the new types *Bag* and *Multimap*, the Immutable collections, and the protective wrapper classes.
== Basic collection types
The most commonly-used Eclipse Collections classes are *{FastList}*, *{UnifiedSet}*, and *{UnifiedMap}*.
These collections serve as drop-in replacements for their corresponding types in the Java Collections Framework (JCF).
Note that these Eclipse Collections classes do not extend the JCF implementations; they are instead new implementations of both JCF and Eclipse Collections interfaces, as this (highly-simplified) diagram summarizes:
image:containerCompare.png[Container comparision]
The methods of the JCF types are primarily focused on adding or removing elements and similar, non-iterative operations.
Eclipse Collections interfaces provide methods for iteration patterns that for the most part, do not modify (mutate) the source collection, but rather return a new collection or information about the source collection.
=== ListIterable
____
An ordered collection that allows duplicate elements.
____
A *{ListIterable}* is a *{RichIterable}* which maintains its elements in insertion order and allows duplicate elements.
*ListIterable* has two mutable subinterfaces, *{MutableList}* and *{FixedSizeList}*, and one <<immutable-collections, immutable>> subinterface, *{ImmutableList}*.
The *ListIterable* interface includes the *binarySearch* method, which is similar to the static method *binarySearch* on *java.util.Collections*, but available from the object-oriented API.
==== MutableList
____
A mutable *ListIterable* that implements java.util.List; the most common implementation is *FastList*.
____
*MutableList* extends the JCF *List* interface and has the same contract.
It also extends *{richiterable-xref}*, which provides the iteration methods common to all collections.
The most common implementation of *MutableList* is *FastList*, which can be used to replace the familiar *java.util.ArrayList*.
Here is a comparison of how the two types can be created.
.*ArrayList* in Java Collections Framework
====
[source,java]
----
List<String> comparison = new ArrayList<String>();
comparison.add("Comcast");
comparison.add("IBM");
comparison.add("Microsoft");
comparison.add("Microsoft");
----
====
.*FastList* in Eclipse Collections
====
[source,java]
----
MutableList<String> comparison =
FastList.newListWith("Comcast", "IBM", "Microsoft", "Microsoft");
----
====
The *MutableList* interface includes the *sortThis* and *reverse* methods, which are similar to the static methods *sort* and *reverse* on *java.util.Collections*.
Both are mutating methods.
Here is an example of *sort* using the JDK API and then Eclipse Collections.
.*ArrayList* (JCF) using an anonymous inner class for *Comparator*
====
[source,java]
----
Collections.sort(people, new Comparator<Person>()
{
public int compare(Person o1, Person o2)
{
int lastName = o1.getLastName().compareTo(o2.getLastName());
if (lastName != 0)
{
return lastName;
}
return o1.getFirstName().compareTo(o2.getFirstName());
}
});
----
====
.*FastList* (EC) using an anonymous inner class for *Comparator*
====
[source,java]
----
people.sortThis(new Comparator<Person>()
{
public int compare(Person o1, Person o2)
{
int lastName = o1.getLastName().compareTo(o2.getLastName());
if (lastName != 0)
{
return lastName;
}
return o1.getFirstName().compareTo(o2.getFirstName());
}
});
----
====
.*FastList* (EC) using a lambda
====
[source,java]
----
people.sortThis((o1, o2) ->
{
int lastName = o1.getLastName().compareTo(o2.getLastName());
if (lastName != 0)
{
return lastName;
}
return o1.getFirstName().compareTo(o2.getFirstName());
});
----
====
*MutableList* adds a new method called *sortThisBy*, which takes an attribute from each element using a *{Function}* and then sorts the list by the natural order of that attribute.
.*ArrayList* (JCF) sorting using a static function
====
[source,java]
----
Collections.sort(people, Functions.toComparator(Person.TO_AGE));
----
====
.*FastList* (EC) sorting using a static function
====
[source,java]
----
people.sortThisBy(Person.TO_AGE);
----
====
.*FastList* sorting using a method reference and lambda
====
[source,java]
----
// Using a method reference
people.sortThisBy(Person::getAge);
// Using a lambda expression
people.sortThisBy(person -> person.getAge());
----
====
Here is an example comparing *reverse* using the JCF and using Eclipse Collections; both are mutating methods.
.*ArrayList* (JCF) reversing a list
====
[source,java]
----
Collections.reverse(people);
----
====
.*FastList* (EC) reversing a list
====
[source,java]
----
people.reverseThis();
----
====
The *toReversed* method on *MutableList* lets you reverse a list _without_ mutating it; that is, it returns a new *MutableList*.
Here is an example of how to accomplish that in the JCF and in Eclipse Collections.
.*ArrayList* (JCF) returning a new reversed list
====
[source,java]
----
List<Person> reversed = new ArrayList<Person>(people)
Collections.reverse(reversed);
----
====
.*FastList* (EC) returning a new reversed list
====
[source,java]
----
MutableList<Person> reversed = people.toReversed();
----
====
The *asReversed* method returns a reverse-order _view_ of the *MutableList* —a {lazy-iterable-xref}, like that returned by *asLazy* —that defers each element's evaluation until it is called for by a subsequent method.
.*FastList* (EC) using lazy evaluation
====
[source,java]
----
ListIterable<Integer> integers = FastList.newListWith(1, 2, 3);
// deferred evaluation
LazyIterable<Integer> reversed = integers.asReversed();
// deferred evaluation
LazyIterable<String> strings = reversed.collect(String::valueOf);
// forces evaluation
MutableList<String> stringsList = strings.toList();
Assert.assertEquals(FastList.newListWith("3", "2", "1"), stringsList);
----
====
==== Migrating ArrayList to FastList
Here are some additional JCF to Eclipse Collections refactoring examples.
Here is a Java Collections' *ArrayList*:
.*ArrayList* (JCF)
====
[source,java]
----
List<Integer> integers = new ArrayList<Integer>();
integers.add(1);
integers.add(2);
integers.add(3);
----
====
And, here is the identical construction in Eclipse Collections:
.*FastList* (EC)
====
[source,java]
----
List<Integer> integers = new FastList<Integer>();
integers.add(1);
integers.add(2);
integers.add(3);
----
====
In Eclipse Collections, the static factory method *newList* can infer generic types
.*FastList* from a *newList* factory method
====
[source,java]
----
List<Integer> integers = FastList.newList();
integers.add(1);
integers.add(2);
integers.add(3);
----
====
The Eclipse Collections *newListWith()* method also provides varargs support, allowing any number of arguments.
.*FastList* from a *newListwith* method
====
[source,java]
----
List<Integer> integers = FastList.newListWith(1, 2, 3);
----
====
There are also factory classes in Eclipse Collections named for each collection type ({Lists}, {Sets}, {Maps}, {Bags}, {Stacks}, {BiMaps}, {Multimaps}, etc.)
.*Lists* factory classs
====
[source,java]
----
List<Integer> integers =
Lists.mutable.with(1, 2, 3);
----
====
You can also use the richer interface:
====
[source,java]
----
MutableList<Integer> integers =
FastList.newListWith(1, 2, 3);
// or
MutableList<Integer> integers =
Lists.mutable.with(1, 2, 3);
----
====
The list is never mutated; it can be made <<unmodifiable-collections,unmodifiable>>:
.Unmodifiable *FastList*
====
[source,java]
----
MutableList<Integer> integers =
FastList.newListWith(1, 2, 3).asUnmodifiable();
----
====
There is also a form of *newList* that takes another iterable.
.*newlist* with *Iterable* as input
====
[source,java]
----
MutableList<Integer> integers =
FastList.newList(listOfIntegers);
----
====
These refactorings are analogous for *UnifiedSet* and *UnifiedMap*.
=== SetIterable
____
A collection that allows no duplicate elements.
____
A *SetIterable* is a *RichIterable* that allows no duplicate elements.
It can be sorted or unsorted.
* A *{SortedSetIterable}* is a *SetIterable* that maintains its elements in sorted order.
* An *{UnsortedSetIterable}* is a *SetIterable* that maintains its elements in a hash table in an unpredictable order.
*UnsortedSetIterable* has two mutable subinterfaces (*MutableSet* and *FixedSizeSet*) and one <<immutable-collections,immutable>> subinterface (*ImmutableSet*).
==== MutableSet
____
A mutable *SetIterable* that implements *java.util.Set*; the most common implementation is *UnifiedSet*.
____
*MutableSet* extends the JCF *Set* interface and has the same contract.
It also extends *{richiterable-xref}*, which provides the iteration methods common to all collections.
An attempt to add duplicate elements to a *MutableSet* container is ignored without throwing an exception.
The order in which the elements are processed during iteration is not specified.
The most common implementation is *UnifiedSet*, which can be used to replace the familiar *java.util.HashSet*.
.*HashSet* in Java Collections Framework
====
[source,java]
----
Set<String> comparison = new HashSet<String>();
comparison.add("IBM");
comparison.add("Microsoft");
comparison.add("Oracle");
comparison.add("Comcast");
----
====
.*UnifiedSet* in Eclipse Collections
====
[source,java]
----
Set<String> comparison = UnifiedSet.newSetWith("IBM", "Microsoft", "Verizon", "Comcast");
----
====
==== MutableSortedSet
____
Contains unique items that are sorted by some comparator or their natural ordering.
____
A *{MutableSortedSet}* follows the same contract as a *MutableSet*, but sorts its elements by their natural order, or through a comparator parameter set by the user.
The implementation for *MutableSortedSet* is *TreeSortedSet*.
Here is an example of a MutableSortedSet containing numbers in reverse order:
.*TreeSortedSet*
====
[source,java]
----
MutableSortedSet<Integer> sortedSetA =
TreeSortedSet.newSet(Collections.<Integer>reverseOrder());
MutableSortedSet<Integer> sortedSetB =
TreeSortedSet.newSet(sortedSetA.with(1).with(2, 3).with(4, 5, 6));
----
====
=== MapIterable
____
A collection of key/value pairs.
____
The *MapIterable* interface (extending *RichIterable*) is the top-level interface for collections of key/value pairs.
*MapIterable* has two mutable subinterfaces (*MutableMap* and *FixedSizeMap*), one link:#immutable-collections[immutable] subinterface (*ImmutableMap*).
It is also is extended in mutable and immutable versions of maps with sorted elements (*SortedMapIterable*) and maps that allow lookup keys by (unique) values as well as the reverse (*BiMap*).
==== MutableMap
____
A mutable *MapIterable* that implements java.util.Map; the most common implementation is *UnifiedMap*.
____
The *{MutableMap}* interface defines an association of key/value pairs.
It extends the *MapIterable* interface, which furnishes a set of iteration methods especially for the key/value structure of a Map collection.
These include unmodifiable views of keys, values or pair-entries using the *keysView*, *valuesView* and *entriesView* methods, respectively.
The mutable subinterfaces of *MapIterable* also extend the JCF Map interface.
.*HashMap* (JDK)
====
[source,java]
----
Map<Integer, String> map = new HashMap<Integer, String>();
map.put(1, "1");
map.put(2, "2");
map.put(3, "3");
----
====
.*UnifiedMap* (EC)
====
[source,java]
----
MutableMap<Integer, String> map = UnifiedMap.newWithKeysValues(1, "1", 2, "2", 3, "3");
----
====
==== MutableSortedMap
____
A sorted Map.
____
A *{MutableSortedMap}* follows the same contract as a *MutableMap*, but sorts its elements by their natural order, or through a comparator parameter set by the user.
The implementation for *MutableSortedMap* is *{TreeSortedMap}*.
This code block creates a TreeSortedMap which sorts in reverse order:
.Reversed sorted map
====
[source,java]
----
MutableSortedMap<String, Integer> sortedMap = TreeSortedMap.newMapWith(Comparators.<String>reverseNaturalOrder(),
"1", 1, "2", 3, "3", 2, "4", 1);
----
====
==== BiMap
____
A map that allows users to add key-value pairs and perform lookups from either direction.
____
*{BiMap}* is an interface that defines a bi-directional *Map*, i.e, a *Map* that allows users to execute a lookup from both directions.
Both the keys and the values in a *BiMap* are unique.
*BiMap* extends *MapIterable* and *{MutableBiMap}* extends *MutableMap*.
The standard implementation is *{HashBiMap}*.
.Bi-directional map
====
[source,java]
----
MutableBiMap<Integer, String> biMap =
HashBiMap.newWithKeysValues(1, "1", 2, "2", 3, "3");
----
====
The distinctive methods on *MutableBiMap* are *put*, *forcePut* and *inverse*.
****
##`*MutableBiMap.put()*`##
****
Behaves like *put* on a regular map, except that it throws an exception when you try to add a duplicate value.
[source,java]
----
MutableBiMap<Integer, String> biMap = HashBiMap.newMap();
biMap.put(1, "1"); // behaves like a regular put()
biMap.put(1, "1"); // no effect
biMap.put(2, "1"); // throws IllegalArgumentException since value "1" is already present
----
****
##`*MutableBiMap.forcePut()*`##
****
Behaves like *MutableBiMap.put*, except that it silently removes the map entry with the same value before putting the key-value pair in the map.
[source,java]
----
MutableBiMap<Integer, String> biMap = HashBiMap.newMap();
biMap.forcePut(1, "1"); // behaves like a regular put()
biMap.forcePut(1, "1"); // no effect
biMap.put(1, "2"); // replaces the [1, "1"] pair with [1, "2"]
biMap.forcePut(2, "2"); // removes the [1, "2"] pair before putting
Assert.assertFalse(biMap.containsKey(1));
Assert.assertEquals(HashBiMap.newWithKeysValues(2, "2"), biMap);
----
****
##`*MutableBiMap.inverse()*`##
****
Returns a reversed view of the *BiMap*. Calling *inverse* is an inexpensive operation as the view is already cached in the *BiMap*. This is the recommend method for looking up the key associated to a value using `biMap.inverse().get(value)`.
[source,java]
----
MutableBiMap<Integer, String> biMap =
HashBiMap.newWithKeysValues(1, "1", 2, "2", 3, "3");
MutableBiMap<String, Integer> inverse = biMap.inverse();
Assert.assertEquals("1", biMap.get(1));
Assert.assertEquals(Integer.valueOf(1), inverse.get("1"));
Assert.assertTrue(inverse.containsKey("3"));
Assert.assertEquals(Integer.valueOf(2), inverse.put("2", 4));
----
=== Bag
____
An unordered collection that allows duplicates.
____
A *{Bag}* is a *RichIterable* that allows duplicate elements and efficient querying of the number of occurrences of each element.
It can be sorted or unsorted.
* A *{SortedBag}* is a *Bag* that maintains its elements in sorted order.
* An *{UnsortedBag}* is a *Bag* that maintains its elements in a hash table in an unpredictable order.
*UnsortedBag* has two subinterfaces, *{MutableBag}* and *{ImmutableBag}*.
*SortedBag* has two subinterfaces, *{MutableSortedBag}* and *{ImmutableSortedBag}*.
A *Bag* is conceptually like a *Map* from elements to the number of occurrences of that element.
For example, this list:
----
Apple
Pear
Orange
Orange
Apple
Orange
----
could create this *Bag*:
[cols=",",]
[%autowidth]
|===
|`Pear` |`1`
|`Orange` |`3`
|`Apple` |`2`
|===
.*Bag*
====
[source,java]
----
MutableBag<String> bag =
HashBag.newBagWith("Apple", "Pear", "Orange", "Apple", "Apple", "Orange");
// or
MutableBag<String> bag =
Bags.mutable.with("Apple", "Pear", "Orange", "Apple", "Apple", "Orange");
----
====
These are the distinctive methods on the *Bag* interface.
****
##`*_Bag_.occurrencesOf(Object _item_)*`##
****
Returns the occurrences of a distinct item in the *Bag*.
****
##`*_Bag_.forEachWithOccurrences(ObjIntProcedure)*`##
****
For each distinct item, with the number of occurrences, executes the specified procedure.
****
##`*_Bag_.toMapOfItemToCount()*`##
****
Returns a map with the item type to its count as an Integer.
==== MutableBag
____
A mutable unordered collection allowing duplicates.
____
The *{MutableBag}* interface includes methods for manipulating the number of occurrences of an item.
For example, to determine the number of unique elements in a MutableBag, use the *sizeDistinct* method.
The most common implementation of *MutableBag* is *{HashBag}*.
.*MutableBag*
====
[source,java]
----
MutableBag<String> bag = HashBag.newBagWith("Apple", "Pear", "Orange", "Apple", "Apple", "Orange");
// or
MutableBag<String> bag = Bags.mutable.with("Apple", "Pear", "Orange", "Apple", "Apple", "Orange");
----
====
These are the distinctive methods on *MutableBag*:
****
##`*MutableBag.addOccurrences(T item, int occurrences)*`##
****
Increments the count of the item in the bag by a count specified by occurrences.
****
##`*MutableBag.removeOccurrences(Object item, int occurrences)*`##
****
Decrements the count of the item in the bag by a count specified by occurrences.
****
##`*MutableBag.setOccurrences(T item, int occurrences)*`##
****
Mutates the bag to contain the specified number of occurrences of the item.
==== MutableSortedBag
____
A sorted collection that allows duplicates.
____
A *{MutableSortedBag}* is a **Bag**that maintains order.
It defaults to natural order, but can take a comparator to sort.
The most common implementation of *MutableSortedBag* is *{TreeBag}* which uses a *SortedMap* as its underlying data store.
For example, this *MutableSortedBag* would contain integers sorted in reverse order:
.EC
====
[source,java]
----
MutableSortedBag<Integer> revIntegers =
TreeBag.newBagWith(Collections.reverseOrder(), 4, 3, 3, 2, 2, 2, 1, 1);
// or
MutableSortedBag<Integer> revIntegers =
SortedBags.mutable.with(Collections.reverseOrder(), 4, 3, 3, 2, 2, 2, 1, 1);
----
====
=== StackIterable
____
A collection that maintains "last-in, first-out" order, iterating over elements in reverse insertion order.
____
A *{StackIterable}* is a *RichIterable* enforcing a "last-in, first-out" order; its methods always iterate over elements in reverse insertion order, (beginning with the most-recently added element).
For example the *getFirst* method returns the the last element to have been added - the "top" of the stack.
*StackIterable* has a mutable and an link:#immutable-collections[immutable] subinterface *{MutableStack}* and *{ImmutableStack}*, respectively.
==== MutableStack
____
A mutable collection that maintains "last-in, first-out" order, iterating over elements in reverse insertion order.
____
The most common implementation of *MutableStack* is *{ArrayStack}*.
The closest JCF equivalent to *ArrayStack* is *java.util.Stack*, which extends *Vector* but does not enforce strict LIFO iteration.
The distinctive methods on *MutableStack* are *push*, *pop*, and *peek*.
[cols="1,2"]
[%autowidth]
|===
|`*push*` |Adds a new element to the top of the stack
|`*pop*` |Returns the top (most recently-added) element and removes it from the collection.
|`*pop(int count)*` |Returns a *ListIterable* of the number of elements specified by the count, beginning with the top of the stack.
|`*peek*` |Returns but does not remove the top element.
Note that, on a stack, *getFirst* likewise returns the top element, and that *getLast* throws an exception.
|`*peek(*int count*)*` |Returns a *ListIterable* of the number of elements specified by the count, beginning with the top of the stack; does not remove the elements from the stack.
|`*peekAt(*int index*)*` |Returns the element at index.
|===
*ArrayStack* can replace *java.util.Stack*.
.*Stack* in Java Collections Framework
====
[source,java]
----
Stack stack = new Stack();
stack.push(1);
stack.push(2);
stack.push(3);
----
====
.*ArrayStack* in Eclipse Collections
====
[source,java]
----
MutableStack mutableStack =
ArrayStack.newStackWith(1, 2, 3);
// or
MutableStack mutableStack =
Stacks.mutable.with(1, 2, 3);
----
====
[#multimap-container]
=== Multimap
____
A map-like container that can have multiple values for each key
____
In a *{Multimap}* container, each key can be associated with multiple values.
It is, in this sense, similar to a Map, but one whose values consist of individual collections of a specified type, called the _backing collection_.
A *Multimap* is useful in situations where you would otherwise use **Map**<K, Collection<V>>.
Unlike the other basic Eclipse Collections containers, *Multimap* does not extend *RichIterable*, but resides along with its subinterfaces in a separate API.
The *RichIterable* methods are extended by the backing collection types.
Depending on the implementation, the "values" in a Multimap can be stored in Lists, Sets or Bags.
For example, the *{FastListMultimap}* class is backed by a *UnifiedMap* that associates each key with a *FastList* that preserves the order in which the values are added and allows duplicate to be added.
A *Multimap* is the type returned by the *{groupby-xref}* method.
Here is an example in which we group a list of words by their length, obtaining a *Multimap* with integer (word=length) keys and lists of words having that length for values.
This simple list:
....
here
are
a
few
words
that
are
not
too
long
....
produces a List-backed *Multimap*:
[cols="1,3",options="header"]
[%autowidth]
|===
|key |value<list>
|`1` |`a`
|`3` |`are, few, are, not, too`
|`4` |`here, that, long`
|`5` |`words`
|===
The code that performs this action uses the *{groupBy-xref}* method.
.*MutableListMultimap* using *groupBy* with *{StringFunctions}*)
====
[source,java]
----
MutableList<String> words = Lists.mutable.with("here", "are", "a", "few",
"words", "that", "are", "not", "too", "long");
MutableListMultimap<Integer, String> multimap =
words.groupBy(StringFunctions.length());
----
====
.*MutableListMultimap* using groupBy* with a method reference
====
[source,java]
----
MutableList<String> words = Lists.mutable.with("here", "are", "a", "few",
"words", "that", "are", "not", "too", "long");
MutableListMultimap<Integer, String> multimap =
words.groupBy(String::length);
----
====
The interface *{MutableListMultimap}* extends the *Multimap* interface and tells us the type of its backing collections.
Since *words* is a *MutableList*, the output is a *MutableListMultimap*.
The word "are" is allowed to occur twice in the list at key 3.
If we change *words* to a *MutableSet*, the result will be a *{MutableSetMultimap}*, which will eliminate duplicate entries.
.*MutableSetMultimap* using *groupBy* with *{StringFunctions}*
====
[source,java]
----
MutableSet<String> words = Sets.mutable.with("here", "are", "a", "few",
"words", "that", "are", "not", "too", "long");
MutableSetMultimap<Integer, String> multimap =
words.groupBy(StringFunctions.length());
----
====
.*MutableSetMultimap* using *groupBy* with a method reference
====
[source,java]
----
MutableSet<String> words = Sets.mutable.with("here", "are", "a", "few",
"words", "that", "are", "not", "too", "long");
MutableSetMultimap<Integer, String> multimap =
words.groupBy(String::length);
----
====
With duplicates removed, only four 3-letter words remain.
[cols="1,3",options="header"]
[%autowidth]
|===
|key |value <list>
|`1` |`a`
|`3` |`too`,`are`,`few`,`not`,
|`4` |`long`,`here`,`that`
|`5` |`words`
|===
== Primitive collections
____
Containers for iterating over collections of Java primitives.
____
Eclipse Collections has memory-optimized lists, sets, stacks, maps, and bags for all the primitive types: *int*, *long*, *float*, *char*, *byte*, *boolean*, *short*, and *double*.
The interface hierarchies for primitive types correspond closely with the interface hierarchy for regular *Object* collections.
For example, the collection interfaces for *int* include the following:
[cols=",",options="header]
[%autowidth]
|===
|*interface* |*analogous to*
|*IntIterable* |*RichIterable*
|*MutableIntCollection* |*MutableCollection*
|*IntList* |*ListIterable*
|*MutableIntList* |*MutableList*
|===
There are some common arithmetic operations that can be performed on all primitive collections (except boolean collections).
.Mathematical operations on primitive collections
====
[source,java]
----
MutableIntList intList = IntLists.mutable.with(1, 2, 3);
Assert.assertEquals(6, intList.sum());
Assert.assertEquals(1, intList.min());
Assert.assertEquals(3, intList.max());
Assert.assertEquals(2.0d, intList.average(), 0.0);
Assert.assertEquals(2, intList.median());
// IntList.summaryStatistics() returns IntSummaryStatistics
Assert.assertEquals(6, intList.summaryStatistics().getSum());
Assert.assertEquals(1, intList.summaryStatistics().getMin());
Assert.assertEquals(3, intList.summaryStatistics().getMax());
Assert.assertEquals(2.0d, intList.summaryStatistics().getAverage(), 0.0);
----
====
=== Primitive lists
The primitive list implementations are backed by an array like *FastList*, but with a primitive array instead of an *Object[]*.
They are named *{IntArrayList}*, *{FloatArrayList}* etc.
*{BooleanArrayList}* is a special case.
Current JVMs use one byte per boolean in a *boolean[]* (instead of one bit per boolean).
Thus the *BooleanArrayList* is backed by a *java.util.BitSet* as an optimization.
.our ways to creCreate an *IntArrayList*, use one of the following:
====
[source,java]
----
IntArrayList emptyList = new IntArrayList();
IntArrayList intList = IntArrayList.newListWith(1, 2, 3);
IntArrayList alternate = IntLists.mutable.with(1, 2, 3);
IntArrayList listFromIntIterable = IntArrayList.newListWith(IntHashSet.newSetWith(1, 2, 3));
----
====
=== IntInterval
An *IntInterval* is a range of **int**s that may be iterated over using a step value.
(Similar to *Interval*, but uses primitive ints instead of the wrapper *Integers*.)
[source,java]
----
Assert.assertEquals(IntLists.mutable.with(1, 2, 3), IntInterval.oneTo(3));
Assert.assertEquals(IntLists.mutable.with(1, 3, 5), IntInterval.oneToBy(5, 2));
----
=== Primitive sets
The primitive set implementations are hash-table backed.
They are named *{IntHashSet}*, *{FloatHashSet}* etc.
*{BooleanHashSet}* is implemented using a single integer to hold one of four possible states: ([], [F], [T], or [T, F]).
All other sets use open addressing and quadratic probing.
=== Primitive stacks
Primitive stack implementations are similar to JCF *ArrayStack* but optimized for primitives.
=== Primitive bags
Primitive bag implementations are similar to JCF *HashBag*, but both _item_ and _count_ are primitives.
=== Primitive Maps
There are three types of primitive maps:
* Object-to-primitive (*{ObjectIntHashMap}*, ObjectFloatHashMap etc.)
* Primitive-to-Object (*{IntObjectHashMap}*, FloatObjectHashMap etc.)
* Primitive-to-primitive (*{IntIntHashMap}*, IntLongHashMap etc.)
All the maps use open addressing and quadratic probing.
Primitive maps with numeric value types (not boolean or Object) have a method *addToValue* that adds the given amount to the value at the given key and returns the updated value.
.*addToValue*
====
[source,java]
----
MutableByteIntMap map =
new ByteIntHashMap();
Assert.assertEquals(1, map.addToValue((byte) 0, 1));
Assert.assertEquals(ByteIntHashMap.newWithKeysValues((byte) 0, 1), map);
Assert.assertEquals(11, map.addToValue((
byte) 0, 10));
Assert.assertEquals(ByteIntHashMap.newWithKeysValues((byte) 0, 11), map);
----
====
All primitive collections have immutable counterparts, and unmodifiable and synchronized wrappers.
See the link:#protective-wrappers[Protecting collections] topic for more information.
[#immutable-collections]
== Immutable collections
____
A read-only snapshot of a collection; once created, it can never be modified.
____
All of the basic containers in Eclipse Collections have interfaces for both _mutable_ and _immutable_ (unchangeable) forms.
This departs somewhat from the model of the Java Collections Framework, in which most containers are mutable.
An _immutable collection_ is just that - once created, it can never be modified, retaining the same internal references and data throughout its lifespan.
An immutable collection is equal to a corresponding mutable collection with the same contents; a *MutableList* and an ImmutableList can be equal.
Because its state does not change over time, an immutable collection is _always_ thread-safe.
Using immutable collections where feasible can serve to make your code easier to read and understand.
All of the interfaces and implementations discussed so far in this topic have been _mutable_ versions of their respective types.
Each of these containers has an immutable counterpart.
These are the corresponding interfaces:
[cols=","]
[%autowidth]
|===
|_Mutable types_ |_Immutable types_
|*MutableList* |*ImmutableList*
|*MutableSet* |*ImmutableSet*
|*MutableBag* |*ImmutableBag*
|*MutableMap* |*ImmutableMap*
|*MutableMultimap* |*ImmutableMultimap*
|*MutableStack* |*ImmutableStack*
|===
The method that returns an immutable collection for all container types is *toImmutable()*:
****
##`*MutableCollection.toImmutable(): ImmutableCollection*`##
****
Returns an immutable copy of a type corresponding to the source *MutableCollection*.
****
##`*StackIterable.toImmutable(): ImmutableStack*`##
****
Returns an immutable copy of a *MutableStack*, returns the same iterable for an *ImmutableStack*.
****
##`*ListIterable.toImmutable(): ImmutableList*`##
****
Returns an immutable copy of a *MutableList*, returns the same iterable for an *ImmutableList*.
****
##`*SortedMapIterable.toImmutable(): ImmutableSortedMap*`##
****
Returns an immutable copy of a *MutableSortedMap*, returns the same iterable for an *ImmutableSortedMap*.
****
##`*UnsortedMapIterable.toImmutable(): ImmutableMap*`##
****
Returns an immutable copy of a *MutableMap*, returns the same iterable for an *ImmutableMap*.
An immutable-collection interface lacks mutating methods, such as *add* and *remove*.
Instead, immutable collections have methods that return new, immutable copies with or without specified elements:
****
##`*ImmutableCollection.
newWith(_element_): ImmutableCollection*`##
****
Returns a new immutable copy of *ImmutableCollection* with _element_ added.
****
##`*ImmutableCollection.
newWithAll(Iterable): ImmutableCollection*`##
****
Returns a new immutable copy of *ImmutableCollection* with the elements of *Iterable* added.
****
##`*ImmutableCollection.
newWithout(_element_): ImmutableCollection*`##
****
Returns a new immutable copy of *ImmutableCollection* with _element_ removed.
****
##`*ImmutableCollection.
newWithoutAll(Iterable): ImmutableCollection*`##
****
Returns a new immutable copy of *ImmutableCollection* with the elements of *Iterable* removed.
Note that the iteration methods of an immutable container - such as *select*, *reject*, and *collect* - also produce new immutable collections.
=== Immutable collection factory classes
The factory classes *Lists*, *Sets*, *Bags*, and *Maps* create immutable collections.
These factories also provide methods for creating fixed-size collections, which have been superseded by immutable collections.
.Factory classes for immutable collections
====
[source,java]
----
ImmutableList<Integer> immutableList = Lists.immutable.of(1, 2, 3);
ImmutableSet<Integer> immutableSet = Sets.immutable.of(1, 2, 3);
Bag<Integer> immutableBag = Bags.immutable.of(1, 2, 2, 3);
ImmutableMap<Integer, String> immutableMap =
Maps.immutable.of(1, "one", 2, "two", 3, "three");
----
====
These factories highlight yet another benefit of immutable collections: they let you create efficient containers that are sized according to their contents.
In cases where there are many, even millions of collections, each with a size less than 10, this is an important advantage.
=== Growing and shrinking immutable collections
There are no mutating methods like *add(), addAll(), remove()* or *removeAll()* on immutable collection interfaces in Eclipse Collections.
However, we may want to add or remove elements.
Methods like *newWith()*, *newWithout()*, *newWithAll()* and *newWithoutAll()* allow for safe copying of immutable collections.
For ImmutableMap implementations, the methods are named *newWithKeyValue()*, *newWithAllKeyValues()*, *newWithoutKey()* and *newWithoutAllKeys()*.
.Add elements to an immutable list
====
[source,java]
----
// persons is an mutable list: MutableList<Person> persons
// Person is a class with attributes name, age and address
PartitionMutableList<Person> partitionedFolks =
persons.partition(person -> person.getAge() >= 18); // defines a partition pattern
ImmutableList<Person> list0 = Lists.immutable.empty();
ImmutableList<Person> list1 = list0.newWith(new Person(...)); // add a single element to the new immutable list
ImmutableList<Person> adults = list1.newWithAll(partitionedFolks.getSelected()); // add none, one or more objects to the new immutable list
I
ImmutableSet<String> set0 = Sets.immutable.empty();
ImmutableSet<String> set1 = set0.newWith("1"); // add a single element to the new immutable set
ImmutableSet<String> set2 = set1.newWithAll(Sets.mutable.with("2")); // add none, one or more objects to the new immutable set
----
====
For ImmutableMap implementations, the methods are named *newWithKeyValue()*, *newWithAllKeyValues()*, *newWithoutKey(}* and *newWithoutAllKeys()*.
.Add elements to an immutable map
====
[source,java]
----
ImmutableMap<String, String> map0 = Maps.immutable.empty();
ImmutableMap<String, String> map1 = map0.newWithKeyValue("1", "1");
ImmutableMap<String, String> map2 = map1.newWithAllKeyValues(Lists.mutable.with(Tuples.pair("2", "2")))
----
====
These methods are available on the primitive containers too though we are missing some symmetry in our immutable primitive map containers.
We *do not* currently have *newWithAllKeyValues()* on immutable primitive maps.
The corresponding feature request is https://github.com/eclipse/eclipse-collections/issues/344[here].
.Immutable primitive containers
====
[source,java]
----
ImmutableIntList list0 = IntLists.immutable.empty();
ImmutableIntList list1 = list0.newWith(1);
ImmutableIntList list2 = list1.newWithAll(IntLists.mutable.with(2));
ImmutableIntSet set0 = IntSets.immutable.empty();
ImmutableIntSet set1 = set0.newWith(1);
ImmutableIntSet set2 = set1.newWithAll(IntSets.mutable.with(2));
ImmutableIntIntMap map0 = IntIntMaps.immutable.empty();
ImmutableIntIntMap map1 = map0.newWithKeyValue(1, 1);
----
====
== Creating collection containers
Eclipse Collections has many *iteration methods* that return new collection containers from existing ones.
These include the *select* and *collect* methods, along with their specialized variants, such as *partition* and *groupBy*.
There are also multiple ways to instantiate new containers, from scratch or *by conversion* from other container types.
=== Creating mutable collections
If you know the implementation class of the container (for example, *FastList* or *UnifiedSet*), you can use either of these two techniques:
* Call the class constructor; for example, **FastList**<String> names = new *FastList*;
* Call a factory method on the collection class, such as *newList* or *newListWith*; for example, **MutableList**<V> result = *FastList.newList*(this.size);
If, however, the specific collection class to implement is unknown, you can call a factory class to create the container.
Eclipse Collections container-factory classes are named for the plural of the respective container name.
For example, for a *List*, the factory class is *{Lists}*; a *Set* is created by the class *{Sets}*.
You can specify the content and mutable state of the container in parameters.
This approach to container creation, when used consistently, has the added benefit of being more readable.
.Container factory classes
====
[source,java]
----
MutableList<String> emptyList = Lists.mutable.empty(); // creates an empty list
MutableList<String> list = Lists.mutable.with("a", "b", "c"); // creates a list of elements a,b,c
MutableList<String> list_a = Lists.mutable.of("a", "b", "c"); // creates a list of elements a,b,c
MutableSet<String> emptySet = Sets.mutable.empty();
// creates an empty set
MutableSet<String> set = Sets.mutable.with("a", "b", "c"); // creates a set of elements a,b,c
MutableSet<String> set_a = Sets.mutable.of("a", "b", "c"); // creates a set of elements a,b,c
MutableBag<String> emptyBag= Bags.mutable.empty();
// creates an empty bag
MutableBag<String> bag = Bags.mutable.with("a", "b", "c"); // creates a bag of elements a,b,c
MutableBag<String> bag_a = Bags.mutable.of("a", "b", "c"); // creates a bag of elements a,b,c
----
====
[cols=","]
[%autowidth]
|===
|_Container_ |_Factory_
|*List* |*Lists*
|*Set* |*Sets, HashingStrategySets*
|*Bag* |*Bags*
|*Stack* |*Stacks*
|*SortedBag* |*SortedBags*
|*SortedSet* |*SortedSets*
|*Map* |*Maps, HashingStrategyMaps*
|*SortedMap* |*SortedMaps*
|*BiMap* |*BiMaps*
|*Multimap* |*Multimaps*
|===
You can also create a mutable container from a instance of the same type:
[[newempty--mutablecollection]]
****
##`*newEmpty() : MutableCollection*`##
****
Creates a new, empty, and mutable container of the same collection type.
For example, if this instance is a *FastList*, this method will return a new empty *FastList*.
If the class of this instance is *immutable* or fixed size (for example, a singleton *List*) then a mutable alternative to the class is returned.
=== Creating immutable collections
There are two ways to create an immutable *List*, *Set*, *Bag*, *Stack* or *Map*.
* Create a mutable version of a container type (as described in the previous section), and then call *toImmutable()* on the result to create an immutable copy.
* Use factory classes to create _immutable_ versions of classes.
.Create an immutable container by calling *toImmutable()*
====
[source,java]
----
ImmutableList<String> list =
Lists.mutable.with("a", "b", "c").toImmutable(); // creates a list of elements a,b,c
ImmutableSet<String> set0 =
Lists.mutable.with("a", "b", "a", "c").toSet().toImmutable(); // creates a set of elements a,b,c
ImmutableSet<String> set1 =
Sets.mutable.with("a", "b", "a", "c").toImmutable(); // creates a set of elements a,b,c
ImmutableMap<Integer, String> map =
Maps.mutable.with(1, "a", 2, "b", 3, "c").toImmutable(); // creates a map with keys 1,2,3
----
====
.Create an immutable container using factory classes
====
[source,java]
----
ImmutableList<String> emptyList_i =
Lists.immutable.empty(); // creates an empty list
ImmutableList<String> list_b =
Lists.immutable.with("a", "b", "c"); // creates a list of elements a,b,c
ImmutableList<String> list_c =
Lists.immutable.of("a", "b", "c"); // creates a list of elements a,b,c
ImmutableSet<String> emptySet_i =
Sets.immutable.empty(); // creates an empty Set
ImmutableSet<String> set_b =
Sets.immutable.with("a", "b", "c"); // creates a Set of elements a,b,c
ImmutableSet<String> set_c =
Sets.immutable.of("a", "b", "c"); // creates a Set of elements a,b,c
ImmutableMap<Integer, String> emptyMap_i =
Maps.immutable.empty(); // creates an empty map
ImmutableMap<Integer, String> map_b =
Maps.immutable.with(1, "a", 2, "b", 3, "c"); // creates a map with keys 1,2,3
ImmutableMap<Integer, String> map_c =
Maps.immutable.of(1, "a", 2, "b", 3, "c"); // creates a map with keys 1,2,3
----
====
=== Creating primitive collections
The techniques for creating mutable or immutable primitive containers are the same as those for object collections.
* If you know the implementation class, you can call its constructor or its factory method.
* Otherwise use the applicable factory class named for the plural of the type, that is, with the pattern:
`*< _primitivetype_ >< _containertype_ >s*`.
For example, to create an *IntList*, you would use the *{IntLists}* factory for the appropriate methods; for a *DoubleSet*, you would call in methods in *{DoubleSets}*.
.Create an immutable primitive class using a factory
====
[source,java]
----
MutableIntList emptyList_pm =
IntLists.mutable.empty(); // creates an empty list
MutableIntList list_d =
IntLists.mutable.with(1, 2, 3); // creates a list of elements a,b,c
MutableIntList list_e =
IntLists.mutable.of(1, 2, 3); // creates a list of elements a,b,c
ImmutableIntList emptyList_pi =
IntLists.immutable.empty(); // creates an empty list
ImmutableIntList list_f =
IntLists.immutable.with(1, 2, 3); // creates a list of elements a,b,c
ImmutableIntList list_g =
IntLists.immutable.of(1, 2, 3); // creates a list of elements a,b,c
----
====
[cols=",,,,",options="header",]
[%autowidth]
|===
|`*type*` \ container|List |Set |Bag |Stack
|`*boolean*` |*BooleanLists* |*BooleanSets* |*BooleanBags* |*BooleanStacks*
|`*byte*` |*ByteLists* |*ByteSets* |*ByteBags* |*ByteStacks*
|`*char*` |*CharLists* |*CharSets* |*CharBags* |*CharStacks*
|`*short*` |*ShortLists* |*ShortSets* |*ShortBags* |*ShortStacks*
|`*int*` |*IntLists* |*IntSets* |*IntBags* |*IntStacks*
|`*float*` |*FloatLists* |*FloatSets* |*FloatBags* |*FloatStacks*
|`*long*` |*LongLists* |*LongSets* |*LongBags* |*LongStacks*
|`*double*` |*DoubleLists* |*DoubleSets* |*DoubleBags* |*DoubleStacks*
|===
== Converting collections
The following methods can be used to convert one container type to another.
All of these methods are on *RichIterable*.
To create immutable and fixed-size collections, see <<immutable-collections>>.
****
##`*toList() : MutableList*`##
****
Converts the collection to the default *MutableList* implementation (*FastList*).
****
##`*toSet() : MutableSet*`##
****
Converts the collection to the default *MutableSet* implementation (*UnifiedSet*).
****
##`*toBag() : MutableBag*`##
****
Converts the collection to the default *MutableBag* implementation (*HashBag*).
****
##`*toMap(_keyFunction_, _valueFunction_): MutableMap*`##
****
Converts the collection to the default *MutableMap* implementation (*UnifiedMap*) using the specified _keyFunctions_ and _valueFunctions_.
[[tosortedlist--mutablelist]]
****
##`*toSortedList() : MutableList*`##
****
Converts the collection to the default *MutableList* implementation (*FastList*) and sorts it using the natural order of the elements.
****
##`*toSortedList(Comparator) : MutableList*`##
****
Converts the collection to the default *MutableList* implementation (*FastList*) and sorts it using the specified *Comparator*.
These methods always return new _mutable_ copies: for example, calling *toList* on a *FastList*, returns a new *FastList*.
== Protective wrappers
____
Wrapper classes providing read-only or thread-safe views of a collection.
____
[#unmodifiable-collections]
=== Unmodifiable collections
In both the JCF and Eclipse Collections, a collection may be rendered _unmodifiable_.
In Eclipse Collections, this is done by means of the *asUnmodifiable* method, which returns a read-only view of the calling collection.
This means that the mutating methods of the collection (e.g., *add*, *remove*) are still present, but throw exceptions if called.
****
##`*MutableCollection.
asUnmodifiable(): MutableCollection _(read-only)_*`##
****
Returns a read-only view of the source collection.
****
##`*MutableIntCollection.asUnmodifiable(): MutableIntCollection.__(read-only)__*`##
****
Returns a read-only view of the source primitive collection.
There are similar methods for each primitive type, e.g., *MutableFloatCollection.asUnmodifiable()*
=== Synchronized collections
Eclipse Collections provides a wrapper for rendering a modifiable but thread-safe view that holds a lock when a method is called and releases the lock upon completion.
****
##`*MutableCollection.
asSynchronized(): MutableCollection*`##
****
Returns a synchronized copy of the source collection.
****
##`*MutableIntCollection.
asSynchronized(): MutableIntCollection.*`##
****
Returns a synchronized copy of the source primitive collection.
There are similar methods for each primitive type, e.g., *MutableFloatCollection.asSynchronized()*.
[cols="3,^1,>3",]
|===
|xref:1-Iteration_Patterns.adoc[previous: Iteration patterns]|xref:0-RefGuide.adoc[top] |xref:3-Code_Blocks.adoc[next: Code blocks]
|===
|