C++11
提供user-defined literals。我刚开始玩它们,这让我想知道是否可以自动将所有SI multipliers添加到我定义的单个文字中?
例如,如果我定义
Length operator "" _m(long double m) {
return Length(m); // Length in meters
}
其中Length
是某个Units
基类的子类,我希望有一个自动添加的机制(与boost operators一样的精神)返回Length
的所有文字的SI乘数:
// these are added automatically when defining the literal "_m":
// Length in:
Length operator "" _Ym(long double Ym); // Yottameters
Length operator "" _Zm(long double Zm); // Zetameters
... // ...
... // ...
Length operator "" _km(long double km); // kilometers
Length operator "" _mm(long double mm); // millimeters
... // ...
... // ...
Length operator "" _zm(long double zm); // zeptometers
Length operator "" _ym(long double ym); // yoctometers
据我所知,除了一些宏魔术之外,没有办法自动执行此操作,因为所有用户定义的文字都需要显式定义。
..还是我忽略了什么?
答案 0 :(得分:3)
如果没有“怪异的宏”,我认为没有办法完全按照你的要求去做。这是我能得到的:
template<typename T, T (*op)(long double)>
struct SI
{
// ...
constexpr static T micro = op (.000001);
constexpr static T milli = op (.001);
constexpr static T kilo = op (1000);
constexpr static T mega = op (1000000);
// ...
};
struct Length
{
constexpr Length(long double d) : _d(d) { }
constexpr operator long double() { return _d; }
long double _d;
};
constexpr Length operator "" _m(long double m) {
return Length(m);
}
typedef SI<Length, ::operator "" _m> SI_Length;
int main()
{
constexpr Length l = 3 * SI_Length::kilo;
static_assert(l == 3000, "error");
}
如果允许使用奇怪的宏,那么下面的内容就可以完成这项工作:
#define DEFINE_SI_MULTIPLIERS(T, unit) \
constexpr T operator "" _u ## unit(long double m) \
{ return ::operator "" _ ## unit(0.000001 * m); } \
constexpr T operator "" _m ## unit(long double m) \
{ return ::operator "" _ ## unit(0.001 * m); } \
constexpr T operator "" _k ## unit(long double m) \
{ return ::operator "" _ ## unit(1000 * m); } \
// ...
DEFINE_SI_MULTIPLIERS(Length, m)
int main()
{
constexpr Length l = 3.0_km;
static_assert(l == 3000, "error");
}
答案 1 :(得分:-1)
只要您愿意自己解析花车,就不能使用operator "" _m(const char *)
味道吗?这样就可以通过调用常见的SI感知解析器来编写1234k_m
来获取浮点值。