std::assignable_from

< cpp‎ | concepts
定义于头文件 <concepts>
template< class LHS, class RHS >

concept assignable_from =
  std::is_lvalue_reference_v<LHS> &&
  std::common_reference_with<
    const std::remove_reference_t<LHS>&,
    const std::remove_reference_t<RHS>&> &&
  requires(LHS lhs, RHS&& rhs) {
    { lhs = std::forward<RHS>(rhs) } -> std::same_as<LHS>;

  };
(C++20 起)

概念 assignable_from<LHS, RHS> 指定能赋值类型和值类别由 RHS 编码的表达式给类型为 LHS 所指定的左值表达式。

语义要求

给定

  • lhs ,指代对象 lcopy 的左值,使得 decltype((lhs))LHS
  • rhs ,使得 decltype((rhs))RHS 的表达式,
  • rcopy ,等于 rhs 的单独对象,

assignable_from<LHS, RHS> 仅若符合下列条件才得到实现

  • std::addressof(lhs = rhs) == std::addressof(lcopy) (即赋值表达式生成指代左操作数的左值);
  • 求值 lhs = rhs 后:
    • lhs 等于 rcopy ,除非 rhs 是指代 lcopy 的非 const 亡值(即赋值为自移动赋值),
    • rhs 是泛左值;
      • 若它是非 const 亡值,则其所指代的对象在合法但未指定的状态;
      • 否则,不修改其所指代的对象;

相等性保持

若表达式对给定的相等输入产生相等输出,则它保持相等性

  • 表达式的输入由其操作数组成。
  • 表达式的输出由其结果和表达式所修改的所有操作数(若存在)组成。

在标准概念的规范中,操作数定义为仅包含下列内容的最大子表达式:

每个操作数的 cv 限定与值类别,是通过假设每个模板类型形参代表一个 cv 无限定的非数组对象类型确定的。

进一步要求每个要求保持相等性的表达式都稳定:这种表达式带相同输入对象的二次求值必须拥有相等的输出,而无任何对这些输入对象的显式中间修改。

除非另外提醒,每个用于 requires 表达式中的表达式都要求保持相等性且稳定,而表达式的求值必须只修改其非常操作数。必须不修改常操作数。

注解

赋值不必为全函数。具体来说,若赋值给某对象 x 可能导致修改某其他对象 y ,则 x = y 可能不在 = 的定义域中。若左操作数直接或间接占有右操作数(例如用指向基于结点的数据结构中的结点的智能指针,或用类似 std::vector<std::any> 的构造),则这常发生。

参阅

检查类型是否拥有针对特定实参的赋值运算符
(类模板)