The class basic_json has an Allocator template parameter and an allocator_type member that indicates
that it is allocator aware. Allocator must be a Scoped Allocator, that is, an allocator
that applies not only to a basic_json's data member, but also to its data member's elements.
In particular, Allocator must be either a stateless allocator,
a std::pmr::polymorphic_allocator,
or a std::scoped_allocator_adaptor.
Non-propagating stateful allocators, such as the Boost.Interprocess allocators,
must be wrapped by a std::scoped_allocator_adaptor.
Every constructor has a version that accepts an allocator argument, this is required for allocator propogation. A long string, byte string, array or object makes use of the allocator argument to allocate storage. For the other data members (short string, number, boolean, or null) the allocator argument is ignored.
A long string, byte string, array and object data member all contain a pointer ptr obtained
from an earlier call to allocate. ptr points to storage that contains both
the data representing the long string, byte string, array or object, and the
allocator used to allocate that storage. To later deallocate that storage, the allocator is
retrieved with ptr->get_allocator().
The allocator applies not only to a basic_json's data member, but also to its data member's elements. For example:
char buffer[1024];
std::pmr::monotonic_buffer_resource pool{ std::data(buffer), std::size(buffer) };
std::pmr::polymorphic_allocator<char> alloc(&pool);
std::string key = "key too long for short string";
std::string value = "string too long for short string";
jsoncons::pmr::json j{ jsoncons::json_object_arg, alloc };
assert(j.get_allocator().resource() == &pool);
j.try_emplace(key, value);
auto it = std::search(std::begin(buffer), std::end(buffer), key.begin(), key.end());
assert(it != std::end(buffer));
it = std::search(std::begin(buffer), std::end(buffer), value.begin(), value.end());
assert(it != std::end(buffer));The copy constructor
Json j1(j);
constructs j1 from the contents of j. If j holds a long string, bytes string, array or object,
copy construction applies allocator traits select_on_container_copy_construction to
the allocator from j (since 0.178.0) For example:
char buffer[1024];
std::pmr::monotonic_buffer_resource pool{ std::data(buffer), std::size(buffer) };
std::pmr::polymorphic_allocator<char> alloc(&pool);
jsoncons::pmr::json j{ "String too long for short string", alloc };
assert(j.is_string());
assert(j.get_allocator().resource() == &pool);
jsoncons::pmr::json j1(j);
assert(j1 == j);
assert(j1.get_allocator() == std::allocator_traits<std::pmr::polymorphic_allocator<char>>::
select_on_container_copy_construction(j.get_allocator()));
assert(j1.get_allocator() == std::pmr::polymorphic_allocator<char>{}); // expected result for pmr allocatorsThe allocator-extended copy constructor
Json j1(j, alloc);
constructs j1 from the contents of j using alloc as the allocator. If j holds a long string, bytes string, array or object,
copy construction uses allocator alloc for allocating storage, otherwise alloc is ignored. For example:
char buffer[1024];
std::pmr::monotonic_buffer_resource pool{ std::data(buffer), std::size(buffer) };
std::pmr::polymorphic_allocator<char> alloc(&pool);
char buffer1[1024];
std::pmr::monotonic_buffer_resource pool1{ std::data(buffer1), std::size(buffer1) };
std::pmr::polymorphic_allocator<char> alloc1(&pool1);
jsoncons::pmr::json j{ "String too long for short string", alloc };
assert(j.is_string());
assert(j.get_allocator().resource() == &pool);
jsoncons::pmr::json j1(j, alloc1);
assert(j1 == j);
assert(j1.get_allocator().resource() == &pool1);The move constructor
Json j1(std::move(j));
constructs j1 by taking the contents of j, which has either a pointer or a trivially copyable value,
and replaces it with null. For example:
char buffer[1024];
std::pmr::monotonic_buffer_resource pool{ std::data(buffer), std::size(buffer) };
std::pmr::polymorphic_allocator<char> alloc(&pool);
jsoncons::pmr::json j{ "String too long for short string", alloc};
assert(j.is_string());
assert(j.get_allocator().resource() == &pool);
jsoncons::pmr::json j1(std::move(j));
assert(j1.is_string());
assert(j1.get_allocator().resource() == &pool);
assert(j.is_null());
The allocator-extended move constructor
Json j1(std::move(j), alloc);
constructs j1 with a copy of the data member j, using alloc as the allocator. For example:
char buffer[1024];
std::pmr::monotonic_buffer_resource pool{ std::data(buffer), std::size(buffer) };
std::pmr::polymorphic_allocator<char> alloc(&pool);
char buffer1[1024];
std::pmr::monotonic_buffer_resource pool1{ std::data(buffer1), std::size(buffer1) };
std::pmr::polymorphic_allocator<char> alloc1(&pool1);
jsoncons::pmr::json j{ "String too long for short string", alloc };
assert(j.is_string());
assert(j.get_allocator().resource() == &pool);
jsoncons::pmr::json j1(std::move(j), alloc1);
assert(j1.is_string());
assert(j1.get_allocator().resource() == &pool1);If the left side is a long string, byte string, array or object, uses its allocator,
otherwise, if the right side is a long string, byte string, array or object,
constructs a copy of the right side as if using copy construction, i.e.
applies allocator traits select_on_container_copy_construction to the
right side allocator. For example:
char buffer[1024];
std::pmr::monotonic_buffer_resource pool{ std::data(buffer), std::size(buffer) };
std::pmr::polymorphic_allocator<char> alloc(&pool);
char buffer1[1024];
std::pmr::monotonic_buffer_resource pool1{ std::data(buffer1), std::size(buffer1) };
std::pmr::polymorphic_allocator<char> alloc1(&pool1);
jsoncons::pmr::json j{ "String too long for short string", alloc };
assert(j.is_string());
assert(j.get_allocator().resource() == &pool);
// copy long string to number
jsoncons::pmr::json j1{10};
j1 = j;
assert(j1.is_string());
assert(j1.get_allocator() == std::allocator_traits<std::pmr::polymorphic_allocator<char>>::
select_on_container_copy_construction(j.get_allocator()));
assert(j1.get_allocator() == std::pmr::polymorphic_allocator<char>{});
// copy long string to array
jsoncons::pmr::json j2{ jsoncons::json_array_arg, {1,2,3,4},
jsoncons::semantic_tag::none, alloc1};
j2 = j;
assert(j2.is_string());
assert(j2.get_allocator().resource() == &pool1);basic_json does not check allocator traits propagate_on_container_copy_assignment.
In the case of arrays and objects, basic_json leaves it to the implementing types (by
default std::vector) to conform to the traits.
If either j or j1 are a long string, byte string, array, or object, basic_json move assignment
j1 = std::move(j);
swaps the two data member values, such that two pointers are swapped or a pointer and a
trivially copyable value are swapped. Otherwise, move assignment copies j's value to j1
and leaves j unchanged. For example:
char buffer[1024];
std::pmr::monotonic_buffer_resource pool{ std::data(buffer), std::size(buffer) };
std::pmr::polymorphic_allocator<char> alloc(&pool);
jsoncons::pmr::json j{ "String too long for short string", alloc };
assert(j.is_string());
assert(j.get_allocator().resource() == &pool);
jsoncons::pmr::json j1{10};
assert(j1.is_number());
j1 = std::move(j);
assert(j1.is_string());
assert(j1.get_allocator().resource() == &pool);
assert(j.is_number());basic_json does not check allocator traits propagate_on_container_move_assignment.
It simply swaps pointers, or a pointer and a trivial value, or copies a trivial value.
basic_json is compatible with boost boost::interprocess::offset_ptr,
provided that the implementing containers for arrays and objects are also compatible. In the example below, boost containers are used.
#include <boost/interprocess/managed_shared_memory.hpp>
#include <boost/interprocess/containers/vector.hpp>
#include <boost/interprocess/allocators/allocator.hpp>
#include <boost/interprocess/containers/string.hpp>
#include <cstdlib> //std::system
#include <jsoncons/json.hpp>
#include <scoped_allocator>
using shmem_allocator = boost::interprocess::allocator<int,
boost::interprocess::managed_shared_memory::segment_manager>;
using cust_allocator = std::scoped_allocator_adaptor<shmem_allocator>;
struct boost_sorted_policy
{
template <typename KeyT,typename Json>
using object = jsoncons::sorted_json_object<KeyT,Json,boost::interprocess::vector>;
template <typename Json>
using array = jsoncons::json_array<Json,boost::interprocess::vector>;
template <typename CharT,typename CharTraits,typename Allocator>
using member_key = boost::interprocess::basic_string<CharT, CharTraits, Allocator>;
};
using cust_json = jsoncons::basic_json<char,boost_sorted_policy, cust_allocator>;
int main(int argc, char *argv[])
{
typedef std::pair<double, int> MyType;
if (argc == 1) // Parent process
{
//Remove shared memory on construction and destruction
struct shm_remove
{
shm_remove() { boost::interprocess::shared_memory_object::remove("MySharedMemory"); }
~shm_remove() noexcept { boost::interprocess::shared_memory_object::remove("MySharedMemory"); }
} remover;
//Construct managed shared memory
boost::interprocess::managed_shared_memory segment(boost::interprocess::create_only,
"MySharedMemory", 65536);
//Initialize shared memory STL-compatible allocator
const shmem_allocator alloc(segment.get_segment_manager());
// Create json value with all dynamic allocations in shared memory
cust_json* j = segment.construct<cust_json>("MyJson")(jsoncons::json_array_arg, alloc);
j->push_back(10);
cust_json o(jsoncons::json_object_arg, alloc);
o.try_emplace("category", "reference");
o.try_emplace("author", "Nigel Rees");
o.insert_or_assign("title", "Sayings of the Century");
o.insert_or_assign("price", 8.95);
j->push_back(o);
cust_json a(jsoncons::json_array_arg, 2, cust_json(jsoncons::json_object_arg, alloc),
jsoncons::semantic_tag::none, alloc);
a[0]["first"] = 1;
j->push_back(a);
std::pair<cust_json*, boost::interprocess::managed_shared_memory::size_type> res;
res = segment.find<cust_json>("MyJson");
std::cout << "Parent process:\n";
std::cout << pretty_print(*(res.first)) << "\n\n";
//Launch child process
std::string s(argv[0]); s += " child ";
if (0 != std::system(s.c_str()))
return 1;
//Check child has destroyed all objects
if (segment.find<MyType>("MyJson").first)
return 1;
}
else // Child process
{
//Open managed shared memory
boost::interprocess::managed_shared_memory segment(boost::interprocess::open_only,
"MySharedMemory");
std::pair<cust_json*, boost::interprocess::managed_shared_memory::size_type> res;
res = segment.find<cust_json>("MyJson");
if (res.first != nullptr)
{
std::cout << "Child process:\n";
std::cout << pretty_print(*(res.first)) << "\n";
}
else
{
std::cout << "Result is null\n";
}
//We're done, delete all the objects
segment.destroy<cust_json>("MyJson");
}
return 0;
}Output:
Parent process:
[
10,
{
"author": "Nigel Rees",
"category": "reference",
"price": 8.95,
"title": "Sayings of the Century"
},
[
{
"first": 1
},
{}
]
]
Child process:
[
10,
{
"author": "Nigel Rees",
"category": "reference",
"price": 8.95,
"title": "Sayings of the Century"
},
[
{
"first": 1
},
{}
]
]
An allocator-aware variant type
C++ named requirements: Allocator