>If the language has the problem that people are fighting with the language in order to become productive with it, perhaps something is wrong with the language, and not the programmers?
If they stop fighting after they learn, then no, it's not a problem with the language. It's the very definition of learning.
Now, the language might be harder to learn than others, and that might be a problem in itself.
But not just because people have to struggle at first "in order to become productive with it". That's true for any language that brings anything new to the table.
>The argument of pass by reference, or borrowing is that it’s more performant than cloning by default. In general, computers are getting faster, but systems are getting more complex.
Rust was not created to piggyback on "computers getting faster".
Good points. Performance not being a primary concern is a meme that really has grown beyond its place. Performance is absolutely a primary concern in very hot code, at execution scales where it means big differences in aggregate hardware/infrastructure cost, energy usage, and so on. Or in space, where transistors necessarily need to be "low tech" huge radiation resistant ones. There will never be a lack of applications for highly optimized software, no matter how small the niche gets. Computing power may be cheap but it is NOT free, nor infinite, nor universally portable.
> If they stop fighting after they learn, then no, it's not a problem with the language. It's the very definition of learning.
I think the problem here is how hard people have to fight.
How about take a step back, and view the language as a completely noob, you may found many complexes in the language is unnecessary.
For example, in the sample code, you can found many #[macro_use]s, they are needed to import macros to the local scope. But hey, why I can't just `use slog::{!macro_name, something_not_macro}`?
Well, the doc said: "macro definitions aren’t namespaced within modules like function definitions are. To prevent unexpected name clashes when using external crates ....".
But why macro is not namespaced then? Why user has to learn "the macro is not namespaced" so they have to #[macro_use]? Why not just implement it in a more intuitive way?
Rust team need to consider this: How many complexities can be completely removed from the language, while maintain it's final goal (be a fast and safe system language).
Eh, in the last press release from Mozilla, they say otherwise:
> Confidence when doing systems-level programming. For old hands, Rust’s guarantees mean you can spend less of your time and attention on avoiding pitfalls, which makes it possible to reach for more ambitious goals. For newer systems programmers, it means a shorter path to being able to start writing production code.
> This second point is worth emphasizing. In traditional systems programming, there’s just an intense amount of discipline you need to learn in order to write production code. Because the discipline takes so long to master, and the stakes of slipping up are crashes and security vulnerabilities, most people will only trust seasoned veterans of systems programming to ship production C++ code. But with Rust, you don’t have to master this discipline to start shipping production code; you know the language has your back.
As far as I can read, it states the purpose of the language is correctness, and developer productivity?
>As far as I can read, it states the purpose of the language is correctness, and developer productivity?
Yes, but _while_ being as fast as C or faster (e.g. without aliasing issues).
With those two goals they do not imply they'd sacrifice speed for them. They had stated elsewhere that speed and no-cost-abstractions are extremely important for Rust.
I think everyone agrees that "fighting" the borrow checker is annoying, however at the risk of sounding like a Rust apologist I find that a couple interesting things have happened over ~2 years with the language:
1. The borrow checker usually kicks in when I want to take shortcuts that would bite me in the future. 90% of issues I hit with it can be solved with copy/clone.
2. When I take the time to satisfy the borrow checker I find my architecture is better. Usually I'm tripping the borrow checker because I don't have a clear single-owner or separation of concerns.
Also, if you're looking for an "easy" to program languages there's tons of them out there and it definitely doesn't make sense in my mind to use Rust where C#/Erlang/Java/etc fit the domain space better.
These points are absolutely consistent with my experience. I'd also dispute the author's claim that, "If the language has the problem that people are fighting with the language in order to become productive with it, perhaps something is wrong with the language, and not the programmers?" And I would argue that this is an ill-placed criticism: it doesn't mean something is "wrong" with either the language or the programmers: instead, it just means that it's a technology that has a learning curve. I also had early awkward periods of low productivity when I first started learning how to touch-type: that doesn't mean that something is wrong with keyboards!
I found out that 90% of the time, I'm more at fault than the borrow checker because I'm writing code that would be unsafe should concurrency/parallelism be introduced. But a lot of people don't naturally write or even know how to write concurrency safe code, especially when there's no intention for the code being parallelized.
I'm not arguing the borrow checker is wrong, I'm just saying, rather than being the Star Ship: Enterprise's computer, and asking me to put up shields, I want the language to put up the shields for me. Yes, they might result in a power drain, but 99% of the time, I don't want to get shot, at the expense of power.
Just treat things as immutable, and copy them around, or automagically allow for wrapping / unwrapping a (Ref)Cell/(A)rc
Rust is designed to make it easy and safe to build the 1% of your application that cares strongly about resource usage, because for certain application domains (like low-level server software or operating systems) that section might make up more like 10% or even 100% of your application. In those situations, Rust can provide a lot of advantages that a less strict tool might not be able to provide.
It sounds to me like the design philosophy that informs Rust is different from the design philosophy you'd prefer, and that's okay! That means Rust might not be a compelling fit for your problems, but not every tool is going to fit every problem domain. That doesn't mean the tool is broken, just that it's designed for a different situation.
I also saw (1) be true most of the time. Why isn't Copy / Clone / Immutability the default (pass by value), as opposed to the exception? It seems like if you found this was slow, you could go back and fix things when you need the performance, as opposed to wasting time doing lots of copy / clones (and really ugly code).
How does binding all sorts of short-lived variables make your architecture better?
Rust seems so low level that I don't get why people flock to it when C# and Java would be much better fits? Is it just because they're bored of C# and Java and want to learn a new language? Even Erlang should scratch that itch! I would guess that only the Servo team really benefits from Rust, or maybe some wall street traders who need hyper speed! Every other app should work fine with something slower.
Wow, big list. I guess more people need the speed Rust gives after all! Would you say the job market is good for Rust? I only sunk my teeth into React and I'm loving it but I'm open minded.
Compared to Java or C#? Miniscule. There are jobs happening, but they're often not pure Rust jobs. It's changing a lot though; there used to be like, two or three at best.
I'm extremely productive in Rust vs other languages, especially for young projects where I spend a lot of time refactoring. Things like borrow checker errors just don't happen very often to me anymore, or they catch legitimate bugs and are generally quite easy to fix. I just don't run into them very often at this point.
I've written Python and Java professionally. In both cases I felt like I just couldn't express what I wanted to, at all times, in my program. This led to me being a lot less comfortable refactoring - relying on tests that were far too heavily tied to implementation details.
I had some really complex logic I needed to write for one of my services. Basically I needed to represent a 'thing' and that 'thing' could be in a few different states. I wrote out the code, types first, so that every state had a type. I was able to do this generically using Rust's associated types in a way that allowed me to implement the state machine for multiple 'things'. It was very testable, very readable code that I had high confidence in. I would have had a much harder time doing this in Java/ Python (I have done similar things, and you can get value from the approach, but not the same level). I've refactored that code a lot with confidence.
Can I ask what you use as a database abstraction layer, if any, and what methods you use to convert your request params/JSON to queries?
I was playing around with trying to do this a while back (while also trying to learn Rust some), and it felt very cumbersome to have to write many specific Diesel update methods to handle different types of requests. It could have been my now knowing a better way to accomplish it, or it could have been the type of queries I was trying to model aren't a good fit for a staticly typed system like that (and there may be a better way to architect it). In any case, it felt like I was poorly approximating the strict typing I had already implemented in Postgres, a few times over, when I would be happy just handing it to Postgres and letting it throw out errors as needed.
The only data stores I currently use are redis and dgraph. I write raw dgraph queries, though I have a small, highly focused library that I use to generate some of them.
As for redis, I use it as a dumb KV store, no need for an abstraction layer.
Dgraph is awesome but lacking some serious basic features such as backups and automatic file format update on new versions. I hate to export it all, update, re-import...
I would be refactoring in any language - the idea behind the service, the requirements, etc, are changing a lot as I'm building these services out. These are external factors that would exist regardless of the language, as I continue to explore the problem space.
Not anything serious, no. I'm decently familiar with it though - I'm fascinated by the actor model, and have read a decent portion of Joe Armstrong's thesis.
I don't understand why anyone would go for C# or Java.
If you want GC and/or fast dev speed, go for JavaScript. If you really need static typing, go for TypeScript. JS has one of the fastest VMs on the planet and virtually everywhere.
If you need speed and don't want to mess around with decades of C/C++ legacy quirks, go for Rust.
> I don't understand why anyone would go for C# or Java.
Library support immediately comes to mind. If I need an implementation of, say, the Kerberos protocol, I'm more likely to find a binding in Java than in JS.
I feel like it's a mixed bag. For every 10 packages, maybe 8 are half-baked and strapped on in a hurry with no care. That said, those other 2 are usually amazing. Java has an edge there, because there's a lot more fully supported, corporate backed, high quality packages for Java.
I recall a few years ago when some people were talking about porting an application to nodejs. The idea was dropped when it was found that there was no LDAP library that supported Kerberos SASL authentication, which was an absolute requirement for the application. (A quick look suggests that there still lacks an implementation of this).
Note that the number of packages is not a great indicator of the quality of implementation.
I'm with you, I love TypeScript! It's my favorite language of all time! But people using Java and C# are usually tied to a platform and SDK that TypeScript can't easily interoperate with. But I guess if you can replace your whole app with an Electron version, go with TypeScript all the way!!
Java and the JVM are a fantastic platform to develop on. The development experience is great and the performance is excellent.
* Java code means what it says and is simple to reason about.
* Top-notch IDEs like Eclipse and IntelliJ IDEA. Great debugging and diagnostic support, both at development-time and in production (e.g., logging and instrumentation frameworks, crash reporting).
* Java is meaningfully cross-platform, though in practice this is mostly useful for portable development (as opposed to portable production deployment). I can develop my services on Windows or Mac, then deploy to Linux in production, with zero fuss. (Don't even need a container)
* Feature-rich standard library. Powerful concurrency primitives and data structures, async, and futures libraries built right into the Java runtime (java.util.concurrent). You can dip into frameworks like Akka for more sophisticated use-cases.
* Huge open source community that's working on projects like Apache Hadoop, Hive, Spark, Avro, Cassandra, etc. Take a look at how many Apache Java projects there are, for example: https://projects.apache.org/projects.html?language#Java . Top-notch DI frameworks like Netty, Guice, Dagger, Spring. Top-notch REST and RPC programming frameworks like gRPC, Thrift, Cap'n Proto, JAX-RS/Jersey.
* Java code is extremely reliable. Java code cannot crash in complex ways. Try/catch covers the vast majority of concerns for practical error handling and recovery. Java code's consistent use of GC, exceptions, and treatment of sync/async means that all libraries are modular and can be meaningfully and safely composed with each other. For comparison see What Color Is Your Function: http://journal.stuffwithstuff.com/2015/02/01/what-color-is-y...
* Multiple languages that can interoperate: Java, Groovy, Scala, Kotlin, Clojure. The GraalVM and GraalJIT show amazing potential: https://www.graalvm.org/
* Because Java is so widely used, many tools and client libraries and such support Java before other platforms.
* A large population of developers understand Java and will be productive immediately on the project (vs. obscure languages).
* Modern, concise syntax that supports anonymous classes and functions, Lambdas, streams, Optionals. Java continues to improve over time and pick up concepts from other languages.
* Java has good support for generic programming, unlike some other modern languages (e.g. Go). Generic programming is efficient and doesn't typically have downsides that make people afraid to use it (like in e.g. C++).
* The JVM is a very high performance implementation that can rival native programs on important benchmarks for many server use-cases. I've built a lot of software and have rarely, if ever, run into performance reasons to use a different language. (Usually a small native library with JNI bindings meets the need when that's the case.) Other platforms rarely achieve a fraction of the JVM's performance in server-side use-cases. See benchmarking results from a previous comment I wrote on this topic (internal citations removed):
> Folks might also be interested in the TechEmpower web framework benchmark. The top Java entry ("rapidoid") is #2 on the benchmark and achieves 99.9% of the performance of the #1 entry (ulib in C++). These frameworks both achieve about 6.9 million requests per second. The top Java Netty server (widely deployed async/NIO server) is about 50% of that, while the Java Jetty server, which is regular synchronous socket code, clocks in at 10% of the best or 710,000 R/s.
> NodeJS manages 320,000 R/s which is 4.6% of the best. In other words, performance-focused async Java achieves 20x, regular asynchronous Java achieves 10x, and boring old synchronous Jetty is still 2x better than NodeJS throughput.
* I'm personally a fan of Rust and appreciate its concepts, but garbage collection suffices and is easier to work with for the majority of use-cases I encounter. C# and .NET also have many of the properties above, though not the large open source community.
People who are starting businesses and have a practical, get-the-job-done attitude would be wise to consider Java. It's versatile enough to do almost anything on the server-side.
The biggest drawback of the JVM is the resource use, warmup time, and unpredictable GC latency. The JVM certainly is freakishly fast once it gets going, but it's not always cost efficient for it's performance.
These are exactly the reasons why JVM is the primary platform for us and Clojure/Java are the primary languages to use. If you compare languages by performance, library availability and developer popularity it is extremely hard to beat Java/JVM
I really hope he doesn't fall into the trap I did when starting out with rust. I came into it with the perspective of an application programmer and I quickly became salty when my knowledge didn't translate smoothly into rust code. "Fighting the borrow checker", for me, was simply a clash of philosophies. My way of thinking versus the rust way of thinking.
The central issue, for me, is that with rust you MUST understand the memory model completely in order to be productive. You cannot wing this, you cannot assume anything. I was not used to this being an application programmer and this lead to the root of all my suffering. Once I wrote down all the borrowing rules and deeply thought about it, I finally became productive in the language.
I see the author shows symptoms of not understanding the memory model. His code proves it. Instead of being angry, he should try to view rust with an open mind and ask "why did they design it this way?". Be curious, create experiments, probe the rust semantics, master the borrow checker. There are rewards to be had for those willing to push through the pain of integrating new knowledge.
Pretell, how does my code not show an understanding of the memory model? I understand how the memory model works, I just want the language to _do the easy thing_ (copy) in most scenarios, and make embedding (Cells, RCs) / unwrapping more straightforward.
> If the language has the problem that people are fighting with the language in order to become productive with it, perhaps something is wrong with the language, and not the programmers?
Or maybe writing correct (i.e. safe) code is more difficult then most programmers realize and they lack the humility to admit it.
The question here is -- are you "productive" because you write _good_ code faster, or because the language allows you to _not think_ about certain aspects of design and just postpone issues to the future (or GC).
OP wants to be fast and "productive", but forgets that in systems programming your first priority is good design and solid implementation. This is because _other_ components will be relying on your code to do their job. It pays to go extra step and make sure your component is properly written. Rust offers more than any other systems language to help with that.
There are two sides to any coin. The complexity of the language is a consequence of the complexity of the problem it is trying to solve and the constraints it is operating under.
The borrow checker is not just some crazy torture device that exists for no reason. Criticizing the difficulty of the borrow checker without considering why it exists in the first place is completely misguided.
False dichotomy -- there are always more than two sides to any issue. People find themselves fighting with the borrow checker because their mental model of what they're writing is not what Rust is interpreting. This is probably indicative of two things: (1) their mental model is incorrect in some ways, and (2) Rust's semantics don't map that well onto how we think about systems programming problems. If satisfying the borrow checker makes you think "whoops, that was dumb" then you're in category 1. If it makes you think "aargh, how do I express this in a way Rust will accept?!?" then you're in category 2. With an ideal language category 2 wouldn't exist.
It is, and it's a claim that I didn't make. But even if it were true, I think people won't believe that it's true, because there will be instances where they won't understand the borrow checker's reasoning.
One of the reasons I love functional programming is it allows me to think less about edge cases, because values are passed as values.
If Rust passed copies instead of references by default, people who wanted performance could fight (or embrace, whatever your preferred term/level of experience is) the borrow checker, while everyone else could enjoy the fundamental safety of not sharing data.
Rust doesn't pass references by default, so I'm not really sure what kind of dichotomy you're trying to draw here. Also, such fundamental changes likely have far reaching effects that would likely change Rust itself in ways you can't anticipate.
> The biggest issue I have with the defaults, and the borrow checker is that places in FP where you would normally pass by copy — pass by value, in Rust instead it assumes you want to pass by reference. Therefore, you need to clone things by hand and pass the cloned versions instead. Although it has a mechanism to do this automatically, it’s far from ergonomic.
Yeah, I don't really understand what the OP is trying to say there other than "I'm frustrated." The lack of examples makes it hard to know. Passing by value works just fine in Rust, so it's hard to unpack what the author was getting at.
Yeah, still doesn't make any sense to me. That request is asking for implicit references (a small syntactical change as far as I can tell), which is different from what you asked for and doesn't seem to match up with what the OP wants either.
Without examples or further clarification, I don't think it's possible to understand what the OP actually meant.
1. Anywhere I call clone(). That should be the default / automatic.
2. Anywhere where I can get copies than explicitly move for lambdas
The problem with the latter is that I have to call clone outside of the lambda and bind it to a short-lived variable into to get moved to the scope of the lambda.
(1) is basically saying "Rust should have copy semantics by default instead of move semantics." I'm very glad that's not the case. Those implicit copies would become easy performance foot guns. In most Rust code I write, using `clone` is fairly uncommon, so it's also optimizing for the wrong thing anyway. Finally, things that aren't cloneable are far more common than things that aren't moveable, so you'd need to smooth out that gap as well.
(2) is interesting and I'd be happy to see work done there to make that nicer, but it's hard for me to understand that being a big sticking point.
Agreed. Whether rust has copied its features - I don't know, but I have re-implemented many of Ada's features for several projects in other languages (mostly C), so it seems there is an amount of basic features like storage pools/memory pools which are usually required for systems programming tasks.
>'Similarly, a Java programmer must understand the runtime behaviour of the JVM, but until OpenJDK, the implementation language was irrelevant."
Can someone say why did the implementation language become relevant with the arrival of OpenJDK? My java app should run on any JVM no? After all that was the raison d'être of Java no? Or is the author referring to something else?
> We have the Rust Evangelism Strikeforce nudging us towards rewriting everything in Rust.
I really wish this would stop being repeated. At this point it's a meme that was rarely, if ever, actually accurate. People just repeat it because they've seen comments urging the use of Rust, and this leap is easy to make and justifies a bit more pushback.
It's not about "rewriting", it's about new projects. People are starting new projects that do the same thing as old ones all the time. What you actually see is a lot of Rust advocates trying to influence those people to use Rust instead of C or C++, so instead of yet another library implementation in C or C++, we have one in Rust.
If it's better than the prior implementation, it might be used. This part has nothing to do with language, and better implementations come along all the time and are adopted to some degree. For example, Nginx. I remember the days when your choice for an open source webserver that could scale was basically Apache (there were a few others, but generally weren't used for serious sites).
Misrepresenting this cycle is just spreading FUD that activates a constituency to circle wagons. It's not helping anyone (as we're seeing from both sides of the political spectrum, which have gone all-in on this tactic lately).
Misrepresenting this cycle is just using FUD to activate a constituency to circle wagons. It's not helping anyone (as we're seeing from both sides of the political spectrum, which have gone all-in on this tactic lately).
It's a offhand remark in a post that's not about any of this. You're going out of your way to interpret this small thing in the worst, flame-inviting way.
> You're going out of your way to interpret this small thing in the worst, flame-inviting way.
Which supposes a different way to interpret this. Feel free to provide a different interpretation or this very short and straightforward sentence, and I'll definitely consider it.
Also, to be clear, I'm not positing that the author intends to cause the stated effect, I'm stating that the meme does this by it's nature, and the meme is also self-propagating by its nature and this same effect. The wording I used (that you quote) can be interpreted assigning intent to the author, which was not my goal, so I've changed them slightly.
(I would note the change with a note on the original, but I think these comments are sufficient to document it).
I feel any statement about about what a group of people are doing, especially when it can be taken derogatorily or dismissively, should be accurate, or at least with some basis provided. It was in the article, in my eyes it's fair game for criticism, even if the only criticism is that I wish people would pay attention to whether it's true, as the criticism in this case is.
If someone described C developers as "wild cards that think all our security concerns are overblown" I think a lot of C developers would feel rightly misrepresented.
> And no interpretation requires some weird political swipe.
I compared how the meme works to how memes in politics are working to divide, as I think that's the most obvious parallel that is easily visible to most people. Notice how I didn't promote one side over the other, as it's a common tactic in politics in general, but more visible at both extremes.
My point is, it's an absurd amount of verbiage and offtopic thread derailment and hot air over some dumb joke that is basically irrelevant. You've written some bizarre grumpy rustacean equivalent of 'this site doesn't scroll properly on Android'.
I had a brief dalliance with pony last year, though I wasn't coming at it from a systems perspective. At the time, it was the "lowest level" programming language I had used.
Maybe the author's perspective is just that much different from mine (he's used to things like C++ and Rust), but "simple" is just about the last word I'd use to describe pony. Don't get me wrong: Pony is a pleasure to work with, and it is no more complicated than necessary for the problem it's trying to solve, but adding the capabilities layer to the normal type system makes for a language that has a lot going on. I'd expect that the amount of fighting newcomers do with the capabilities system is similar to how people fight Rust's borrow checker. Maybe the author just didn't have enough actors that wanted to share state/communicate where it wasn't an issue for him.
In any case, I highly recommend anyone thinking about picking up a new programming language to give pony a look, especially if problems you're trying to solve are amenable to high levels of concurrency and need to run fast. The pony capability system basically eliminates data race bugs (within pony), and also guarantees against runtime crashes. It's got a bit of a learning curve, and the package ecosystem around it isn't great yet, but it's doing unique things to solve problems many languages aren't attempting to address.
NOTE: The following is my understanding, from discussions on IRC and reading articles. If I'm wrong, please, please, please correct me (and provide good backup to what you are saying), because this is fundamental, and for me is one of the biggest misunderstanding in Rust.
When deciding to use passing by value vs passing by reference, the only deciding factor should be ownership. If it makes sense to give ownership, pass by value, if it makes sense to retain ownership, pass by reference. This will NOT affect performance, the actual implementation of copying memory around vs reassigning which part of the program owned that piece of memory is an implementation detail and the rust compiler will try to do the right thing.
I stopped fighting the borrow checker once I stopped treating Rust references as if there were C pointers, using them to minimize memory copy instead of concerning myself with ownership. I find the book does a bad job of explaining this, I'd be happy to give a try to write a chapter about it, but need to first confirm that my understanding is correct.
I'm disappointed about the lack of D, too, since the author gave Nim a try. And it's definitely more mature than, say, Pony (which is an interesting language for other reasons). Also, you can get excellent performance in D, really close to what you will get with C, and that's very often the crucial factor in systems programming.
I definitely consider D one since it's meant to be a continuation to C / C++ without caring about 'backwards compatibility' which is a good thing imho.
If they stop fighting after they learn, then no, it's not a problem with the language. It's the very definition of learning.
Now, the language might be harder to learn than others, and that might be a problem in itself.
But not just because people have to struggle at first "in order to become productive with it". That's true for any language that brings anything new to the table.
>The argument of pass by reference, or borrowing is that it’s more performant than cloning by default. In general, computers are getting faster, but systems are getting more complex.
Rust was not created to piggyback on "computers getting faster".
It was created to be as fast as possible.