Navigation Menu

Skip to content

Instantly share code, notes, and snippets.

@avdi
Created November 28, 2011 20:01
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save avdi/1401793 to your computer and use it in GitHub Desktop.
Save avdi/1401793 to your computer and use it in GitHub Desktop.
Function return as lvalue in C++
#include <iostream>
int& foo() {
static int pointless_static_var = 23;
return pointless_static_var;
}
int main(int argc, char** argv) {
std::cout << "foo() = 42: " << (foo() = 42) << std::endl;
}
// Outputs:
// foo() = 42: 42
@avdi
Copy link
Author

avdi commented Nov 28, 2011

Completely consistent with the C++ memory model... and completely surprising to most programmers. Especially when they accidentally type = when they meant == in an if statement.

@avdi
Copy link
Author

avdi commented Nov 28, 2011

...you can also do this with a plain local variable, but my compiler at least has the sense to issue a warning in that case.

@clifton
Copy link

clifton commented Nov 28, 2011

You should be able to do an assignment on the return value if it's a reference though. Instead of the pointless_static_var you could be modifying the value of something meaningful. It's an odd language construct, but could reduce some code duplication in writing getters and setters.

@TwP
Copy link

TwP commented Nov 28, 2011

And this is why I always put my constants on the left-hand side when performing equality comparisons

if (42 = foo()) { std::cout << "compiler error" << std::endl; }

@avdi
Copy link
Author

avdi commented Nov 28, 2011 via email

@avdi
Copy link
Author

avdi commented Nov 28, 2011

@clifton: yeah, it's totally legit, it just surprises people. I don't think I'd ever use it for a getter/setter, but there are reasons to have it there.

@raggi
Copy link

raggi commented Nov 28, 2011

I've always disliked how that looks, but I can't deny the logic of it.

Actually, it can be important to follow this general pattern in other languages too. Another classic example in ruby, of doing equality comparisons, is that the equality 'operator', being just a method dispatch, you want to execute on developer supplied data not user supplied data, as that's more deterministic. In short, exactly as above, put your 'constants' (even if that's just your "baseline object") on the left hand side to increase the likelihood of deterministic and expected dispatching.

@avdi
Copy link
Author

avdi commented Nov 28, 2011 via email

@kikito
Copy link

kikito commented Nov 28, 2011

"constants on the left-hand side when performing equality comparisons"

Some people call those "Yoda-assignations", for obvious reasons.

My only complains are: 1) They don't look very nice and 2) You don't always have a constant to compare - you could do foo() = bar(). Help you Yoda will not on that one.

@avdi
Copy link
Author

avdi commented Nov 28, 2011 via email

@clifton
Copy link

clifton commented Nov 28, 2011

A similar argument is applicable in JavaScript-land, where I see many (particularly nodejs) programmers use comma-first variable declaration.

var ruby = "is pretty cool"
  , yoda = "not so pretty think yoda does"

This type of notation will trigger syntax errors when you make a mistake, which is preferable to creating globals accidentally.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment