• m_‮f@discuss.online
    link
    fedilink
    English
    arrow-up
    59
    ·
    3 months ago

    Relevant comment

    I don’t use Rust much, but I agree with the thrust of the article. However, I do think that the borrowchecker is the only reason Rust actually caught on. In my opinion, it’s really hard for a new language to succeed unless you can point to something and say “You literally can’t do this in your language”

    Without something like that, I think it just would have been impossible for Rust to gain enough momentum, and also attract the sort of people that made its culture what it is.

    Otherwise, IMO Rust would have ended up just like D, a language that few people have ever used, but most people who have heard of it will say “apparently it’s a better safer C++, but I’m not going to switch because I can technically do all that stuff in C++”

    • ZILtoid1991@lemmy.worldOP
      link
      fedilink
      arrow-up
      33
      ·
      3 months ago

      D has a lot of other reasons why it haven’t caught on.

      • The language is way too similar to both C and C++. In fact, you can get C and C++ code working in D with minimal changes (most annoying part is int meaning different things in both languages), and nowadays I can pull the conversion off on the regular. It’s like the only changes that were made to the language are making things more consistent, making things work better. etc. Rust is more like a language that is a marriage between C-style languages and OCaml. Rust is also a “functional first” language with the ability of opting out from the functional part, first when I’ve heard about Rust it was through this “fancy and new programming paradigm, that promises to fix everything wrong with programming”, and the D team wanted to be left out from making const the default mutability.
      • The GC sounds scary for most people, who want to develop desktop applications. It’s rapid fast, will be even faster soon, and there’s tricks to either opt-out from it, or to get a @nogc thread going (using that for audio).
      • Marketing issues. Walter Bright, while a genius compiler developer, is not a good marketing person.
      • Formerly D was quite infamous for its toxic users, with Walter going infamous for his “let nazis code” line for a while. Allegedly, most of the toxicity moved over to openD, which made some people question why Adam D. Ruppe forked the language.
      • The whole shenanigans around moving from D1 to D2, also Tango.
      • 6nk06@sh.itjust.works
        link
        fedilink
        arrow-up
        8
        ·
        3 months ago

        moving from D1 to D2

        I was learning a few different languages at the time, and it’s a bit fuzzy in my head, but I remember the confusion between the 2 compilers. One was incomplete and/or obsolete, and the other one was in beta or something and you shouldn’t use it anyway. I chose nothing and tried something else because it was annoying to watch. Walter, as the BDFL, should have decided on something but it never arrived, and the hype went away.

        All I remember is that some Japanese guy wrote a bullet-hell library and game in D, and that was great. But the community was too confusing for me.

    • calcopiritus@lemmy.world
      link
      fedilink
      arrow-up
      28
      ·
      3 months ago

      I kinda disagree. The reason rust caught on is because it is much safer than C++ while having the same or even better performance. And in some contexts, being garbage collected means bad performance.

      Before rust you could either have a fast language (C/C++) or a memory safe language (any other language. That is, languages with garbage collector). But if you required memory safety and peak performance, there wasn’t any option.

      Yes, the reason that rust is both memory safe and fast is because it has a borrow checker. But the borrow checker is the means, not the end.

      • AnyOldName3@lemmy.world
        link
        fedilink
        arrow-up
        10
        arrow-down
        1
        ·
        3 months ago

        Garbage collection doesn’t guarantee memory safety and it’s perfectly possible to create a memory-safe language without garbage collection. There are plenty of garbage collectors for C++ (and until C++23, support for garbage collection was part of the standard, although no one implemented it), and languages like C# let you interact with garbage-collected objects in unsafe blocks.

        • Decq@lemmy.world
          link
          fedilink
          arrow-up
          15
          ·
          3 months ago

          Exactly, if garbage collection meant memory safety then why do we get null pointer exceptions about every 5 minutes in Java. Garbage collection is about memory leaks, not safety. Imho the borrow checker is a better solution than garbage collection and faster to boot.

          • calcopiritus@lemmy.world
            link
            fedilink
            arrow-up
            13
            ·
            3 months ago

            Null safety and memory safety are different features.

            Null safety means that you cannot access a struct’s fields without first checking if the pointer to that struct isn’t null. And this must be a compile-time check.

            Memory safety means that you cannot read or write to/from memory that has been free-ed. Without leaks ofc, otherwise it would be very easy.

        • calcopiritus@lemmy.world
          link
          fedilink
          arrow-up
          1
          ·
          3 months ago

          How can you not have memory-safety while also having a garbage collector?

          Garbage collection means that all objects live as long as you have a reference to it. Which means that you can only dereference a pointer to invalid memory if you willingly create an invalid pointer, or reinterpret the type of one pointer into another. Going out of bounds of an array counts as the first case.

          If a language has garbage collection but no compiler/interpreter supports it, then the language doesn’t have garbage collection.

          • AnyOldName3@lemmy.world
            link
            fedilink
            arrow-up
            1
            ·
            3 months ago

            For a start, having a garbage collector doesn’t mean its use is mandatory, but even in a language where the garbage collector is mandatory, keeping an array alive as long as any references to it exist doesn’t stop you doing things like getting muddled about its length and reading/writing past the end. Mandatory garbage collection only prevents temporal memory bugs like use-after-free, not spatial memory safety bugs like buffer overruns, which need to be prevented by other mechanisms like bounds checks.

            • calcopiritus@lemmy.world
              link
              fedilink
              arrow-up
              1
              ·
              3 months ago

              As I said, I don’t consider going out of bounds of a buffer a memory safety issue. Forcing the programmer to handle an out-of-bounds case every time there is an array access can be incredibly tedious. So much that not even rust forces you to do so. And if that language has iterators, it’s even less of an issue.

              I consider out-of-bounds array access to same as casting a pointer to another type. Just because a language lets you do it, it doesn’t mean that it is not memory safe. It is a performance feature, since checking the bounds every time is always possible (and incredibly easy to implement), but also with too big of an impact when you could just check the length once per loop instead of per loop iteration.

              • sus@programming.dev
                link
                fedilink
                arrow-up
                4
                ·
                edit-2
                3 months ago

                buffer overflows are critical for memory safety since they can cause silent data corruption (bad) and remote code execution (very bad). Compared to those a “clean” unhandled runtime error is far preferable in most cases.

              • AnyOldName3@lemmy.world
                link
                fedilink
                arrow-up
                0
                ·
                3 months ago

                If you’re going to change the definition of words, it’s pretty easy to show that garbage collection on its own is sufficient, but it’s not possible to have a useful conversation if someone’s using their own personal definition of the terms being discussed. The generally accepted definition of memory safety includes deeming out-of-bounds accesses and other spatial memory safety issues unsafe.

                • calcopiritus@lemmy.world
                  link
                  fedilink
                  arrow-up
                  1
                  ·
                  3 months ago

                  With your definition this conversation doesn’t make sense though. Since rust’s direct array access doesn’t perform bounds checks when building in release mode. And it doesn’t require using unsafe.

                  • AnyOldName3@lemmy.world
                    link
                    fedilink
                    arrow-up
                    1
                    ·
                    3 months ago

                    That’s not what Rust’s documentation says. It does a compile-time bounds check if it can prove what the index might be during compilation, and a runtime bounds check if it can’t. In release mode, it tries harder to prove the maximum index is below the minimum length, but it still falls back to a runtime bounds check if it can’t unless you use get_unchecked, which is unsafe.

          • Euphoma@lemmy.ml
            link
            fedilink
            English
            arrow-up
            1
            ·
            edit-2
            3 months ago

            I’ve gotten segfaults in python with only the standard library

      • Phen@lemmy.eco.br
        link
        fedilink
        arrow-up
        4
        arrow-down
        1
        ·
        3 months ago

        At one point long ago (just for a short while), I thought Delphi was destined to take that place. It was much higher level while still letting you go as low level as you wanted- it didn’t have garbage collection but it made it pretty easy to keep track of what is or isn’t allocated, on top of having good tools to find leaks on runtime. But it had too many problems too: the Pascal base and the association with drag and drop coders being some of the first ones, followed by a series of bad decisions by whatever company was responsible for it at any given week.

      • ZILtoid1991@lemmy.worldOP
        link
        fedilink
        arrow-up
        1
        arrow-down
        1
        ·
        3 months ago

        The “better” performance is due to the built-in multi-threading support, and that functional programming makes it relatively safer to pull off. Otherwise single-threaded Rust is very hard to optimize.

        • anton@lemmy.blahaj.zone
          link
          fedilink
          arrow-up
          1
          ·
          3 months ago

          Rust has monomorphisation like C++ and every function has the aliasing guarantees of restrict, a keyword rarely seen in C code bases use and C++ doesn’t even support.
          This means you can get more optimisations while writing in an intuitive style, where C/C++ requires some changes to the code.

          On the other hand rustc has some hiccups with argument passing and rvo. One could argue that that’s just the compiler while the aliasing problems are part of the language in the C/C++ case, but while there is only one rust compiler its performance is the languages performance.

          For most use cases they are about equally fast.

    • soc@programming.dev
      link
      fedilink
      arrow-up
      7
      ·
      edit-2
      3 months ago

      “apparently it’s a better safer C++, but I’m not going to switch because I can technically do all that stuff in C++”

      The main difference between C++ and D was that (for most of the time in the past) D required a garbage collector.

      So, D was a language with similar Algol-style syntax targeting a completely different niche from C++.

      Trying to correct your quote, it should read something like “I’m not going to switch because I can’t technically do all that stuff in D that I’m doing in C++” for it to make any sense.

      • adb@lemmy.ml
        link
        fedilink
        English
        arrow-up
        4
        ·
        3 months ago

        “Before rust you could either have a fast language (C/C++) or a memory safe language (any other language. That is, languages with garbage collector).”

        Ada managed to do safe and fast over forty years ago.

        • BatmanAoD@programming.dev
          link
          fedilink
          arrow-up
          6
          ·
          3 months ago

          Before Rust, the main argument I heard from C++ enthusiasts against Ada was that it was a nice idea but too “design-by-committee”. Which, yeah, is an ironic thing for C++ fans to say, but I guess that was enough 🤷

          • adb@lemmy.ml
            link
            fedilink
            English
            arrow-up
            1
            ·
            3 months ago

            I’ve never used Rust but from my very cursory knowledge of what the borrow checker entails, it wouldn’t add so much to Ada.

            Use of pointers is already strongly discouraged by the simple fact that the language is designed to rarely truly need them. Besides, the compiler itself chooses automatically whether to pass data by value or by reference depending on the required results and what is most efficient. You can also specify parameters passed to a function as read-only. Finally, another thing that Ada does to prevent yourself shooting in your foot is to enforce strong encapsulation of functions and data.

            Overall, one way to put it in simple terms is that Ada requires the programmer to give enough information to the compiler so as to ensure that it actually outputs what you want it to, which as far as I understand, is sort of what the borrow checker is there for. The downside to the Ada approach is that it is very verbose, but with the kind of code editing capabilities we have nowadays it’s certainly not as much a hassle as it was when Ada came out.

            Anyways, I suspect that both languages are different enough in overall paradigm that trying to solve problems in Ada the way you would in Rust would probably be quite frustrating and give rather poor result.

            • bitcrafter@programming.dev
              link
              fedilink
              arrow-up
              1
              ·
              3 months ago

              The borrow checker is a lot deeper than merely making pointer use a bit safer; it provides a model for data ownership based on affine types.

              Also, to the extent that “Ada requires the programmer to give enough information to the compiler so as to ensure that it actually outputs what you want it to,” if I understand you correctly, that is basically true of every language with a static type system. At the extreme end, we have dependently typed languages which essentially let you express arbitrary mathematical propositions in your types and, if the program compiles, then you know that they will always be satisfied without having to do any checks at runtime.

    • CanadaPlus@lemmy.sdf.org
      link
      fedilink
      arrow-up
      5
      ·
      edit-2
      3 months ago

      Yep, and that’s not a bad thing. Switching has a cost, and using the old standby is probably the way to go if there’s a bunch of new things, none of which are definitely, inarguably better than it.