Почему неконстантная ссылка не может привязываться к временному объекту?
Почему нельзя получить неконстантную ссылку на временный объект, функция которого getx()
возвращается? Понятно, что это запрещено C++ Standard, но меня интересует цель такого ограничения, а не ссылка на стандарт.
struct X
{
X& ref() { return *this; }
};
X getx() { return X();}
void g(X & x) {}
int f()
{
const X& x = getx(); // OK
X& x = getx(); // error
X& x = getx().ref(); // OK
g(getx()); //error
g(getx().ref()); //OK
return 0;
}
- Понятно, что время жизни объекта не может быть причиной, потому что постоянная ссылка на объект не запрещена стандартом C++.
- Ясно, что временный объект не является константой в приведенном выше примере, потому что вызовы непостоянных функций разрешены. Например,
ref()
может изменить временный объект. - К тому же,
ref()
позволяет обмануть компилятор и получить ссылку на этот временный объект, и это решает нашу проблему.
К тому же:
Они говорят: "Присвоение временного объекта константной ссылке продлевает время жизни этого объекта" и "Ничего не сказано о неконстантных ссылках, хотя". Мой дополнительный вопрос. Продлевает ли следующее назначение время жизни временного объекта?
X& x = getx().ref(); // OK
11 ответов
Из этой статьи блога Visual C++ о ссылках rvalue:
... C++ не хочет, чтобы вы случайно изменяли временные значения, но прямой вызов неконстантной функции-члена для модифицируемого значения r является явным, поэтому это разрешено...
По сути, вы не должны пытаться модифицировать временные объекты по той самой причине, что они являются временными объектами и теперь умирают в любой момент. Причина, по которой вам разрешается вызывать неконстантные методы, заключается в том, что вы можете делать некоторые "глупые" вещи, если вы знаете, что делаете, и вы явно об этом (например, используете reinterpret_cast). Но если вы связываете временную ссылку с неконстантной ссылкой, вы можете продолжать передавать ее "навсегда", просто чтобы ваши манипуляции с объектом исчезли, потому что где-то по пути вы полностью забыли, что это было временным.
На вашем месте я бы переосмыслил дизайн своих функций. Почему g() принимает ссылку, изменяет ли она параметр? Если нет, сделайте это константной ссылкой, если да, почему вы пытаетесь передать ему временное значение, не волнует ли это, что вы изменяете временное значение? Почему getx() все равно возвращается временно? Если вы поделитесь с нами своим реальным сценарием и тем, что вы пытаетесь достичь, вы можете получить несколько хороших советов о том, как это сделать.
Идти против языка и дурачить компилятор редко решает проблемы - обычно это создает проблемы.
Изменить: Решение вопросов в комментарии: 1)
X& x = getx().ref(); // OK when will x die?
- Я не знаю, и мне все равно, потому что это именно то, что я имею в виду, говоря "против языка". Язык говорит: "Временные символы умирают в конце утверждения, если они не связаны с константной ссылкой, и в этом случае они умирают, когда ссылка выходит из области видимости". Применяя это правило, кажется, что x уже мертв в начале следующего оператора, поскольку он не связан с константной ссылкой (компилятор не знает, что возвращает ref()). Это только предположение однако.2) Я четко сформулировал цель: вам не разрешено изменять временные файлы, потому что это просто не имеет смысла (игнорирование ссылок на значения C++0x). Вопрос "тогда почему мне разрешено звонить неконстантным участникам?" хороший, но у меня нет лучшего ответа, чем тот, который я уже сказал выше.
3) Ну, если я прав насчет х в X& x = getx().ref();
Умирая в конце заявления, проблемы очевидны.
В любом случае, основываясь на вашем вопросе и комментариях, я не думаю, что даже эти дополнительные ответы вас удовлетворит. Вот последняя попытка / краткое изложение: Комитет C++ решил, что не имеет смысла изменять временные данные, поэтому они запретили привязку к неконстантным ссылкам. Может быть, некоторые реализации компилятора или исторические проблемы также были вовлечены, я не знаю. Затем возник какой-то конкретный случай, и было решено, что, несмотря ни на что, они по-прежнему будут позволять прямое изменение посредством вызова неконстантного метода. Но это исключение - вы, как правило, не можете изменять временные данные. Да, C++ часто такой странный.
В вашем коде getx()
возвращает временный объект, так называемое "значение". Вы можете скопировать значения r в объекты (так называемые переменные) или связать их с константными ссылками (что продлит их время жизни до конца жизни ссылки). Вы не можете привязать значения к неконстантным ссылкам.
Это было преднамеренное проектное решение, чтобы предотвратить случайное изменение пользователями объекта, который погибнет в конце выражения:
g(getx()); // g() would modify an object without anyone being able to observe
Если вы хотите сделать это, вам сначала нужно будет сделать локальную копию или объект или связать его с константной ссылкой:
X x1 = getx();
const X& x2 = getx(); // extend lifetime of temporary to lifetime of const reference
g(x1); // fine
g(x2); // can't bind a const reference to a non-const reference
Обратите внимание, что следующий стандарт C++ будет включать ссылки на rvalue. Поэтому то, что вы знаете как ссылки, становится так называемым "ссылками lvalue". Вам будет разрешено связывать rvalue со ссылками на rvalue, и вы можете перегрузить функции на "rvalue-ness":
void g(X&); // #1, takes an ordinary (lvalue) reference
void g(X&&); // #2, takes an rvalue reference
X x;
g(x); // calls #1
g(getx()); // calls #2
g(X()); // calls #2, too
Идея, стоящая за ссылками на rvalue, заключается в том, что, поскольку эти объекты все равно умрут, вы можете воспользоваться этими знаниями и реализовать так называемую "семантику перемещения", определенный вид оптимизации:
class X {
X(X&& rhs)
: pimpl( rhs.pimpl ) // steal rhs' data...
{
rhs.pimpl = NULL; // ...and leave it empty, but deconstructible
}
data* pimpl; // you would use a smart ptr, of course
};
X x(getx()); // x will steal the rvalue's data, leaving the temporary object empty
То, что вы показываете, это то, что оператор цепочки разрешен
X& x = getx().ref(); // OK
Выражение 'getx(). Ref();' и это выполняется до завершения перед присваиванием 'x'.
Обратите внимание, что getx () не возвращает ссылку, но полностью сформированный объект в локальный контекст. Объект является временным, но он не является константным, что позволяет вам вызывать другие методы для вычисления значения или возникновения других побочных эффектов.
// It would allow things like this.
getPipeline().procInstr(1).procInstr(2).procInstr(3);
// or more commonly
std::cout << getManiplator() << 5;
Посмотрите на конец этого ответа для лучшего примера этого
Вы не можете привязать временную ссылку к ссылке, потому что при этом будет сгенерирована ссылка на объект, который будет уничтожен в конце выражения, что оставит вас с висячей ссылкой (что неопрятно, а стандарт не любит неопрятность).
Значение, возвращаемое ref(), является допустимой ссылкой, но метод не обращает никакого внимания на срок службы возвращаемого объекта (поскольку он не может иметь эту информацию в своем контексте). Вы в основном только что сделали эквивалент:
x& = const_cast<x&>(getX());
Причина, по которой это можно сделать с помощью константной ссылки на временный объект, состоит в том, что стандарт продлевает срок службы временного элемента до срока службы ссылки, поэтому срок службы временных объектов продлевается после окончания оператора.
Таким образом, единственный оставшийся вопрос заключается в том, почему стандарт не хочет разрешать использование временных ссылок для продления срока службы объекта до конца утверждения?
Я полагаю, что это потому, что из-за этого компилятору будет очень трудно получить правильные значения для временных объектов. Это было сделано для постоянных ссылок на временные файлы, так как это ограничивало использование и, таким образом, заставляло вас делать копию объекта, чтобы делать что-нибудь полезное, но предоставляет некоторые ограниченные функциональные возможности.
Подумайте об этой ситуации:
int getI() { return 5;}
int x& = getI();
x++; // Note x is an alias to a variable. What variable are you updating.
Продление срока службы этого временного объекта будет очень запутанным.
Пока следующее:
int const& y = getI();
Даст вам код, который будет интуитивно понятен в использовании и понимании.
Если вы хотите изменить значение, вы должны возвращать значение в переменную. Если вы пытаетесь избежать затрат на копирование объекта обратно из функции (так как кажется, что объект является копией, созданной обратно (технически это так)). Тогда не беспокойтесь, компилятор очень хорош в "Оптимизации возвращаемого значения"
Почему это обсуждается в C++ FAQ (жирный шрифт):
В C++ неконстантные ссылки могут связываться с lvalues, а const ссылки могут связываться с lvalue или rvalue, но нет ничего, что могло бы связываться с неконстантным rvalue. Это защищает людей от изменения временных ценностей, которые уничтожаются до того, как их новое значение может быть использовано. Например:
void incr(int& a) { ++a; }
int i = 0;
incr(i); // i becomes 1
incr(0); // error: 0 is not an lvalue
Если бы это incr(0) было разрешено, либо какое-то временное значение, которое никто никогда не видел, было бы увеличено, или, что гораздо хуже, значение 0 стало бы 1. Последнее звучит глупо, но на самом деле была такая ошибка в ранних компиляторах Фортрана, которые устанавливали в сторону ячейки памяти для хранения значения 0.
Похоже, на первоначальный вопрос о том, почему это не разрешено, дан четкий ответ: "потому что это, скорее всего, ошибка".
FWIW, я думал, что покажу, как это можно сделать, хотя я не думаю, что это хорошая техника.
Причина, по которой я иногда хочу передать временное значение методу, принимающему неконстантную ссылку, заключается в том, что он намеренно отбрасывает значение, возвращаемое по ссылке, о которой вызывающий метод не заботится. Что-то вроде этого:
// Assuming: void Person::GetNameAndAddr(std::string &name, std::string &addr);
string name;
person.GetNameAndAddr(name, string()); // don't care about addr
Как объяснялось в предыдущих ответах, это не компилируется. Но это компилируется и работает правильно (с моим компилятором):
person.GetNameAndAddr(name,
const_cast<string &>(static_cast<const string &>(string())));
Это просто показывает, что вы можете использовать приведение, чтобы лгать компилятору. Очевидно, было бы намного понятнее объявить и передать неиспользованную автоматическую переменную:
string name;
string unused;
person.GetNameAndAddr(name, unused); // don't care about addr
Этот метод вводит ненужную локальную переменную в область действия метода. Если по какой-то причине вы хотите предотвратить его использование в дальнейшем в методе, например, чтобы избежать путаницы или ошибки, вы можете скрыть его в локальном блоке:
string name;
{
string unused;
person.GetNameAndAddr(name, unused); // don't care about addr
}
Крис
Основная проблема заключается в том, что
g(getx()); //error
это логическая ошибка: g
модифицирует результат getx()
но у вас нет шансов осмотреть измененный объект. Если g
не нужно было изменять его параметр, тогда не потребовалась бы ссылка на lvalue, он мог бы взять параметр по значению или по константной ссылке.
const X& x = getx(); // OK
допустимо, потому что иногда вам нужно повторно использовать результат выражения, и совершенно ясно, что вы имеете дело с временным объектом.
Однако это не возможно сделать
X& x = getx(); // error
действителен без принятия g(getx())
действительный, чего и пытались избежать дизайнеры языка в первую очередь.
g(getx().ref()); //OK
допустимо, потому что методы знают только о постоянстве this
они не знают, вызваны ли они на lvalue или на rvalue.
Как всегда в C++, у вас есть обходной путь для этого правила, но вы должны сообщить компилятору, что вы знаете, что делаете, будучи явным:
g(const_cast<x&>(getX()));
Зачем тебе вообще X& x = getx();
? Просто используйте X x = getx();
и полагаться на RVO.
Злой обходной путь включает ключевое слово "изменяемый". На самом деле, быть злом - упражнение для читателя. Или смотрите здесь: http://www.ddj.com/cpp/184403758
Отличный вопрос, и вот моя попытка получить более краткий ответ (так как много полезной информации есть в комментариях и трудно разобраться в шуме.)
Любая ссылка, связанная непосредственно с временным, продлит его жизнь [12.2.5]. С другой стороны, ссылка, инициализированная с другой ссылкой, не будет (даже если в конечном итоге она будет такой же временной). Это имеет смысл (компилятор не знает, на что в конечном счете ссылается эта ссылка).
Но вся эта идея чрезвычайно запутана. Например const X &x = X();
сделает временным до тех пор, пока x
ссылка, но const X &x = X().ref();
не будет (кто что знает ref()
на самом деле вернулся). В последнем случае деструктор для X
вызывается в конце этой строки. (Это можно наблюдать с помощью нетривиального деструктора.)
Таким образом, это кажется в целом сбивающим с толку и опасным (зачем усложнять правила относительно времени жизни объекта?), Но, по-видимому, была необходимость, по крайней мере, для ссылок на const, поэтому стандарт устанавливает для них такое поведение.
[Из комментария sbi ]: обратите внимание, что тот факт, что привязка его к константной ссылке увеличивает время жизни временного объекта, является исключением, которое было намеренно добавлено (TTBOMK для обеспечения ручной оптимизации). Не было добавлено исключение для неконстантных ссылок, потому что привязка временного к неконстантной ссылке, скорее всего, была ошибкой программиста.
Все временные сохраняются до конца полного выражения. Однако, чтобы использовать их, вам нужен трюк, как у вас с ref()
, Это законно. Кажется, нет веской причины для перехода через лишний обруч, кроме как для напоминания программисту о том, что происходит что-то необычное (а именно, эталонный параметр, модификации которого будут быстро потеряны).
[Другой комментарий sbi ] Причина, по которой Страуструп дает (в D&E) запрет на привязку значений r к неконстантным ссылкам, заключается в том, что, если g() Алексея изменит объект (что можно ожидать от функции, принимающей неконстантное значение) ссылка), он изменил бы объект, который собирается умереть, так что никто не мог получить в любом случае измененное значение. Он говорит, что это, скорее всего, ошибка.
"Понятно, что временный объект не является константой в приведенном выше примере, потому что разрешены вызовы непостоянных функций. Например, ref() может изменить временный объект".
В вашем примере getX() не возвращает const X, поэтому вы можете вызывать ref () почти так же, как вы можете вызывать X(). Ref(). Вы возвращаете неконстантную ссылку и поэтому можете вызывать неконстантные методы, но вы не можете назначить ссылку на неконстантную ссылку.
Наряду с комментарием SadSidos это делает ваши три пункта неверными.
У меня есть сценарий, которым я хотел бы поделиться, где я хотел бы сделать то, что просит Алексей. В плагине Maya C++ мне нужно выполнить следующий шенаниган, чтобы получить значение в атрибуте узла:
MFnDoubleArrayData myArrayData;
MObject myArrayObj = myArrayData.create(myArray);
MPlug myPlug = myNode.findPlug(attributeName);
myPlug.setValue(myArrayObj);
Это утомительно писать, поэтому я написал следующие вспомогательные функции:
MPlug operator | (MFnDependencyNode& node, MObject& attribute){
MStatus status;
MPlug returnValue = node.findPlug(attribute, &status);
return returnValue;
}
void operator << (MPlug& plug, MDoubleArray& doubleArray){
MStatus status;
MFnDoubleArrayData doubleArrayData;
MObject doubleArrayObject = doubleArrayData.create(doubleArray, &status);
status = plug.setValue(doubleArrayObject);
}
И теперь я могу написать код с начала поста как:
(myNode | attributeName) << myArray;
Проблема в том, что он не компилируется вне Visual C++, потому что он пытается связать временную переменную, возвращаемую из | оператор на ссылку MPlug оператора <<. Я хотел бы, чтобы это было ссылкой, потому что этот код вызывается много раз, и я бы предпочел, чтобы MPlug не копировался так много. Мне нужен только временный объект, чтобы жить до конца второй функции.
Ну, это мой сценарий. Просто подумал, что покажу пример, где хочется сделать то, что описывает Алексей. Я приветствую все критические замечания и предложения!
Благодарю.