From the point-of-view of serialization/deserialization, the two definitions are equivalent.
The resulting JSON strings (or any other format) will be the same.
From the point-of-view of programming, there is an important difference: Structs are nominally typed
and named tuples are structurally typed (confusingly, structs are not structurally typed).
In plain language, that means that the compiler will regard this as absolutely equivalent to Person, the named tuple:
Structural typing also means that you can declare new types on-the-fly. For instance, in order
to create a Person named tuple, you don't actually have to declare it at all. The following will do:
In this example, Person is recursively defined (because of the field children).
This is impossible to accomplish using structural typing and rfl::NamedTuple, just like it is impossible to have a recursively defined lambda function.
Fields inside the named tuple can be accessed using rfl::get or the .get method:
constautoperson=rfl::Field<"first_name",std::string>("Homer")*rfl::Field<"last_name",std::string>("Simpson")*rfl::Field<"birthday",rfl::Timestamp<"%Y-%m-%d">>("1987-04-19");// OK in most circumstances (there are restrictions// due to the way C++ templates work).constautofirst_name=person.get<"firstName">();// Always OKconstautofirst_name=person.templateget<"firstName">();// Always OKconstautofirst_name=rfl::get<"firstName">(person);
Fields can also be iterated over at compile-time using the apply() method:
autoperson=rfl::Field<"first_name",std::string>("Bart")*rfl::Field<"last_name",std::string>("Simpson");person.apply([](constauto&f){autofield_name=f.name();constauto&value=*f.value();});person.apply([]<typenameField>(Field&f){// The field name can also be obtained as a compile-time constant.constexprautofield_name=Field::name();usingfield_pointer_type=typenameField::Type;field_pointer_type*value=f.value();});
Named tuples also contain compile-time monadic operations.
.transform(f) expects a function f of type Field -> Field.
transform then applies that function to each field of the named tuple.
It can be used to change either the values or the names of the fields, but
not their overall number.
constautolisa=Person{.first_name="Lisa",.last_name="Simpson",.age=8};constautoto_bart=[](autofield){ifconstexpr(decltype(field)::name()=="first_name"){field="Bart";returnfield;}elseifconstexpr(decltype(field)::name()=="age"){field=10;returnfield;}else{returnfield;}};// bart will now be a named tuple with first_name="Bart",// last_name="Simpson", age=10constautobart=rfl::to_named_tuple(lisa).transform(to_bart);
.and_then(f) expects a function f of type Field -> NamedTuple.
and_then then applies that function to each field of the named tuple
and finally concatenates the resulting named tuple to form a new named tuple.
Note that the named tuple returned by f may be empty. .and_then(f) can be used
to change either the values or the names of the fields, and can also affect
their overall number.
constautolisa=Person{.first_name="Lisa",.last_name="Simpson",.age=8};constautoto_bart=[](autofield){ifconstexpr(decltype(field)::name()=="first_name"){field="Bart";returnrfl::make_named_tuple(field);}elseifconstexpr(decltype(field)::name()=="age"){returnrfl::make_named_tuple();}else{returnrfl::make_named_tuple(field);}};// bart will now be a named tuple with first_name="Bart",// last_name="Simpson". Since we have returned and empty// named tuple for the field "age", there will be no such// field in bart.constautobart=rfl::to_named_tuple(lisa).and_then(to_bart);
constautolisa=rfl::Field<"firstName",std::string>("Lisa")*rfl::Field<"lastName",std::string>("Simpson");// Returns a deep copy of the original object,// replacing first_name.constautomaggie=rfl::replace(lisa,rfl::make_field<"firstName">(std::string("Maggie")));// Also OKconstautobart=lisa.replace(rfl::make_field<"firstName">(std::string("Bart")));
Transforming structs to named tuples and vice versa¶
You can transform structs to named tuples and vice versa (this will only work with the rfl::Field-syntax:
autobart=Person{.first_name="Bart",.last_name="Simpson",.birthday="1987-04-19"};// bart_nt is a named tupleconstautobart_nt=rfl::to_named_tuple(bart);// You can also retrieve the equivalent named tuple// type to a struct:usingPersonNamedTuple=rfl::named_tuple_t<Person>;// rfl::to_named_tuple also supports move semanticsPersonNamedTuplebart_nt=rfl::to_named_tuple(std::move(bart_nt));// You can also go the other wayconstautobart_struct=rfl::from_named_tuple<Person>(bart_nt);constautobart_struct=rfl::from_named_tuple<Person>(std::move(bart_nt));