Most active commenters
  • shortrounddev2(9)
  • neonsunset(6)
  • cjbgkagh(6)
  • DeathArrow(3)

←back to thread

Why F#?

(batsov.com)
438 points bozhidar | 66 comments | | HN request time: 1.069s | source | bottom
1. rockyj ◴[] No.43546371[source]
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.

replies(11): >>43546449 #>>43546522 #>>43546629 #>>43546648 #>>43546914 #>>43546918 #>>43547243 #>>43547274 #>>43547691 #>>43548828 #>>43553311 #
2. djha-skin ◴[] No.43546449[source]
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.
3. owenm ◴[] No.43546522[source]
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.
4. afavour ◴[] No.43546629[source]
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.
5. neonsunset ◴[] No.43546648[source]
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)

replies(1): >>43546713 #
6. twodave ◴[] No.43546713[source]
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.
replies(1): >>43547401 #
7. 8s2ngy ◴[] No.43546914[source]
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.
replies(1): >>43547025 #
8. sodapopcan ◴[] No.43546918[source]
> 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.

9. jen20 ◴[] No.43547025[source]
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.
replies(2): >>43547427 #>>43554703 #
10. raphinou ◴[] No.43547243[source]
It is possible to start your project with the script possibilities offered by F# (as mentioned in the blog post). It is absolutely a viable approach and I even blogged about it a couple of months ago: https://www.asfaload.com/blog/fsharp-fsx-starting-point/
11. shortrounddev2 ◴[] No.43547274[source]
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.

replies(4): >>43547546 #>>43547687 #>>43548815 #>>43552613 #
12. shortrounddev2 ◴[] No.43547401{3}[source]
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
replies(1): >>43560901 #
13. homebrewer ◴[] No.43547427{3}[source]
It is extremely niche outside of this bubble.
replies(3): >>43548059 #>>43549836 #>>43550169 #
14. smoothdeveloper ◴[] No.43547546[source]
In C#, you can't use the await keyword in a non async method, so I find the argument short sighted.
replies(2): >>43547584 #>>43547833 #
15. shortrounddev2 ◴[] No.43547584{3}[source]
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
replies(1): >>43547973 #
16. malakai521 ◴[] No.43547687[source]
`var post = await _postService.getById(id);`

the F# equivalent is

`let! post = _postService.getById id`

replies(1): >>43549167 #
17. tester756 ◴[] No.43547691[source]
>" 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

18. jayd16 ◴[] No.43547833{3}[source]
If your code base is already using async await it's really not an issue.
replies(1): >>43548793 #
19. malakai521 ◴[] No.43547973{4}[source]
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#

replies(1): >>43548150 #
20. vlovich123 ◴[] No.43548059{4}[source]
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.

replies(2): >>43548838 #>>43554736 #
21. shortrounddev2 ◴[] No.43548150{5}[source]
It's absolutely not exactly the same; let! is only available within a computation block. If you want to return some value from the computation block and return to Functional land without having to pause the thread you need to use a continuation, which C# has built in syntactic sugar for in async/await and F# does not.
replies(2): >>43548389 #>>43548964 #
22. sparkie ◴[] No.43548389{6}[source]
`await` can only be used in an `async` function. How is that so different from `let!` only being available in a computation expression?
replies(1): >>43548847 #
23. int_19h ◴[] No.43548793{4}[source]
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.
24. cjbgkagh ◴[] No.43548815[source]
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
replies(1): >>43548876 #
25. Foofoobar12345 ◴[] No.43548828[source]
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.
replies(1): >>43549421 #
26. johnisgood ◴[] No.43548838{5}[source]
Personally I will always prefer C's simplicity to Rust's complexity. Could be just me.
replies(2): >>43549749 #>>43552342 #
27. shortrounddev2 ◴[] No.43548847{7}[source]
because an async function doesn't require you to change syntaxes to get them to work
replies(1): >>43550216 #
28. neonsunset ◴[] No.43548876{3}[source]
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.

replies(2): >>43549020 #>>43551121 #
29. xigoi ◴[] No.43548964{6}[source]
A computation block is the equivatent of an async function;
30. cjbgkagh ◴[] No.43549020{4}[source]
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.
replies(1): >>43549170 #
31. alternatex ◴[] No.43549167{3}[source]
You're missing the task {} block
replies(2): >>43549495 #>>43554862 #
32. neonsunset ◴[] No.43549170{5}[source]
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.

33. elcritch ◴[] No.43549421[source]
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.

34. neonsunset ◴[] No.43549495{4}[source]
This assumes the context is already a task computation expression, which is what you'd have in asynchronous code.
35. kstrauser ◴[] No.43549749{6}[source]
I find Rust vastly simpler than C. If the code compiles, it's probably a valid expression of the business logic I encoded. I might've screwed up that logic, of course, and no language can prevent me from messing that up. I know! Many have tried, and I've defeated them with my ability to misrepresent my ideas! But at least with Rust, I'm reasonably confident that the code will actually do the thing I asked it to do. I'm never confident like that with C until I've run it a few hundred times without crashing.

(Yes, I'm familiar with the rich ecosystem around helping devs not write crummy C. I worked at Coverity at one point. If anything, that gave me enormous fear and respect of the hoops you have to jump through to be reasonably sure C code isn't completely broken.)

replies(1): >>43555885 #
36. mmoskal ◴[] No.43549836{4}[source]
According to Stack Overflow developer survey [0] Rust is at 12.5%, roughly a half of C# or Java and a quarter of Python. Also more than twice Ruby. So definitely not niche.

[0] https://survey.stackoverflow.co/2024/technology#most-popular...

replies(3): >>43551417 #>>43554714 #>>43558534 #
37. sterlind ◴[] No.43550169{4}[source]
MS is starting to use Rust pretty extensively internally. That's a lot of developers outside the "bubble."
38. Smaug123 ◴[] No.43550216{8}[source]
It's actually sort of the other way round. C# has hardcoded syntax for async/await. F#'s syntax for async/await is a fully-general user-accessible mechanism.
replies(1): >>43554413 #
39. shortrounddev2 ◴[] No.43551121{4}[source]
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
replies(3): >>43551302 #>>43551470 #>>43551479 #
40. neonsunset ◴[] No.43551302{5}[source]
Please stop insisting on this. Task CE exists since F# 6.0 and handles awaiting the CoreLib Tasks and ValueTasks without any ceremony.
41. askonomm ◴[] No.43551417{5}[source]
In my mind not niche means having jobs, and Rust has no jobs, not in any meaningful amount at least, and none at all in most countries. That puts it deep in the niche category for me.
replies(1): >>43554724 #
42. ◴[] No.43551470{5}[source]
43. cjbgkagh ◴[] No.43551479{5}[source]
Are you saying you prefer Ocaml to F# or C# to F#? Your example was indeed inelegant but it is also poorly designed as you take 4 lines to reproduce a function that is already built in, people can poorly design code in any language.
replies(1): >>43556410 #
44. vlovich123 ◴[] No.43552342{6}[source]
Rust like any language can be as simple or as complex as you want it to be. The complexity raises with the performance of the code you write but that’s also true of C. Unlike C that complexity is combined with guarantees that your code won’t crash in weird and unpredictable ways.
45. throw234234234 ◴[] No.43552613[source]
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
46. KurtMueller ◴[] No.43553311[source]
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.
47. sparkie ◴[] No.43554413{9}[source]
They're not so different in that regard. C# `await` can be adapted by making an awaitable and awaiter type[1], which isn't to dissimilar to how a computation expression in F# needs to implement methods `Bind`, `Return`, `Yield`, etc.

In both languages these patterns make up for the absence of typeclasses to express things like a functor, applicative, monad, comonad, etc.

[1]https://ecma-international.org/wp-content/uploads/ECMA-334_7...

48. DeathArrow ◴[] No.43554703{3}[source]
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.

49. throwaway2037 ◴[] No.43554714{5}[source]
To be clear, that developer survey asked:

    > Which programming, scripting, and markup languages have you done extensive development work in over the past year, and which do you want to work in over the next year?
It does not ask if you are gainfully employed and using this language for your job.

Also, in the same results, just above Rust, I see:

    > PowerShell 13.8%
<sarcasm> So, I guess that we can safely say that Microsoft PowerShell is still more popular than Rust. </sarcasm>
replies(1): >>43554785 #
50. DeathArrow ◴[] No.43554724{6}[source]
It's popular in the "let's rewrite X in Rust" community which are very actively posting on HN, Reddit and wherever they can. That gives the impression it is not niche.

But the moment you search Rust on LinkedIn, you can see the truth.

51. DeathArrow ◴[] No.43554736{5}[source]
I will take C, C++ or Zig over Rust any day. For some people, like me, the Rust way of doing things isn't a good fit. It's not a model I enjoy working with.

I like F#, Haskell, Elixir but not Rust.

52. winrid ◴[] No.43554785{6}[source]
Powershell is probably more popular, it's used a lot for IT stuff so we never hear about it but it's there.
53. sWW26 ◴[] No.43554862{4}[source]
and the C# is missing the `async Task` boilerplate
54. dboreham ◴[] No.43555885{7}[source]
This seems...a very contrarian sentiment. Imho while C might lead you to create slightly fragile code once in a while, Rust is something like two orders of magnitude more complex.
replies(2): >>43557675 #>>43559130 #
55. shortrounddev2 ◴[] No.43556410{6}[source]
I'm saying that I wish computation blocks looked better in F#. Instead of:

    let foo id = async {
      let! bar = getBar id
      return bar
    }
I would prefer

    let async foo id =
      let! bar = getBar id
      bar
or even something like

    let async foo id =
        getBar! id
So that computation blocks don't feel like you're switching to an entirely different language. Just wrap the ugliness in the same syntactic sugar that C# does. As it is, C# can achieve arrow syntax with async methods more elegantly than F# can:

    async Task<string> foo(int id) => await getBar(id);
This, to me, is also part of a larger problem of F# introducing unique keywords for specific language functions instead of reusing keywords, like

    member this.Foo = ...
and

    member val Foo = ...
replies(1): >>43558250 #
56. ninetyninenine ◴[] No.43557675{8}[source]
No I know what he’s saying. C is not “slightly fragile code once in a while”. When you up the complexity of the code and the amount of code and the people working on the code the fragility becomes pervasive.
replies(1): >>43558164 #
57. kstrauser ◴[] No.43558164{9}[source]
That's a huge part of it. If I stumble across random Rust code, I can assume that it's using typed data correctly, that it's not accessing freed memory, that it's not allocating but never freeing, that length checks are being enforced, etc. If they weren't, it wouldn't even compile (and the compiler would explain why).

Glancing at random C code tells you nothing about what happens with the data flowing into and out of it.

In my experience with it, rustc has been insistent on making me write code that's actually correct. I could translate that code back to C and have better C code than I would likely have written on my own. If there were something similar to `gcc -Werror -Weverything-rust-would-complain-about` — and if that thing were even possible — I very well might stick with C. Oh, and something as fast and ergonomic and informative as rust-analyzer would be hugely welcome.

58. cjbgkagh ◴[] No.43558250{7}[source]
Your criticism is rather incoherent and it is difficult for me to make sense of it.

You don't even have to use the computation block for that and can use the built in functions as I mentioned earlier and gave 3 examples of.

You're both complaining about extra keywords while trying to make the case of adding yet another one. Thus your complaint boils down to F# not picking the exact keywords that you like - that the language is not specialized to exactly how you want to use it. In language design there are always tradeoffs but I'm unable to see how your suggestions would improve the language in the general case or even in your specific case.

Computation expressions are a generalized concept which are there to add the exact kind of syntactic sugar that you're after. It's better than C# in that you can create your own as a first class concept in addition to using the built in ones. It's there for the exact purpose of creating mini-embedded DSLs, the very thing you're complaining about is the exact point of it.

F# is not for everyone, nor should it be.

replies(1): >>43558582 #
59. trott ◴[] No.43558534{5}[source]
> According to Stack Overflow developer survey [0] Rust is at 12.5%, ... So definitely not niche.

The annual survey is very popular in the Rust community. Its results are often used for advocacy. Participation by Rust developers is very high. So what you have is a classic case of a selection bias.

60. shortrounddev2 ◴[] No.43558582{8}[source]
> You're both complaining about extra keywords while trying to make the case of adding yet another one

I did no such thing. async is already a keyword in F#, I'm just saying they should drop the brackets and remove the required return statement.

> In language design there are always tradeoffs but I'm unable to see how your suggestions would improve the language in the general case or even in your specific case

It would make the language easier to read, for one, and would reduce the amount of specialized syntax needed for specific features. It would preserve the original ML-style syntax for an extremely common operation and not force users into wrapping anything upstream of an async call in a computation block, which is the ugliest syntax feature of F#

> Computation expressions are a generalized concept which are there to add the exact kind of syntactic sugar that you're after

I understand that, and my argument is they failed to do so. The syntax looks bad. They could keep it for all I care, but they should add even more sugar on top to make it not look so bad.

replies(1): >>43558917 #
61. cjbgkagh ◴[] No.43558917{9}[source]
'async' is not a keyword in F#, it's a builder instance no different to the ones that you can create. It's just built in to the standard library.

The return statement is only required if you want to return something form the computation expression. In your example you use async { let! x = f(); return x}, which can be reduced to async { return! f()}, which can be reduced to f().

The rest is your opinion that I don't agree with.

replies(1): >>43559651 #
62. aldanor ◴[] No.43559130{8}[source]
Rust is like two orders of magnitude more simple (if you're not going to delve into its darker corners).

You have a single line serialization into/from absolutely anything. You have logging, tracing, cli libraries, error handling - most of those are one liners.

You have enums. Enums are business logic. Enums are often the way the world works.

You press enter and it builds, no pre setups, sub modules, cmake files and whatnot.

63. shortrounddev2 ◴[] No.43559651{10}[source]
The distinction in this case is utterly meaningless. This is about the ergonomics of the language. Which are lacking the minute to break out of pure functional land
replies(2): >>43560391 #>>43560987 #
64. cjbgkagh ◴[] No.43560391{11}[source]
I disagree with both of your statements
65. twodave ◴[] No.43560901{4}[source]
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...
66. neonsunset ◴[] No.43560987{11}[source]
Read this first: https://learn.microsoft.com/en-us/dotnet/fsharp/language-ref...

There are multiple alternate asynchronous computation expression implementations which give different ergonomics and behavior (like https://github.com/TheAngryByrd/IcedTasks). There's an entire CE extension to specifically enable this kind of convenience and flexibility too: https://github.com/fsharp/fslang-design/blob/main/FSharp-6.0...

None of this is possible in C#, at least without jumping through many hoops and ending up with extra boilerplate (and this is okay, C# has enough of its own complexity). As cjbgkagh noted, providing this type of control is the whole point and what makes F# so powerful.