Correcting Common Async/Await Mistakes in .NET 8 - Brandon Minnick - NDC London 2024

Поділитися
Вставка
  • Опубліковано 13 січ 2025

КОМЕНТАРІ • 52

  • @MaQy
    @MaQy 9 місяців тому +73

    The part about FrozenSet and FrozenDictionary is a misconception. Those two collections were not added because they are immutable, you already have ImmutableHashSet and ImmutableDictionary. You should keep using those in most circumstances. The Frozen ones are meant to be used when they are are created on startup, kept in memory, and repeatedly used afterwards. The creation cost is way higher, but the reading part is significantly faster, so they are not suitable for operations that create them on the fly and discard them after they are finished.

    • @gregh2327
      @gregh2327 9 місяців тому +7

      I didn't want to watch the whole video to see if Brandon Minnick changed anything, but I've seen an similar video from him over the years with the same misguided info. Here's the last one; my complaints are the top comment: ua-cam.com/video/zhCRX3B7qwY/v-deo.html. Wish I had this guys confidence in being wishy-washy with the details.

    • @brandonminnick
      @brandonminnick 8 місяців тому

      Totally fair feedback. You're absolutely right - there is a higher cost to create `FrozenSet` than `IReadOnlyList` and `ImmutableList`. When running benchmarks on my app, I found that the additional performance cost was around 1ms. For this app, it's not much of a performance hit; for example, I'm not dropping any frames due to using `FrozenSet`. That being said, I appreciate the feedback and I will update my recommendations for using `FrozenSet` to include this caveat 💯

    • @AndreMantas
      @AndreMantas 7 місяців тому +5

      Amazing how the same presentation is given over and over with the same errors and misconceptions. 5 months after the previous one, the frozen collections and thread switching misinformation is still there

    • @Daniel-hb4nh
      @Daniel-hb4nh 4 місяці тому +2

      @@gregh2327 I agree. I've seen that video last year also, and now UA-cam suggests me this one. I just came here knowing for sure that a comment(s) pointing he is wrong somewhere will exist. Here they are. Now I am pretty sure the video is not worth watching

  • @49riddickful
    @49riddickful 9 місяців тому +7

    Great voice, great tempo and super captivating. Fantastic listen

  • @berkanbilgin2287
    @berkanbilgin2287 8 місяців тому +6

    THIS WAS AWESOME.. RIGHT ON POINT. NOT TOO LONG, NOT TOO SHORT. GREAT JOB, THANKS..

  • @timonkrebs6978
    @timonkrebs6978 4 місяці тому +3

    50:40 SuppressThrowing is useful when you need to wait for all tasks being finished. This is not what WhenAll does by default, because it does not wait any longer if some Task throws. Js has Promise.All and Promise.AllSettled for that and finally we can achieve the same in a concise way in C#.

  • @Ainglish-qj5bb
    @Ainglish-qj5bb 5 місяців тому

    Really quality material. Thank you for this!

  • @clystian
    @clystian 8 місяців тому +7

    46:36 Async/Await Best Practices

  • @xaviar7247
    @xaviar7247 8 місяців тому +1

    Awesome presentation. Thanks for showing the iasynenumerable idea . I saw it somewhere, used it, lost the code, and couldn't find it again. Gonna go back to using this. HERO!

  • @woutervugt
    @woutervugt 9 місяців тому +7

    Great talk. Quick question though. Why would you do work in a constructor at all? Constructors initialize memory, methods do logic. Most coding platforms I worked with (winforms, wpf, asp,etc) have a 'starting' method or event that you can use so that your constructor doesn't need any weird 'safe way to execute async void code'.
    For instance, even though you created this safe approach to async void, you still cannot handle the error in a meaningful way (eg, showing it to the user). So, instead you choose the only left option which is to trace it out. Why would you do it in that way?

    • @djupstaten2328
      @djupstaten2328 9 місяців тому

      Some like the "init" qualifier for properties because they allow for flexibility while retaining safety. Unfortunately that initialization can't be deferred to an init-method (unless you call it in the constructor). Its also a good thing to have the object in the expected state whenever a check for "is not null" or "is " returns true, and there's a potential opening between ctor --> init-method for any such checks to occur, where some properties may not have been initiated. That would open the can of worms of locking etc.

    • @TheRPGminer
      @TheRPGminer 9 місяців тому +2

      ​@@djupstaten2328But why can't we have private constructor, and have public async factory method?

    • @newbiex11
      @newbiex11 8 місяців тому

      @@djupstaten2328 yes why don't just use Onload / Init method?

    • @berkanbilgin2287
      @berkanbilgin2287 8 місяців тому

      @@TheRPGminer you can, but this more of a self triggerred/invoked operations where you just creata an instance from a factory and it kicks in

    • @brandonminnick
      @brandonminnick 8 місяців тому

      Yup - agreed, I don't recommend triggering any async code in the constructor. In the .NET MAUI apps I publish to the app stores, I trigger this logic using the `OnAppearing()` method, not in the constructor. But, for this talk it's a quick + dirty example to segueway into the pros + cons of `async void`.
      To handle the exception using `.SafeFireForget()`, you can pass in an Excecption Handler: `.SafeFireAndForget(ex => Trace.WriteLine(ex))`

  • @LZE.
    @LZE. 9 місяців тому +3

    Great talk!

  • @jinparksoul
    @jinparksoul 9 місяців тому +13

    It doesn't feel right to put work in the constructor. Especially when you need to create a lot of view models at once for say a list . I usually have some other type of async Task method to initialize work it so I or my team don't need to remember some extension method I made a year ago to handle doing work in the constructor safely.
    You also have to be very careful with using ConfigureAwait(false) before a collection clear. The ViewModel should have little to no knowledge of what or how the UI is bound to it. A UI team member may bind a list control to the Collection that is being cleared here. However if you add a ConfigureAwait(false) in the call before and the execution continues in a thread that is not the UI thread when the collection clear occurs it will fire change notification events in a different thread from the UI. This will crash the app as the control bound to the collection cannot be modified from a non-ui thread.

    • @brandonminnick
      @brandonminnick 8 місяців тому

      Agreed. In the .NET MAUI apps I've published to the app stores, I trigger this logic using the `OnAppearing()` method. But, for this talk it's a quick + dirty example to segueway into the pros + cons of `async void`.

  • @firstsector99
    @firstsector99 6 місяців тому

    Very engaging. Thank you.

  • @realrs123
    @realrs123 4 місяці тому

    Good summary of what is known for years.

  • @kylewilt3409
    @kylewilt3409 6 місяців тому +2

    Be weary of async apis that don’t provide a means of cancellation. Using the workaround described here with WaitAsync will not actually cancel the operation, it will simply cause the task to complete via cancellation while the main operation continues. This can lead to all kinds of problems.

  • @garcipat
    @garcipat 9 місяців тому +1

    For the issue with intelisense there is a recommendation to suffix async methods with Async. Not my favorit but in this case it would help

  • @zolniu
    @zolniu 10 годин тому

    43:00 I don't think that it wroks like that. async/await on it's own does not cause context or thread switch. It only re-implements your method as a state machine which can interact with Tasks. To switch the thread there has to be a code that acturally runs a task on a spearate thread - if you just create a Task using Task.FromResult() for example, all of the execution would stay on the main thread no matter how many await's were involved in calling that method.

  • @tugayersoy4489
    @tugayersoy4489 9 місяців тому +1

    it was cool, ty

  • @dismuter_yt
    @dismuter_yt 4 місяці тому

    Doesn't WaitAsync re-create the conditions of async void in case of a cancellation? I would expect the work in the initial task to keep going, because it does not observe the token. And if it raises an exception, there is no longer anything to observe it. Am I missing something?

  • @shavais33
    @shavais33 2 місяці тому

    Good grief, I did not know that await foreach or AsyncEnumerable existed. I see it now, in
    MS Learn -> C# Documentation -> Concepts -> Iterators

  • @mrqbboy
    @mrqbboy 9 місяців тому +4

    Again?

    • @gregh2327
      @gregh2327 9 місяців тому +2

      Is it still inaccurate?

    • @brandonminnick
      @brandonminnick 8 місяців тому

      @@gregh2327 If there's any innacuraces in the video, please do let me know! I've iterated this talk over the last 8 years and have certainly made improvements based on the feedback I receive. It's never my intention to mislead anyone.

    • @gregh2327
      @gregh2327 8 місяців тому +1

      @@brandonminnick Thank you for responding and being open. I unfortunately don't have time to watch and do a write up; but a couple comments in here are astute. In previous talks, my frustration has been regarding discrepancies in how the CLR works and how you describe it working.

  • @alfflasymphonyx
    @alfflasymphonyx 9 місяців тому +2

    this feels like the same video as this one:
    ua-cam.com/video/zhCRX3B7qwY/v-deo.html

  • @pakvg85
    @pakvg85 4 місяці тому +2

    the nervous manner of speaking is kind of irritating. otherwise - great presentation

  • @garcipat
    @garcipat 9 місяців тому +8

    ConfigureAwait being true by default was a dumb desicion.

    • @gregh2327
      @gregh2327 8 місяців тому

      I initially disagreed, but now I'm on the fence. I think you'd find an equal number of people that disagree.

    • @MRender32
      @MRender32 4 місяці тому

      It was the right default at the time. That’s no longer the case

  • @GeorgeTsiros
    @GeorgeTsiros 4 місяці тому

    In the first example, there will be no thread switching or even asynchronous work done. You call async code and immediatelly await on it. It is the exact same code as if you did response = httpClient.Get(...); stream = response.Content.ReadAsStream();, etc. If you want to demonstrate anynchronous code, you need to show asynchronous code. This, is not.

    • @andrewdatar9880
      @andrewdatar9880 4 місяці тому

      What would be a demonstration of asynchronous code?

  • @kahnfatman
    @kahnfatman 8 місяців тому

    The whole .NET is a mistake ... let alone async