Comment by rockyj

Comment by rockyj 3 days ago

65 replies

I did try F#, but I was new to .NET ecosystem. For 1 "hello world" I was quite surprised by how many project files and boilerplate was generated by .NET, which put me off.

I am all for FP, immutable, and modern languages. But then where are the jobs and which companies care if you write good code?

Now everyone wants languages which are easy to use with AI, while reducing workforce and "increased productivity". I have been programming for 20 years and know 4-5 languages, in India it was worse but in EU at-least I can make a sustainable living by writing Java / TypeScript. I cannot even find jobs with Kotlin + TypeScript which pay well, forget getting jobs in Elixir / Clojure / F# (there maybe a handful of opportunities if I will relocate for around 70K/year). That is why I have mostly given up on learning niche languages.

shortrounddev2 3 days ago

I like F#'s syntax when all you're doing is pure logic. But when you have to interface with any IO like a database or REST call or something, you have to abandon the elegance of ML syntax and use these ugly computation blocks. In C# you can do something like this:

    var post = await _postService.getById(id);
in F# the equivalent is basically

    let getPostById id = async {
        let! post = blogPostService.getPostById id
        return post
    }

    let post = getPostById 42 |> Async.RunSynchronously
But not really, because RunSynchronously isn't the same thing as `await`. Realistically if you wanted to handle the result of an async computation you would need to create continuations. F# isn't the only ML-family language to suffer from this; Ocaml does as well. It always seemed to me like the pattern with any asynchronous operations in F# is to either:

1. Do all logic in ML-syntax, then pass data into a computation block and handle I/O operations as the last part of your function, then return unit OR

2. Return a C#-style Task<> and handle all I/O in C#

Either way, ML-style languages don't seem like they're designed for the kind of commercial CRUD-style applications that 90% of us find ourselves paid to do.

  • throw234234234 2 days ago

    I find it personally better for CRUD applications than C# and I've written my share in both languages. Your syntax comparisons aren't exactly comparable in the sense that you haven't put in the wrapping/boilerplate around the C# code - you can't just await anywhere. You are also using an async which to run needs to know which context - this can be nice when you don't want to run the composed Task/Async on the current sync context. These days you stick to tasks if you want C# like behavior - and there's libraries to take away some SyncContext overload via custom F# CE's if you want.

    The equivalent C# to your F# would be

      task { return! _postService.getById(id) }
    
    Which is somewhat pointless anyway - just return the task from postService directly. There's also no need to run the async synchronously then - Async allow you to run the logic on task, thread, sync over and over - a very different model than tasks.

    To make C# comparable to your F# code (tasks are not the same so not quite true) you would need to define a method around it, and find a way if you want to run the resulting Task synchronously to do that safely.

      public async Task<Post> GetPostById(id) => await blogPostService.getPostById(id);
    
      // This is not entirely eq - since tasks are hot
      this.GetPostById(42).Result
  • cjbgkagh 3 days ago

    F# is a big language, it is a ML multi paradigm language that interoperates with C# so there is a lot of necessary complexity and many ways to do the same thing. A strong benefit of this is the ability to create a working functional paradigm prototype that can iteratively be refined to a faster version of itself by hot spot optimizing the slower parts with equivalent highly mutable functions while staying within the same language. Similar how one would use python and C++ and over time replace the python code with C++ code where performance is important.

    For the specific case of C# use of await it is unfortunate that C# didn't design this feature with F# interop in mind so it does require extra steps. F# did add the task builder to help with this so the 'await' is replaced with a 'let!' within a task builder block.

      let getById(id:int) : Task<string> = failwith "never"
      let doWork(post:string) : unit = failwith "never"
      let doThing() = task { 
        let! post = getById(42); 
        doWork(post); }
    
    
    
    Alternatively the task can be converted to a normal F# async with the Async.AwaitTask function.

      let getPostById1(id:int) : Async<string> = async { return! getById(id) |> Async.AwaitTask }
      let getPostById2(id:int) : Async<string> = getById(id) |> Async.AwaitTask 
      let getPostById3 : int -> Async<string> = getById >> Async.AwaitTask
    • neonsunset 3 days ago

      It is best to just use task CE full-time unless you need specific behavior of async CEs.

      The author of the original comment, however, does not know this nor tried verifying whether F# actually works seamlessly with this nowadays (it does).

      Writing asynchronous code in F# involves less syntax noise than in C#. None of that boilerplate is required, F# should not be written that way at all.

      • cjbgkagh 3 days ago

        F# is a big language so I think it is to be expected that beginners will not know these things. I don't think the fix is to simplify F# we should just understand that F# is not for everyone and that is ok.

        • neonsunset 3 days ago

          This is perfectly fine, but I think it's better to be unsure about specific language feature than confidently state something that is not correct (anymore).

          Personally, I'm just annoyed by never-ending cycle of ".NET is bad because {reason x}", "When was this the case?", "10 years ago", "So?".

          Like in the example above, chances are you just won't see new F# code do this.

          It will just use task { ... } normally.

      • shortrounddev2 2 days ago

        I understand that you CAN do this, I'm saying that it makes your code look like shit and takes away some of the elegance of ML

  • malakai521 3 days ago

    `var post = await _postService.getById(id);`

    the F# equivalent is

    `let! post = _postService.getById id`

    • alternatex 3 days ago

      You're missing the task {} block

      • neonsunset 2 days ago

        This assumes the context is already a task computation expression, which is what you'd have in asynchronous code.

      • sWW26 2 days ago

        and the C# is missing the `async Task` boilerplate

  • smoothdeveloper 3 days ago

    In C#, you can't use the await keyword in a non async method, so I find the argument short sighted.

    • shortrounddev2 3 days ago

      I don't see how that changes things. You'd have to async it all the way to the top but the syntax is still cleaner than F#. If you're using an Asp.Net controller you just declare the handler as async Task<IActionResult> and it's fine. Even program main methods can be async these days

      • malakai521 3 days ago

        The syntax is exactly the same. You have `var x = await` in C# and `let! x =` in F#

        The controller handler is also the same. It will be marked with `async` keyword in C# and `task` CE in F#

    • jayd16 3 days ago

      If your code base is already using async await it's really not an issue.

      • int_19h 3 days ago

        The point is that it's not actually different from C#, especially once you consider that F# also has task{} blocks that give you a .NET Task directly.

8s2ngy 3 days ago

I understand your perspective. I like to view niche languages as a medium for learning. For instance, I enjoy using Rust in my personal projects—even if many of these projects may never be released—because the lessons on immutability, functional programming constructs, and trait-oriented programming significantly enhance my day-to-day work. Therefore, I believe that learning niche languages, even in the absence of a robust job market, is worthwhile.

  • jen20 3 days ago

    I'm not sure I'd call Rust a "niche language" any more (perhaps in ~2018) - it's in common use across many big technology companies.

    • homebrewer 3 days ago

      It is extremely niche outside of this bubble.

      • sterlind 2 days ago

        MS is starting to use Rust pretty extensively internally. That's a lot of developers outside the "bubble."

      • vlovich123 3 days ago

        F# will likely remain niche forever. It’s likely that Rust will not given its growing and accelerating adoption by Microsoft, Google and the Linux Kernel.

        It just takes time to defeat the 40+ years of c and c++ dominance.

    • DeathArrow 2 days ago

      Just look at the job market. There are far more jobs for Go programmers and Go isn't particularly huge.

      Compared with C/C++, Java, C#, Javascript, Python, Typescript, PHP, all the rest can be considered niche.

tester756 3 days ago

>" I was quite surprised by how many project files and boilerplate was generated by .NET, which put me off.

With which language are you comparing with?

Because there's afaik csproj and maybe .sln

and both of them are let's be frank - foundational for almost all projects that arent just hello world.

Otherwise you end up with some cmakes or something similar that want to achieve something similar

owenm 3 days ago

I hear you on the opportunity side and I can't see that changing. The good news is in recent releases there's a lot less boilerplate - "dotnet new console -lang F#" results in two files, a short fsproj file and a single line of Hello World.

neonsunset 3 days ago

As sibling comment pointed out, it's just .fsproj manifest and Program.fs file. What boilerplate do you speak of? It's on the opposite end boilerplate-wise to projects made in e.g. Java or TypeScript.

For F#, projects are needed to make full applications or libraries. Otherwise, you can simply write F# scripts with .fsx and execute them via 'dotnet fsi {SomeScript.fsx}'.

(obviously you can also specify dotnet fsi as shebang and integrate these scripts into general scripting on Unix systems - it's very productive)

  • twodave 3 days ago

    I suspect they were either referring to pre-.NET Core days before the new project formats came out or they're creating projects in Visual Studio and checking all the optional boxes. There indeed did used to be a lot more required boilerplate to get some code running. Now you can run a .NET project quite nicely in VS Code with 2 total files.

    • shortrounddev2 3 days ago

      Well also if you're using Visual Studio it will generate solution files as well, not just fsproj. I grew up doing C/C++ so boilerplate project/IDE/make files as well as build objects are something I expect to see. I think people who work in primarily JIT'd/interpreted languages are used to just having a directory tree full of source files and having some CLI tool manage everything for them. Maybe a dependency list file as well, but that's about it. Python is like this, and javascript CAN be like this

      • twodave a day ago

        Yeah, overall I agree. And I honestly can't imagine anyone who works with any popular javascript framework would flinch at the addition of a .sln file or /Properties folder lol...

Foofoobar12345 3 days ago

F# is quite usable with AI. All AI models are perfectly capable of generating idiomatic F# code. In fact, because it has a nice type system, if you ask the AI to model the problem well with types before implementing, hallucinated bugs are also easier caught.

  • elcritch 2 days ago

    Same with Nim. It works surprisingly well with AI tools. I think both have more straightforward syntax so it’s easy to generate. I’m curious how more complex languages do like C++ / Rust.

    Last time I tried C++ with Copilot it was terrible.

KurtMueller 2 days ago

I find F# easy to use with AI, mainly because it's statically typed (which results in compiler errors when the LLM generates non-working code) and it's very expressive, which allow me to more easily comprehend what the LLM is trying to do.

sodapopcan 3 days ago

> But then where are the jobs and which companies care if you write good code?

Oh man, that is poignant :( They always say they do in the job description, but it always a different story once you get there.

djha-skin 3 days ago

I learn them as a fun hobby, with no salary expectations. It keeps the dream alive, and I learn a lot from the Common Lisp community that I do use in my job.

afavour 3 days ago

Opportunities do exist, even when they’re few and far between. I learned Rust in my spare time because I was really interested in it. Then we stumbled across something that would have really benefitted from a cross platform library and lo and behold, I got to use my Rust knowledge, even though the vast majority of my day job doesn’t use it.