加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

c – 在不使用宏的情况下减少语法“噪声”

发布时间:2020-12-16 05:00:31 所属栏目:百科 来源:网络整理
导读:我试图找到一种方法来减少一些语法“噪音”而不诉诸宏.对于以下代码: struct base { base() = delete; }; struct tag1 final : private base{ static constexpr const char* name = "tag1";};template typename T std::string name() { return T::name; }//
我试图找到一种方法来减少一些语法“噪音”而不诉诸宏.对于以下代码:
struct base { base() = delete; };    
struct tag1 final : private base
{
    static constexpr const char* name = "tag1";
};
template <typename T> std::string name() { return T::name; }
// ...
int main()
{
   const std::string name1(name<tag1>());
   return 0;
}

摆脱一些静态的constexpr const char *(更不用说其他的)语法会很好,因为它很难重复tag2,tag3等等.另外,所有这一切的唯一部分是真的很有趣的是tag1,剩下的就是“噪音”.直接的解决方案是使用宏:

#define MAKE_TAG(tag_name) struct tag_name final : private base { 
    static constexpr const char* name = #tag_name; }    
MAKE_TAG(tag2);
// ...
const std::string name2(name<tag2>());

基于宏的MAKE_TAG(tag2);语法已经删除了所有的“噪音”,使tag2非常突出.宏的另一个好处是tag_name可以很容易地转换为字符串文字,以防止复制粘贴错误.

一个“明显的”可能的解决方案可能是通过name as a template argument

template<const char* name> base { ... };
struct tag3 final : private base<"tag3"> {};

但这是not supported by C++.从下面的answer开始,一个聪明的解决方法是使用可变参数模板:

template<char... S> struct base { base() = delete;
    static std::string name() { return{ S... }; } };    
struct tag4 final : base<'t','a','g','4'> { };
template <typename T> std::string name() { return T::name(); }

这确实减少了很多噪音,但需要写’t’,’a’,’g’,’4’而不是“tag4”.运行时解决方案相当简洁

struct base {
    const std::string name;
    base(const std::string& name) : name(name) {} };    
struct tag5 final : base { tag5() : base("tag5") {} };
template <typename T> std::string name() { return T().name; }

但这并不完全令人满意,因为tag5现在可以实例化,理想情况下没有意义.此外,现在需要编写tag5三次,这不是很多DRY.

有没有办法进一步简化(即减少打字)上面的代码? …没有使用宏?

解决方法

如果您愿意单独输入字符,我们可以执行以下操作:
template<char... S>
struct base { 
    base() = delete;
    static std::string name(){
        return {S...};
    }
};

struct tag1 final : private base<'t','1'>
{using base::name;};
struct tag2 final : private base<'t','2'>
{using base::name;};

Demo

这样称呼它:

std::cout << tag1::name() << std::endl;
std::cout << tag2::name() << std::endl;

我必须在派生类中使用base :: name添加,因??为您正在使用私有继承.如果继承变得受保护或公开,那么您不需要它.

基础中name()函数的要点是创建一个我们可以从中构造字符串的字符数组.我们使用variadic parameter pack expansion来创建数组.

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读