Treating Arc<T>.f()
as T.f()
is arguably unexpected behavior to the average non-rust programmer. Why should Arc<T>
suddenly act like T
only for method calls? Why shouldnât Arc::new(1) + Arc::new(1)
also work? It also allows the outer typeâs methods to shadow the inner typeâs methods, which is a very strange thing to see in a language that doesnât allow overloading. The fact that Arc recommends this workaround of calling methods as if they were static and manually passing &self as the first argument makes it seem like this is undesired.
And on the contrary, C++ has the arrow operator which is much more explicit, doesnât suffer from the name collision issues, and allows you to do this:
headânextânext.value += 1;
Instead of this:
(*(*head).next).value += 1;
Just seems like what Rust chose to do is the worst - verbose, inconsistent, implicit.
Comments
solidiquis1 ⢠23 points ⢠2023-08-30
- Deref coercion is incredibly important for ergonomics. Itâs so that we can effectively interact with all of the different types of smart pointers as though they were regular olâ references e.g. allowing
Box<T>
where&T
is expected.Arc::new(1) + Arc::new(0)
does not work becausestd::ops::Add
takesself
rather than&self
. Keep in mindArc<T>
derefs to&T
and notT
.- Yes, I agree with you that deref coercion is black magic to Rust newcomers.
edit: spelling
[deleted] ⢠0 points ⢠2023-08-30
âTurned out to be the wrong choiceâŚâ is it really so controversial though? Yes, thereâs a threshold in at which the ergonomic benefit isnât worth the cost of the cognitive overhead but this hasnât crossed it. Ergonomics is important in a modern language like Rust. You just need to familiarize yourself with the few sugary areas: Async, closures, and Deref.
solidiquis1 ⢠1 points ⢠2023-08-30
âTurned out to be the wrong choiceâŚâ is it really so controversial though? Yes, thereâs a threshold in at which the ergonomic benefit isnât worth the cost of the cognitive overhead but this hasnât crossed it. Ergonomics is important in a modern language like Rust. You just need to familiarize yourself with the few sugary areas: Async, closures, and Deref.
simony2222 ⢠11 points ⢠2023-08-30
I think the point of
Deref
is actually to make the behavior of pointers consistent. That is, as youâve pointed out, rust doesnât have an->
operator and it kind of auto-decides whether to apply*
on your pointer or not.Deref
gives access to the same syntactic sugar to custom pointer types. Note that because of the implicit behavior it implies, the documentation strongly advises to implement this trait only on pointer-like types.Although I agree that I find this confusing at times (especially with generics), and I wouldnât mind calling a
as_ref()
(or equivalent) all the time to disambiguate things.solidiquis1 ⢠1 points ⢠2023-08-30
I donât think thatâs a proper use of
AsRef
, however.AsRef
is meant for converting between different types of references, thus the result ofas_ref()
could result in a type with completely different behavior e.g. callingas_ref
on aString
to get anOsStr
.In the case of an
Arc
we want the behavior ofArc
as well as the underlying typeâs behavior soDeref
is appropriate.
mina86ng ⢠3 points ⢠2023-08-30
Saying that itâs against everything Rust stands for is a stretch but I agree that
->
for method calls and field access would be better (possibly with additionalAutoDeref
marker trait for types which do want auto deref in those situations).
Trader-One ⢠3 points ⢠2023-08-30
In Rust, the Deref trait allows you to override the * operator. When you call a method on a type that implements Deref, Rust will automatically dereference the value for you as many times as needed to call the method on the inner type. This is called âderef coercionâ.
this behavior only applies to method calls. Operators such as + are not methods. the + operator is associated with the Add trait. To make an expression like Arc::new(1) + Arc::new(1) work, you would need to implement the Add trait for Arc
. kernelmethod ⢠1 points ⢠2024-09-22
I know Iâm resurrecting an old comment here, but is there any writing anywhere as to why Rust doesnât do deref coercion on operators? Itâs an interesting choice, since you could always opt into the coercion by calling `.add` explicitly.
Trader-One ⢠1 points ⢠2024-09-23
discuss it at rust forum.
puttak ⢠-11 points ⢠2023-08-30
Rust by nature prefer productivity over explicitly.
tauphraim ⢠5 points ⢠2023-08-30
To explain the downvotes: this is just false. Rust usually prefers explicit, and OP noted that deref is an exception.
Also I donât think rust aims at âproductivityâ. To me productivity is a negative term, used by people only concerned with outputting amounts of code, regardless of the readability and maintenability of said code.
puttak ⢠2 points ⢠2023-08-30
If Rust prefer explicit what about lifetime elision? What about trait object become
'static
by default? A lot of things that implicit by Rust that cause headache for me when I still learning it.tauphraim ⢠1 points ⢠2023-08-30
I guess there are more than one exception. For me, lifetime elision tended to make a first learning pass easier, if anything (you can assume it âjust worksâ in simple cases), but someone elseâs experience may differ. I think itâs not a feature aimed at making learning easier though.
puttak ⢠1 points ⢠2023-08-30
That why I said Rust prefer implicitly over explicitly. When you know how lifetime elision works you donât need to type it and let Rust implicit for you. The problem is only who understand Rust will know this rule.
Another example of implicitly is an extreme type inference. You donât need to specify the type for
let
most of the time and the compiler will pick it for you automatically, including type parameters.glasket_ ⢠1 points ⢠2023-09-03
Preference for explicit doesnât mean everything has to be explicit. Sometimes itâs beneficial to have implicit behaviors for readability or usability since explicit boilerplate can introduce a lot of visual and mental noise. I do agree that some of the implicit behaviors are kind of âfootgunnyâ when youâre first learning, but I think in general Rust has struck a pretty good balance on what should be implicit, while still leaning towards making the user be explicit with more things than most other languages.
puttak ⢠1 points ⢠2023-09-04
Preference for explicit doesnât mean everything has to be explicit.
Not true in my experience. See some examples from my previous comments. Another example is blanket implementation.
Sometimes itâs beneficial to have implicit behaviors for readability or usability since explicit boilerplate can introduce a lot of visual and mental noise.
That only applied when the reader is already know Rust.
while still leaning towards making the user be explicit with more things than most other languages.
Can you show some example of the other languages?
glasket_ ⢠1 points ⢠2023-09-04
Not true in my experience. See some examples from my previous comments. Another example is blanket implementation
I think you may have misunderstood what I meant here. âPreference for explicit doesnât mean everything has to be explicitâ isnât something you can prove false. Iâm saying that a language can prefer explicitness and still have parts of it be implicit, it just means the language will have a tendency to lean towards being explicit.
If you mean that you donât think Rust prefers being explicit, then fair, but from the way the languageâs designers talk they make things explicit then go back and add implicit behaviors for ergonomic reasons. To me, thatâs a preference for being explicit while still being willing to allow implicit when itâs beneficial.
That only applied when the reader is already know Rust
Yeah, a common understanding is that explicit code favors novices while implicit code reduces the burden on veterans. The important thing is striking a good balance of having enough explicit to not make a write-only language while also not having too little implicit so that it isnât overly-verbose boilerplate-laden code. This is part of the concept of language ergonomics.
As noted in that post though, some instances of implicit behavior aid learnability, with lifetime elision being the example used that I agree with. It reduces the burden on newcomers since they can typically ignore lifetimes until they hit a roadblock that forces them to deal with them explicitly.
Can you show some example of the other languages
- Unrestricted implicit type conversions, what most people point to as âbad implicitnessâ, is present in a lot of languages. C and C++ have an entire rule-set for determining how the implicit conversions will resolve. Rust chose to limit this solely to deref coercion, everything else must be explicitly cast.
- Ruby on Rails, notoriously, implicitly generates plurals of models. The plural of Cow? Kine.
- Haskell (primarily, there are some others) has an interesting notation called âpointfreeâ or tacit programming where function parameters themselves are implicit.
Rust, while still having implicit behaviors, tries to focus on taking explicit pieces of the language and shifting them towards implicitness in order to improve ergonomics without allowing the behaviors to cause problems.
- Deref coercion is the easiest to abuse (inheritance hacks, for example), but itâs still limited to dereferencing.
- Lifetime elision, which seems to be the bulk of what you dislike, primarily focuses on removing repetitive lifetime annotations like how type inference removes repetitive type annotations. It is noted to be a bit too implicit in some cases (reborrows can occur implicitly, as noted in the ergonomics blog post), but you can also enforce explicit lifetimes if necessary. Otherwise, it doesnât cause problems that wouldnât arise with explicit lifetimes.
- Personally, I donât think of blanket impls as implicit. I feel like that stretches the definition of implicit a bit too much. Implicit implies that the language is doing something without input: types are inferred implicitly, lifetimes are elided implicitly, references autoderef implicitly on access, etc. But calling a blanket impl an implicit behavior feels off, like saying the addition operator working on all numbers is implicit. Dunno, to me that isnât the same as the usual implied magic of implicit behaviors, itâs something thatâs defined and you have to explicitly make use of the trait functions anyways.
Alright, this post has gotten too long. If you think Rust is focused on productivity at the expense of explicitness then there isnât much I can do to convince you, but in my opinion the language designers clearly consider things carefully before moving away from fully explicit implementations; ergonomics/productivity are concerns, but I feel like theyâre secondary to avoiding the pitfalls of âimplicit magic.â
puttak ⢠2 points ⢠2023-09-04
Thanks for your time written this post.
poelzi ⢠-9 points ⢠2023-08-30
I love defer because you can elegantly simulate inheritance by deref to the base class, which is an embedded struct
[deleted] ⢠8 points ⢠2023-08-30
This is cursed.
mina86ng ⢠1 points ⢠2023-08-30
And yet, people will keep doing it regardless of how much some of the Rust users hate inheritance.
poelzi ⢠1 points ⢠2023-08-30
I donât understand the negative reaction of this pattern. I would link some advice how to implement a better API, in a more rusty way:
I have plugin system in which the implementations must implement some traits. There is also some shared functionality which each backend must implement. The backend struct just stores a pointed to the shared struct and I have a macro that implants deref to the shared struct.
I can easily share functionality between backends and can write obj.name() instead of obj.shared().name()
Is this a bad design and if yes, why ?
mina86ng ⢠0 points ⢠2023-08-31
I donât understand the negative reaction of this pattern.
C++ does inheritance thus inheritance is bad. /s
glasket_ ⢠1 points ⢠2023-09-03
The unofficial Rust Design Patterns book gives a pretty good explanation of why most people donât like seeing this pattern used in Rust. The tl;dr is that itâs not the intended use of Deref and itâs slightly different from and weaker than true inheritance so it causes some things to work in unexpected ways.
Most importantly this is a surprising idiom - future programmers reading this in code will not expect this to happen. Thatâs because we are misusing the Deref trait rather than using it as intended (and documented, etc.). Itâs also because the mechanism here is completely implicit.
This pattern does not introduce subtyping between Foo and Bar like inheritance in Java or C++ does. Furthermore, traits implemented by Foo are not automatically implemented for Bar, so this pattern interacts badly with bounds checking and thus generic programming.
Using this pattern gives subtly different semantics from most OO languages with regards to self. Usually it remains a reference to the sub-class, with this pattern it will be the âclassâ where the method is defined.
Finally, this pattern only supports single inheritance, and has no notion of interfaces, class-based privacy, or other inheritance-related features. So, it gives an experience that will be subtly surprising to programmers used to Java inheritance, etc.
Now for your specific implementation I sort of grasp what youâre doing, and it sounds like delegation, i.e. obj delegates the name function call to its shared member in your example, correct? If thatâs the case, then you can use a facade and crates like delegate in order to map outer facade functions to the inner shared fieldâs methods. If Iâm wrong then please correct me, Iâm just not 100% on exactly what youâre doing without seeing the code.
phazer99 ⢠1 points ⢠2023-08-30
> It also allows the outer typeâs methods to shadow the inner typeâs methods, which is a very strange thing to see in a language that doesnât allow overloading.
Rust supports overloading with traits, and thatâs what
Deref
uses.I agree implicit
Deref
âing feels a bit magical at the beginning when most everything else in Rust leans towards explicitness. After a while you get used to it and, IMHO, the improved convenience is worth it in this case.
[deleted] ⢠1 points ⢠2023-08-30
golang does the same exact thing with structs. it works fine.
kohugaly ⢠1 points ⢠2023-08-30
It is the one place where Rust violates the explicit-over-implicit guideline to significantly improve ergonomics.
The
Deref
andDerefMut
trait basically override the*
dereference operator, to make smart pointers behave like actual pointers. Without it, youâd have to explicitly cast smart pointers to references, before calling methods of the inner type. This would be especially annoying forString
andVec<T>
which coerce tostr
and[T]
respectively. It would also be somewhat annoying for various kinds of mutex guards.Rust basically unifies the
.
and->
operators into one, by deducing which one is applicable.Rustâs ability to insert
*&
until it finds appropriate method can be used for limited form of method overloading. (for examplefor &T { fn my_method(self)...}
takes precedence overfor T {fn my_method(&self)...}
but has the same function signature).
[deleted] ⢠1 points ⢠2023-08-30
Yeah this took me a while to wrap my head around back when I first learned Rust. Itâs treated very lightly in the book and itâs very confusing to a newcomer.
You get used to it but I still dislike it.
[deleted] ⢠0 points ⢠2023-08-30
That page one should also say:
Thou shalt not implement
Deref
to emulate OOP inheritance.buldozr ⢠1 points ⢠2023-08-31
That page one should also say:
Thou shalt not implement
Deref
to emulate OOP inheritance.