C++ require 没有隐式转换的函数

发布于 2025-01-12 06:30:06 字数 1219 浏览 1 评论 0原文

我使用 boost::variant 来模仿具有值语义的继承。

有一个类可以被打印:

struct Printable { /* ... */ };

void print(const Printable &) { /* ... */ }

和类可能不会:

struct NotPrintable { /* ... */ };

最后,有一个带有隐式强制转换的“Base”类:

struct Base : boost::variant<Printable, NotPrintable>
{
    Base(const Printable &) {}    // constructor for implicit cast
    Base(const NotPrintable &) {} // constructor for implicit cast
};

// Print, if printable, throw exception, if not
void print(const Base &base) 
{ 
    Printer printer{};
    base.apply_visitor(printer);
}

问题是如何检查访问者内部是否可打印:

struct Printer
{
   using result_type = void;

   // If printable 
   template<typename PrintableType> requires 
   requires(const PrintableType &v) { {print(v)}; }  // (1)
   void operator()(const PrintableType &v) { print(v); }

   // If not printable
   void operator()(const auto &v) { throw /*...*/; }
}; 

要求 由于隐式转换为 const Base &,(1) 始终为 true。如何避免只在那个确切的地方进行转换?

I'm using boost::variant to imitate inheritance with value semantics.

There is one class that may be printed:

struct Printable { /* ... */ };

void print(const Printable &) { /* ... */ }

And class that may not:

struct NotPrintable { /* ... */ };

Finally, there is "Base" class with implicit cast:

struct Base : boost::variant<Printable, NotPrintable>
{
    Base(const Printable &) {}    // constructor for implicit cast
    Base(const NotPrintable &) {} // constructor for implicit cast
};

// Print, if printable, throw exception, if not
void print(const Base &base) 
{ 
    Printer printer{};
    base.apply_visitor(printer);
}

The problem is how to check for printable inside of visitor:

struct Printer
{
   using result_type = void;

   // If printable 
   template<typename PrintableType> requires 
   requires(const PrintableType &v) { {print(v)}; }  // (1)
   void operator()(const PrintableType &v) { print(v); }

   // If not printable
   void operator()(const auto &v) { throw /*...*/; }
}; 

Requirement (1) is always true due-to implicit conversion to const Base &. How to avoid conversion only in that exact place?

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(2

晚雾 2025-01-19 06:30:06

正如@Jarod42在评论中所说,为了保证不发生隐式转换,需要定义一个模板print()函数来“吸收”其他类型(NotPrintable 在您的示例中)并将其设置为删除

void print(const auto&) = delete;

然后可以将 printable 概念定义为

template<class T>
concept printable = requires (const T& x) { print(x); };

要求表达式 print(x) 格式良好。

T 的类型为 PrintableBase 时,表达式 print(x) 有效,因为您已定义相应的 print() 函数。当T的类型为NotPrintable或其他类型时,将调用删除的print(),这使得表达式格式错误,因此约束条件不满足。

然后你可以使用这个概念来约束 Printer::operator()

struct Printer {
  using result_type = void;

  // If printable 
  template<printable PrintableType>
  void operator()(const PrintableType& v) { print(v); }

  // If not printable
  void operator()(const auto&) { throw /*...*/; }
};

请注意,由于 print(const auto&) 可以实例化为任何类型,这将禁止所有隐式转换,但我们仍然可以通过添加约束来允许某些隐式转换

#include <concepts>

struct Boolean { };
struct Integer {
  Integer();
  Integer(Boolean);
  friend Integer operator+(Integer, Integer);
};
struct String { };

template<class T, class U>
  requires (!std::convertible_to<U, T>)
void operator+(T, U) = delete;

int main() {
  Integer{} + Integer{}; // OK
  Integer{} + String{};  // ERROR (as expected)
  Integer{} + Boolean{}; // OK
};

As @Jarod42 said in the comments, in order to ensure that implicit conversion does not occur, you need to define a template print() function to "absorb" other types (NotPrintable in your example) and set it to delete

void print(const auto&) = delete;

Then the printable concept can be defined as

template<class T>
concept printable = requires (const T& x) { print(x); };

which requires the expression print(x) to be well-formed.

When the type of T is Printable or Base, the expression print(x) is valid since you have defined the corresponding print() function for them. When the type of T is NotPrintable or other types, the deleted print() will be invoked which makes the expression ill-formed, so that the constraint is not satisfied.

Then you can use this concept to constrain Printer::operator()

struct Printer {
  using result_type = void;

  // If printable 
  template<printable PrintableType>
  void operator()(const PrintableType& v) { print(v); }

  // If not printable
  void operator()(const auto&) { throw /*...*/; }
};

Note that since print(const auto&) can be instantiated to any type, this will prohibit all implicit conversions, but we can still allow some implicit conversions by adding constraints to it

#include <concepts>

struct Boolean { };
struct Integer {
  Integer();
  Integer(Boolean);
  friend Integer operator+(Integer, Integer);
};
struct String { };

template<class T, class U>
  requires (!std::convertible_to<U, T>)
void operator+(T, U) = delete;

int main() {
  Integer{} + Integer{}; // OK
  Integer{} + String{};  // ERROR (as expected)
  Integer{} + Boolean{}; // OK
};
落在眉间の轻吻 2025-01-19 06:30:06

我找到了一个基于内联好友定义的完美解决方案

根据标准:

这样的函数隐式是内联函数 (10.1.6)。 类中定义的友元函数位于(词法)中
定义它的类的范围。
在类外部定义的友元函数不是 (6.4.1)。


其他一些可能有用的东西:

  1. 删除的模板功能

    void print(auto) = 删除;
    

缺点:

  • 每个函数都有样板

  • 禁止所有隐式转换

    结构布尔值;
    
    结构字符串;
    
    结构体整数
    {
      整数(布尔值);
    
      朋友 整数运算符+(整数, 整数);
    }
    
    模板
    需要 (!std::convertible_to)
    void 运算符+(T, U) = 删除;
    
    整数 + 整数 // 确定
    整数 + 字符串 // 错误(如预期)
    整数 + 布尔值 // 确定
    
  1. 更改接口

    结构基础:/* ... */
    {
      /* ... */
    
      静态无效打印(基础);
    }
    

缺点:

  • 没有运算符支持 (Base + Base -> Base::add(Base, Base))
  • 接口有点差
  1. 添加特征
    模板<类型名称 T>结构特征{
      静态 constexpr bool is_printable = true;
    }
    

缺点:

  • 每个类和方法的样板
  • 如何处理隐式转换(Boolean + Integer)?
  • 因延期而关闭

I found a perfect solution that is based on Inline friend definition.

According to standard:

Such a function is implicitly an inline function (10.1.6). A friend function defined in a class is in the (lexical)
scope of the class in which it is defined.
A friend function defined outside the class is not (6.4.1).


Some other things that may work:

  1. Deleted template function

    void print(auto) = delete;
    

cons:

  • Boilerplate for every function

  • Forbids all implicit conversions

    struct Boolean;
    
    struct String;
    
    struct Integer
    {
      Integer(Boolean);
    
      friend Integer operator+(Integer, Integer);
    }
    
    template<class T, class U>
    requires (!std::convertible_to<U, T>)
    void operator+(T, U) = delete;
    
    Integer + Integer // OK
    Integer + String  // ERROR (as expected)
    Integer + Boolean // OK
    
  1. Change interface

    struct Base : /* ... */
    {
      /* ... */
    
      static void print(Base);
    }
    

cons:

  • No operators support (Base + Base -> Base::add(Base, Base))
  • Interface a little bit worse
  1. Add traits
    template<typename T> struct Traits {
      static constexpr bool is_printable = true;
    }
    

cons:

  • Boilerplate for every class and method
  • How to handle implicit conversion (Boolean + Integer)?
  • Closed for extension
~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文