Swift Collections is an open-source package of data structure implementations for the Swift programming language.
Read more about the package, and the intent behind it, in the announcement on swift.org.
The package currently provides the following implementations:
-
Deque<Element>
, a double-ended queue backed by a ring buffer. Deques are range-replaceable, mutable, random-access collections. -
OrderedSet<Element>
, a variant of the standardSet
where the order of items is well-defined and items can be arbitrarily reordered. Uses aContiguousArray
as its backing store, augmented by a separate hash table of bit packed offsets into it. -
OrderedDictionary<Key, Value>
, an ordered variant of the standardDictionary
, providing similar benefits.
The following data structures are currently being worked on but they aren't ready for inclusion in a tagged release:
Heap
andPriorityQueue
, min-max heaps backed by an array.SortedSet
andSortedDictionary
, sorted collections backed by in-memory persistent b-trees.HashSet
andHashMap
, persistent hashed collections implemented as Compressed Hash-Array Mapped Prefix-Trees (CHAMP).BitArray
andBitSet
, dynamic bit vectors.SparseSet
, a constant time set construct, trading off memory for speed.
Swift Collections uses the same modularization approach as Swift Numerics: it provides a standalone module for each thematic group of data structures it implements. For instance, if you only need a double-ended queue type, you can pull in only that by importing DequeModule
. OrderedSet
and OrderedDictionary
share much of the same underlying implementation, so they are provided by a single module, called OrderedCollections
. However, there is also a top-level Collections
module that gives you every collection type with a single import statement:
import Collections
var deque: Deque<String> = ["Ted", "Rebecca"]
deque.prepend("Keeley")
deque.append("Nathan")
print(deque) // ["Keeley", "Ted", "Rebecca", "Nathan"]
The Swift Collections package is source stable. The version numbers follow Semantic Versioning -- source breaking changes to public API can only land in a new major version.
The public API of version 1.0 of the swift-collections
package consists of non-underscored declarations that are marked public
in the Collections
, DequeModule
and OrderedCollections
modules.
Interfaces that aren't part of the public API may continue to change in any release, including patch releases. If you have a use case that requires using underscored APIs, please submit a Feature Request describing it! We'd like the public interface to be as useful as possible -- although preferably without compromising safety or limiting future evolution.
By "underscored declarations" we mean declarations that have a leading underscore anywhere in their fully qualified name. For instance, here are some names that wouldn't be considered part of the public API, even if they were technically marked public:
FooModule.Bar._someMember(value:)
(underscored member)FooModule._Bar.someMember
(underscored type)_FooModule.Bar
(underscored module)FooModule.Bar.init(_value:)
(underscored initializer)
Note that contents of the Tests
, Utils
and Benchmarks
subdirectories aren't public API. We don't make any source compatibility promises about them -- they may change at whim, and code may be removed in any new release. Do not rely on anything about them.
Future minor versions of the package may update these rules as needed.
We'd like this package to quickly embrace Swift language and toolchain improvements that are relevant to its mandate. Accordingly, from time to time, we expect that new versions of this package will require clients to upgrade to a more recent Swift toolchain release. (This allows the package to make use of new language/stdlib features, build on compiler bug fixes, and adopt new package manager functionality as soon as they are available.) Requiring a new Swift release will only need a minor version bump.
To use this package in a SwiftPM project, you need to set it up as a package dependency:
// swift-tools-version:5.6
import PackageDescription
let package = Package(
name: "MyPackage",
dependencies: [
.package(
url: "https://github.com/apple/swift-collections.git",
.upToNextMajor(from: "1.0.3") // or `.upToNextMinor
)
],
targets: [
.target(
name: "MyTarget",
dependencies: [
.product(name: "Collections", package: "swift-collections")
]
)
]
)
We have a dedicated Swift Collections Forum where people can ask and answer questions on how to use or work on this package. It's also a great place to discuss its evolution.
If you find something that looks like a bug, please open a Bug Report! Fill out as many details as you can.
We have some basic documentation on package internals that will help you get started.
By submitting a pull request, you represent that you have the right to license your contribution to Apple and the community, and agree by submitting the patch that your contributions are licensed under the Swift License, a copy of which is provided in this repository.
- Submit a PR with your change. If there is an existing issue for the bug you're fixing, please include a reference to it.
- Make sure to add tests covering whatever changes you are making.
- Raise a Feature Request. Discuss why it would be important to implement it.
- Submit a PR with your implementation, participate in the review discussion.
- When there is a consensus that the feature is desirable, and the implementation works well, it is fully tested and documented, then it will be merged.
- Rejoice!
- Start a topic on the forum, explaining why you believe it would be important to implement the data structure. This way we can figure out if it would be right for the package, discuss implementation strategies, and plan to allocate capacity to help.
- When maintainers agreed to your implementation plan, start work on it, and submit a PR with your implementation as soon as you have something that's ready to show! We'd love to get involved as early as you like.
- Participate in the review discussion, and adapt the code accordingly. Sometimes we may need to go through several revisions! This is fine -- it makes the end result that much better.
- When there is a consensus that the feature is ready, and the implementation is fully tested and documented, the PR will be merged by a maintainer.
- Celebrate! You've achieved something great!
Like all Swift.org projects, we would like the Swift Collections project to foster a diverse and friendly community. We expect contributors to adhere to the Swift.org Code of Conduct. A copy of this document is available in this repository.
The current code owner of this package is Karoy Lorentey (@lorentey). You can contact him on the Swift forums, or by writing an email to klorentey at apple dot com. (Please keep it related to this project.)
In case of moderation issues, you can also directly contact a member of the Swift Core Team.