Std:: addressof как постоянное выражение в С++ 17

Спецификация std::addressof была изменена для С++ 17: теперь это разрешено быть постоянным выражением. Однако cppreference говорит, что:

Выражение std::addressof(E) является постоянным подвыражением, если E равно постоянное подвыражение lvalue.

  • Что такое постоянное подвыражение?
  • Что такое пример, когда std::addressof(E) будет постоянным выражением?
  • Что такое пример, где std::addressof(E) НЕ будет постоянным выражением?

Ответ 1

Это объясняется здесь.

Ввести следующее новое определение в существующий список в 17.3 [определения]: [Редакционная записка: если LWG 2234 принимается до этого вопрос, следует использовать принятую формулировку для нового определения вместо этого - редакционная заметка]

**constant subexpression** [defns.const.subexpr]

an expression whose evaluation as a subexpression of a *conditional-expression* *CE* (5.16 [expr.cond]) would not prevent *CE* from being a core constant expression (5.20 [expr.const]).

Итак, "постоянное подвыражение" примерно означает "вы можете использовать его в постоянном выражении".

Что такое пример, где std:: addressof (E) будет постоянным выражением?

Я полагаю, что он должен давать постоянное выражение всякий раз, когда &E (при условии, что & вызывает встроенный адрес-оператора).

constexpr int x  = 42; // static storage duration
constexpr int* p1 = &x; // x is an lvalue constant subexpression
constexpr int* p2 = std::addressof(x); // x is an lvalue constant subexpression

Что такое пример, где std:: addressof (E) НЕ будет постоянным выражением?

std::map<int, int> m;
void f() {
    int& r = m[42];
    constexpr int* z1 = &r; // error: r is not a constant subexpression
    constexpr int* z2 = std::addressof(r); // likewise

    constexpr int x = 43; // automatic storage duration
    constexpr const int y1 = *&x;                // ok; x is a constant subexpression
    constexpr const int y2 = *std::addressof(x); // likewise
    constexpr const int* p1 = &x;                // error: p1 points to an automatic object
    constexpr const int* p2 = std::addressof(x); // likewise

}