this post was submitted on 15 Sep 2025
79 points (87.6% liked)

Programmer Humor

38564 readers
217 users here now

Post funny things about programming here! (Or just rant about your favourite programming language.)

Rules:

founded 6 years ago
MODERATORS
top 13 comments
sorted by: hot top controversial new old
[–] DickFiasco@sh.itjust.works 12 points 2 weeks ago (1 children)

What's wrong with constexpr?

[–] voodooattack@lemmy.world 10 points 2 weeks ago* (last edited 2 weeks ago) (1 children)

Back when I made this, GCC/clang were crashing left and right while compiling my project because of constexpr and auto usage with nested lambdas. It got worse with every template being evaluated until the compiler and my IDE started crashing.

I was making a react-like UI component library with all the new bells and whistles of modern C++. It was fun at first then the issues cropped up and it kinda killed my passion for the language and drove me away entirely.

Not sure about its state nowadays though.

[–] DickFiasco@sh.itjust.works 16 points 2 weeks ago (2 children)

I see. I think auto gets overused a lot by people just being lazy about writing out the full type, but constexpr is good practice in my opinion. Never had a compiler issue with them, but then I don't think I've ever used a nested lambda either.

[–] thebestaquaman@lemmy.world 8 points 2 weeks ago* (last edited 2 weeks ago) (1 children)

I really like C++ (I know, shoot me), and I think auto should be avoided at (almost) all costs.

One of the things I love about a language like C++ is that I can take one glance at the code and immediately know what types I'm working with. auto takes that away while adding almost no benefit outside of a little convenience while writing.

If I'm working with some very big template type that I don't want to write out, 99/100 times I'll just have a using somewhere to make it more concise. Hell, I'll have using vectord = std::vector<double> if I'm using a lot of them, because I think it makes the code more readable. Just don't throw auto at me.

Of course, the worst thing ever (which I've seen far too often) is the use of auto in examples in documentation. Fucking hell! I'm reading the docs because I don't know the library well! When you first bother to write examples, at least let me know the return type without needing to dig through your source code!

[–] ugo@feddit.it 4 points 2 weeks ago (1 children)
[–] thebestaquaman@lemmy.world 3 points 2 weeks ago (1 children)

Thanks, that was a good read :)

However, my impression is that he's largely using the existence of templates and polymorphism as arguments that "we don't really care about type". I disagree: A template is essentially a generic type description that says something about what types are acceptable. When working with something polymorphic, I'll prefer ParentClass&, to indicate what kind of interface I'm working with.

Sure, it can be very useful to hide exact type information in order to generalise the code, but I think that's a weak argument for hiding all type information by default, which is what auto does.

[–] ugo@feddit.it 1 points 2 weeks ago (1 children)

auto doesn’t really hide all type information, but that is besides the point. The point is that in the vast majority of cases you don’t really care about the specific types your code is dealing with, you only care about the properties of the type.

If I write get_descriptors()[some_idx] what is the type returned by get_descriptors()? You can’t know, because nothing tells you, but you know it must be a type that supports random access indexing. If I put the result of the function call in a for loop, it must support iteration. If I return it by move, it must support being moved.

99% of code is like this. Then you have specialised code that might be doing bit wrangling and what not, in which case you must know that you are dealing exactly with, say, an uint32_t and not just something that supports left shift.

For the 99% of cases, auto makes the code simpler, more correct, more robust, more consistent, and shorter to write and read.

The post I replied to in particular got angry at documentation that uses auto. I have written documentation that looks like this

auto my_var = MyConcreteType { /* … */ };
auto const* x = my_var.get<uint32_t>();
auto const& y = my_var.get_or<uint32_t>(42);

How would such code benefit from not using auto?

MyConcreteType my_var = MyConcreteType {}; just repeats the type, adds zero useful info.

MyConcreteType my_var {}; is inconsistent. Where is the equal sign signifying that this is an assignment?

Similarly, what would you gain by saying uint32_t const* x = my_var.get<uint32_t>();? Nothing, you just doubled your maintenance when you need to change the type passed to the template parameter.

[–] thebestaquaman@lemmy.world 1 points 2 weeks ago* (last edited 2 weeks ago) (1 children)

Similarly, what would you gain by saying uint32_t const* x = my_var.get<uint32_t>();

To be frank: You gain the information that MyConcreteType::get<uint32_t> returns a uint32_t, which I otherwise couldn't infer from the docs. Of course, I could assume it, based on the template parameter, but I don't want to go around assuming a bunch of stuff in order to read docs.

Take an example like auto x = my_var.to_reduced_form(), it's very clear that x is the "reduced form" of my_var, which could be meaningful in itself, but what type is it? I need to know that if I want to do anything with x. Can I do x += 1? If I do, will that modify my_var? Let's say I want to make a vector of whatever to_reduced_form returns... and so on.

All these questions are very easily answered by MyConcreteType x = my_var.to_reduced_form(). Now I immediately know that everything I can do with my_var, I can also do with x. This makes me happy, because I need to do less digging, and the code becomes clearer to read.

[–] ugo@feddit.it 2 points 2 weeks ago

To answer you: no, x += 1 cannot mutate my_var, because it’s a copy. If you wanted something else you would say auto& or auto*.

And if the type of x is such that even a copy can mutate the original (e.g. maybe it’s a wrapper around an index with an overloaded operator+=() that mutates the original entry in the storage), you are probably working with a specialized system of some kind, since that breaks the semantics of the language, and hopefully you would know from context that you have such cases.

And yes, in such an environment I would see “never use auto for this wrapper type” as a valid additional strategy to help ensure code correctness.

But by and large my direct experience is that using auto “almost always”, as Herb Sutter puts it, is beneficial and doesn’t harm readability or understandability.

[–] voodooattack@lemmy.world 3 points 2 weeks ago* (last edited 2 weeks ago) (1 children)

The problem is that lambdas with a capture ~~aren’t strongly typed~~ are uniquely typed, so you have to use decltype/auto. And if you pass such a lambda to a function you’ll have to use auto as well.

If you write a lambda with a capture that calls itself recursively you’ll have to pass it to itself as an auto argument as part of the call signature.

I think this article explains it better: https://artificial-mind.net/blog/2020/09/12/recursive-lambdas

Edit: fixed wrong terminology

[–] barubary@infosec.exchange 2 points 2 weeks ago (1 children)

This is an entirely new way to misuse "strongly typed" that I was not aware of before. Amazing.

Thank you!

[–] voodooattack@lemmy.world 3 points 2 weeks ago

You’re welcome. Just don’t blame me when your brain starts cursing in foreign languages you don’t even know. ;)

[–] ame@lemmy.frozeninferno.xyz 3 points 2 weeks ago

Integrated divinity environment