/* * Copyright (c) 2011-2017 Isode Limited. * All rights reserved. * See the COPYING file for more information. */ #pragma once #include <algorithm> #include <type_traits> namespace Swift { template<typename V1, typename V2, typename... Rest> std::size_t getReserveAmount(V1 const& v1, V2 const& v2, Rest const&... rest) SWIFTEN_NOEXCEPT { // Total size of the concatenated output is equal to sum of sizes of all the containers that were passed in. // Two first can be accessed directly: std::size_t sizeSum = v1.size() + v2.size(); // While the rest needs to be calculated in parameter pack expansion: // http://stackoverflow.com/a/25683817/61574 // Pack expansion can occur inside a braced-init-list - dummy array is used for that (with side effect of accumulating the size) int dummy[] = {0, ((sizeSum += rest.size()), void(), 0)...}; (void)dummy; // dummy is unused - suppress compiler warning return sizeSum; } template<typename V1, typename V2, typename... Rest> typename std::decay<V1>::type concat(V1&& v1, V2&& v2, Rest&&... rest) { // Create the output container and reserve the required amount of space typename std::decay<V1>::type v; v.reserve(getReserveAmount(v1, v2, rest...)); // Insert the elements from input containers to output. Directly for the first two: v.insert(v.cend(), v1.cbegin(), v1.cend()); v.insert(v.cend(), v2.cbegin(), v2.cend()); // Use pack expansion for the others: int dummy[] = {0, ((v.insert(v.cend(), rest.cbegin(), rest.cend())), void(), 0 )...}; (void)dummy; return v; } }