Difference between const auto & and auto & if object of reference is const

Difference between const auto & and auto & if object of reference is const


// case 1
const int i = 42;
const auto &k = i;

// case 2
const int i = 42;
auto &k = i;



Do we need the const keyword before auto in this scenario? After all, a reference (k) to an auto-deduced type will include the top level const of the object (const int i). So I believe k will be a reference to an integer that is constant (const int &k) in both cases.


const


auto


k


const


const


int i


k


const int &k



If that is true, does that mean that const auto &k = i; in case 1 is replaced by the compiler as just const int &k = i; (auto being replaced with int)? Whereas in case 2, auto is replaced with const int?


const auto &k = i;


const int &k = i;


auto


int


auto


const int






Possible duplicate of How to use auto with const and & in c++? (though this one has better title, just I would remove those ks).

– Daniel Langr
Sep 6 '18 at 12:34



k






Both are similar for compiler. But as human reader, I prefer the first one.

– Jarod42
Sep 6 '18 at 12:34






I wonder who found this "unclear". There's no lacking in clarity, you expressed yourself very well.

– StoryTeller
Sep 6 '18 at 12:35







@StoryTeller - A maintenance script identified it and posted it to the SOCVR room. I'm guessing it is a Type I error in the script. The question was misclassified and incorrectly rejected.

– jww
Sep 6 '18 at 13:41







@SteveCho - You can only address one person with a mention(@). jww didn't get pinged. As for your question, the SOCVR is a chat where users who care about content quality work together to remove bad content. There are bots who read new posts and bring it to the attention of the chat if they seem like bad questions. But it's still people who need to cast the close votes. jww and I were discussing the close votes on your post.

– StoryTeller
Sep 6 '18 at 15:01




5 Answers
5



auto keyword automatically decides the type of the variable at compile time.


auto



In your first case, auto is reduced to int where it's reduced to const int in the second case. So, both of your cases are reduced to the same code as:


auto


int


const int


const int &k = i;



However, it's better to have the const explicitly for better readability and to make sure your variable TRULY is const.


const






It also means "I want this to be a const reference, no matter whether it was already const before" because whether it was const before isn't really relevant to whether you want to add const now.

– Lightness Races in Orbit
Sep 6 '18 at 12:40



Type deduction with auto works like template argument type deduction with a few exceptions that don't apply in the given example. Hence


auto


const int i = 42;
const auto& k1 = i; // same as const int& k1 = i;
auto& k2 = i; // same as (const int)& k2 = i;



It is probably more readable to add the const qualifier nevertheless.


const



Here is another example where favoring brevity with auto is misleading:


auto


int *i;
const auto k1 = i; // same as int* const k1 = i;
const auto *k2 = i; // same as const int *k2 = i;



In the first case, the object that i points to can be modified through k1, in the second case, it can't.


i


k1






"with a few exceptions" Can you give an example? I was not aware that there were any differences :)

– Rakete1111
Sep 6 '18 at 12:41






@Rakete1111 You can have auto a = 1, 2, 3 ;, which deduces a std::initializer_list<int>, but template<class T> void f(T&&) can't be called via f(1, 2, 3).

– lubgr
Sep 6 '18 at 12:46


auto a = 1, 2, 3 ;


std::initializer_list<int>


template<class T> void f(T&&)


f(1, 2, 3)






Ahh that's what you mean. But there is no difference in the actual deduction, like a an expression that is deduced to two different types with auto and template deduction?

– Rakete1111
Sep 6 '18 at 12:53


auto






@Rakete1111 No, that I would find quite disturbing :) Is the wording misleading in the answer?

– lubgr
Sep 6 '18 at 13:00






Not at all, just curious :)

– Rakete1111
Sep 6 '18 at 13:01



Hi and welcome to stack overflow.



As this little test program shows, no matter how you specify the type of k, the compiler will never let you lose the constness of i.


k


i


#include <iostream>
#include <type_traits>
#include <string>

#define XSTR(s) STR(s)
#define STR(s) #s

template<class T>
struct type;

template<>
struct type<int>

static std::string name() return "int";
;

template<class T>
struct type<T&&>

static std::string name() return type<T>::name() + " &&";
;

template<class T>
struct type<T&>

static std::string name() return type<T>::name() + " &";
;

template<class T>
struct type<T const>

static std::string name() return type<T>::name() + " const";
;

#define REPORT_CONST(decl, var, assign)

decl var = assign;
do_it(STR(decl var = assign;), var);


template<class Var>
void do_it(const char* message, Var&&)

std::cout << "case: " << message << " results in type: " << type<Var>::name() << 'n';


int main()

const int i = 42;

REPORT_CONST(const auto &, k, i);
REPORT_CONST(auto &, k, i);
REPORT_CONST(auto &&, k, std::move(i));
REPORT_CONST(auto const &&, k, std::move(i));
REPORT_CONST(int const&, k, i);
// REPORT_CONST(int &, k, i); // error: binding reference of type 'int&' to 'const int' discards qualifiers



Expected results:


case: const auto & k = i; results in type: int const &
case: auto & k = i; results in type: int const &
case: auto && k = std::move(i); results in type: int const &
case: auto const && k = std::move(i); results in type: int const &
case: int const& k = i; results in type: int const &



http://coliru.stacked-crooked.com/a/7c72c8ebcf42c351



Note also the decay of named r-values to l-values.



The accepted answer is correct, i.e. there is no difference in regards to the compiled result. What's important to note is that the auto& version is coupling the const-ness of the k reference with the const-ness of the variable i. I figured since the question is titled 'Difference between const auto & and auto & ...' then it's important to emphasize the pragmatic difference here, in which if you don't put a the const keyword, you cannot guarantee the reference will have this cv-qualification. In some scenarios where this is desired, why leave it to chance that i will remain const in the future?


auto&


const


k


const


i


const


i


const



There is a slight difference in first case auto will be deduced to const int and in the second case to int (as you explcitly stated the const).


auto


const int


int



cpp-reference states



The keyword auto may be accompanied by modifiers, such as const or &, which will participate in the type deduction. For example, given const auto& i = expr;, the type of i is exactly the type of the argument u in an imaginary template template void f(const U& u) if the function call f(expr) was compiled. Therefore, auto&& may be deduced either as an lvalue reference or rvalue reference according to the initializer, which is used in range-based for loop.



So for you this means


// case 1
const int i = 42;
const auto &k = i; // auto -> int

// case 2
const int i = 42;
auto &k = i; // auto -> const int



More important in my opinion is, however, that the intent is stated more clearly if you state const explcitly and the guarantees the constness. Therefore in this case I would clearly prefer it.



Thanks for contributing an answer to Stack Overflow!



But avoid



To learn more, see our tips on writing great answers.



Required, but never shown



Required, but never shown




By clicking "Post Your Answer", you acknowledge that you have read our updated terms of service, privacy policy and cookie policy, and that your continued use of the website is subject to these policies.

Popular posts from this blog

𛂒𛀶,𛀽𛀑𛂀𛃧𛂓𛀙𛃆𛃑𛃷𛂟𛁡𛀢𛀟𛁤𛂽𛁕𛁪𛂟𛂯,𛁞𛂧𛀴𛁄𛁠𛁼𛂿𛀤 𛂘,𛁺𛂾𛃭𛃭𛃵𛀺,𛂣𛃍𛂖𛃶 𛀸𛃀𛂖𛁶𛁏𛁚 𛂢𛂞 𛁰𛂆𛀔,𛁸𛀽𛁓𛃋𛂇𛃧𛀧𛃣𛂐𛃇,𛂂𛃻𛃲𛁬𛃞𛀧𛃃𛀅 𛂭𛁠𛁡𛃇𛀷𛃓𛁥,𛁙𛁘𛁞𛃸𛁸𛃣𛁜,𛂛,𛃿,𛁯𛂘𛂌𛃛𛁱𛃌𛂈𛂇 𛁊𛃲,𛀕𛃴𛀜 𛀶𛂆𛀶𛃟𛂉𛀣,𛂐𛁞𛁾 𛁷𛂑𛁳𛂯𛀬𛃅,𛃶𛁼

Edmonton

Crossroads (UK TV series)