-
Notifications
You must be signed in to change notification settings - Fork 71
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Introducing variable_t
and similar types for convenience
#378
Comments
@jhcarl0814 |
@HDembinski Sorry I have to mention https://www.hyrumslaw.com/ again. We need to know how many users are relying on e.g. Some old code that template<typename T = int>
struct variable
{
std::conditional_t<std::is_same_v<T, void>, int, T> data;
};
template<typename T>
void f1(variable<T>)
{
}
void f2()
{
f1(variable<int>());
} When replace template<typename T = int>
struct variable_t
{
T data;
};
template<typename T = int>
using variable = variable_t<std::conditional_t<std::is_same_v<T, void>, int, T>>;
template<typename T>
void f1(variable<T>) // Candidate template ignored: couldn't infer template argument 'T'
{
}
void f2()
{
f1(variable<int>()); // No matching function cal to 'f1'
} So require every user to change code to: template<typename T = int>
struct variable_t
{
T data;
};
template<typename T = int>
using variable = variable_t<std::conditional_t<std::is_same_v<T, void>, int, T>>;
template<typename T>
void f1(variable_t<T>) ///////////////////////////////////////
{
}
void f2()
{
f1(variable<int>());
} All other non-generic code (e.g. |
@HDembinski The type alias should work with template deduction guides, it's just not working now.
template<
/*only `variable_t`'s definition's template parameters that appear in the deduction result*/,
/*the constructor's template parameters that are not deduced from template arguments given in `variable_t`'s definition*/>
variable(/*the constructor's function parameters (the constructor's template parameters that are deduced from template arguments given in `variable_t`'s definition are replaced with the deduction result)*/)
->
variable<
std::conditional_t<std::is_same_v<Value, use_default>, double, Value>,
std::conditional_t<std::is_same_v<MetaData, use_default>, std::string, MetaData>,
std::conditional_t<std::is_same_v<Options, use_default>, decltype(option::underflow | option::overflow), Options>,
std::conditional_t<std::is_same_v<Allocator, use_default>, std::allocator<std::conditional_t< std::is_same_v<Value, use_default>, double, Value>>, Allocator>>;
// e.g.
template</*variable_t's*/class Value, /*variable_t's*/class MetaData, /*variable_t's*/class Options, /*variable_t's*/class Allocator, /*variable's*/class U>
variable(
std::initializer_list<U>,
typename boost::histogram::axis::metadata_base<
std::conditional_t<
std::is_same_v<std::conditional_t<std::is_same_v<MetaData, use_default>, std::string, MetaData>, boost::use_default>,
std::string,
std::conditional_t<std::is_same_v<MetaData, use_default>, std::string, MetaData>>,
(std::is_empty_v<std::conditional_t</* ... */>> && std::is_final_v<std::conditional_t</* ... */>>)
>::metadata_type,
std::conditional_t<
std::is_same_v<std::conditional_t<std::is_same_v<Options, use_default>, decltype(option::underflow | option::overflow), Options>, boost::use_default>,
boost::histogram::axis::option::bitset<3>,
std::conditional_t<std::is_same_v<Options, use_default>, decltype(option::underflow | option::overflow), Options>>,
std::conditional_t<std::is_same_v<Allocator, use_default>, std::allocator<std::conditional_t< std::is_same_v<Value, use_default>, double, Value>>, Allocator>>
)->
variable<
std::conditional_t<std::is_same_v<Value, use_default>, double, Value>,
std::conditional_t<std::is_same_v<MetaData, use_default>, std::string, MetaData>,
std::conditional_t<std::is_same_v<Options, use_default>, decltype(option::underflow | option::overflow), Options>,
std::conditional_t<std::is_same_v<Allocator, use_default>, std::allocator<std::conditional_t< std::is_same_v<Value, use_default>, double, Value>>, Allocator>
>;
// U is in a deduced context and is deduced to float/double/long double etc,
// Value is in a non-deduced context and uses default template argument (double) in the end
template<
/*only `variable_t`'s definition's template parameters that appear in the deduction result*/,
/*the deduction guide's template parameters that are not deduced from template arguments given in `variable_t`'s definition*/>
variable(/*the deduction guide's function parameters (the deduction guide's template parameters that are deduced from template arguments given in `variable_t`'s definition are replaced with the deduction result)*/)
->
variable</*the deduction guide's template arguments (the deduction guide's template parameters that are deduced from template arguments given in `variable_t`'s definition are replaced with the deduction result)*/>;
// e.g.
template </* none of variable_t's template parameters appear in the deduction result */ /*variable's*/class T>
variable(std::initializer_list<T>) -> variable<detail::convert_integer<T, double>, null_type>;
// T is in a deduced context and is deduced to float/double/long double etc,
// so we get variable<float/double/long double, null_type> #include <boost/histogram/axis.hpp>
#include <boost/type_index.hpp>
#include<iostream>
namespace boost {
namespace histogram {
namespace axis {
template<class Value___ = double, class MetaData___ = use_default, class Options___ = use_default, class Allocator___ = std::allocator<Value___>>
using variable_t = boost::histogram::axis::variable<
detail::replace_default<Value___, double>,
detail::replace_default<MetaData___, std::string>,
detail::replace_default<Options___, decltype(option::underflow |
option::overflow)>, detail::replace_default<Allocator___,
std::allocator<detail::replace_default<Value___, double>>>>;
} // namespace axis
} // namespace histogram
} // namespace boost
static_assert(__cpp_deduction_guides >= 201907L);
int main()
{
{
boost::histogram::axis::variable<> a1; std::cout<<boost::typeindex::type_id_with_cvr<decltype(a1)>().pretty_name()<<'\n';
boost::histogram::axis::variable_t<> b1; std::cout<<boost::typeindex::type_id_with_cvr<decltype(b1)>().pretty_name()<<'\n'; std::cout<<'\n';
boost::histogram::axis::variable a2; std::cout<<boost::typeindex::type_id_with_cvr<decltype(a2)>().pretty_name()<<'\n';
boost::histogram::axis::variable_t b2; std::cout<<boost::typeindex::type_id_with_cvr<decltype(b2)>().pretty_name()<<'\n'; std::cout<<'\n';
boost::histogram::axis::variable a3({1,2,3}); std::cout<<boost::typeindex::type_id_with_cvr<decltype(a3)>().pretty_name()<<'\n';
boost::histogram::axis::variable_t b3({1,2,3}); std::cout<<boost::typeindex::type_id_with_cvr<decltype(b3)>().pretty_name()<<'\n'; std::cout<<'\n';
boost::histogram::axis::variable a4({1.0f,2.0f,3.0f}); std::cout<<boost::typeindex::type_id_with_cvr<decltype(a4)>().pretty_name()<<'\n';
boost::histogram::axis::variable_t b4({1.0f,2.0f,3.0f}); std::cout<<boost::typeindex::type_id_with_cvr<decltype(b4)>().pretty_name()<<'\n'; std::cout<<'\n';
boost::histogram::axis::variable a5({1.0,2.0,3.0}); std::cout<<boost::typeindex::type_id_with_cvr<decltype(a5)>().pretty_name()<<'\n';
boost::histogram::axis::variable_t b5({1.0,2.0,3.0}); std::cout<<boost::typeindex::type_id_with_cvr<decltype(b5)>().pretty_name()<<'\n'; std::cout<<'\n';
boost::histogram::axis::variable a6({1.0l,2.0l,3.0l}); std::cout<<boost::typeindex::type_id_with_cvr<decltype(a6)>().pretty_name()<<'\n';
boost::histogram::axis::variable_t b6({1.0l,2.0l,3.0l}); std::cout<<boost::typeindex::type_id_with_cvr<decltype(b6)>().pretty_name()<<'\n'; std::cout<<'\n';
}
} PS: |
I am well aware of Hyrum's law, that's why I stated my preference so carefully. Okay, so replacing the struct name with an alias name does not work. :( Do you think it is still useful to introduce variable_t in addition to variable? And likewise for every axis? |
That seems to be an issue with gcc only, the other compilers use the deduction guides. I implemented some workarounds just recently here #373 to fix gcc>=11. |
In my opinion, the current version is better than the boost::histogram::axis::variable<double, boost::use_default, boost::use_default, std::allocator<double> >
boost::histogram::axis::variable<double, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, boost::histogram::axis::option::bitset<3u>, std::allocator<double> > You should not make type checks like this anyway (what is the use case for such type checks)? std::is_same_v<
boost::histogram::axis::variable<double, std::string, boost::use_default>,
boost::histogram::axis::variable<double, std::string, decltype(boost::histogram::axis::option::underflow | boost::histogram::axis::option::overflow)>
> If you want to know what the options are, call |
The main point you have in favor is:
boost::histogram::axis::variable<double, std::string, boost::use_default> a;
boost::histogram::axis::variable<double, std::string, decltype(boost::histogram::axis::option::underflow | boost::histogram::axis::option::overflow)> b;
a = b;
boost::histogram::axis::variable_t<double, std::string, boost::use_default> a;
boost::histogram::axis::variable_t<double, std::string, decltype(boost::histogram::axis::option::underflow | boost::histogram::axis::option::overflow)> b;
a = b; This could be fixed with a converting ctor. |
@HDembinski You are right. It's not transparent to switch to the type alias (and I forgot there's no CTAD for template type alias during [C++17, C++20)). So I'd better only use it in my own projects when need it. The tricky part is, there are already so many places scattered around various libraries (e.g. standard library) that uses the "same type" type trait/concept that:
searching "same type" can give us an impression of how many places depend on the "same type" type trait/concept in standard library (maybe the doc should add a note to warn users so they will not shoot themselves in the foot?):
|
Thanks for listing all that. I agree with you in general, but most of these caveats do not seem to matter for Boost.Histogram. In practice, it does not seem to be an issue.
etc. That being said, I am not against supporting |
is
false
,is
true
;This causes a compile error:
This does not cause a compile error:
In templates, you can get the meta type, options type, allocator type conveniently if template users write
variable_t
, not if template users writevariable
:In debugger, you can not see what options
a
has but can see what optionsb
has:boost::histogram::axis::variable<> a; boost::histogram::axis::variable_t<> b;
Originally posted by @jhcarl0814 in #372 (comment)
The text was updated successfully, but these errors were encountered: