What is Span in C# and why you should be using it

Поділитися
Вставка
  • Опубліковано 12 лип 2021
  • Become a Patreon and get source code access: / nickchapsas
    Check out my courses: dometrain.com
    Hello everybody I'm Nick and in this video I am going to talk about Span of T in C#. Span was introduced in C# and .NET alongside a series of optimisations all the way back in .NET 2.1 and it has come to my attention that it is really confusing to understand if you just read the documentation for it. In this video I will break it down for you and help you understand exactly what it is, how it works and how you can use it in your applications today!
    Don't forget to comment, like and subscribe :)
    Social Media:
    Follow me on GitHub: bit.ly/ChapsasGitHub
    Follow me on Twitter: bit.ly/ChapsasTwitter
    Connect on LinkedIn: bit.ly/ChapsasLinkedIn
    Keep coding merch: keepcoding.shop
    #csharp #dotnet #span

КОМЕНТАРІ • 274

  • @nickchapsas
    @nickchapsas  2 роки тому +239

    Hello everybody. As some people have already pointed out, after 13:50, when I'm returning a ReadOnlySpan, the ToString() on line 21 should be removed. I didn't notice because of the implicit operator. If you leave the ToString() in then you still allocate the string you return.
    - Keep coding

    • @lipatovsa7
      @lipatovsa7 2 роки тому +2

      Great video. But where could I check source code?

    • @nickchapsas
      @nickchapsas  2 роки тому +3

      @@lipatovsa7 The source code is available to my Patreons

    • @PeterManger
      @PeterManger 2 роки тому +6

      Thought you were just testing if we paid attention!

    • @roflex2
      @roflex2 2 роки тому +1

      @@nickchapsas What would be the benefit/differences in using method(in string text) method(ref string text) ?

    • @nickchapsas
      @nickchapsas  2 роки тому +3

      @@roflex2 strings are immutable in C#. Even if you pass down explicitly by reference your can't change the value of the string. You just point to a new string reference type

  • @GeraldOSteen
    @GeraldOSteen 2 роки тому +277

    I'm a systems developer and primarily work with low-level languages like ASM, C, C++, etc., so I don't have a lot of experience on the intricate details of optimizations for managed languages. Explanations like these are invaluable and I find them immeasurably useful so I thank you very much for this.

    • @KayOScode
      @KayOScode 2 роки тому +6

      Yeah typically managed languages have to add complex features to get the same speeds we get using lower level languages. Its a big trade off in my opinion. In those languages you usually either have to swallow performance penalties or readability penalties. In languages like c++, using a pointer isnt going to confuse anyone. So really I prefer those languages, but it is cool to learn the intricacies of these languages. In particular because I use it for a my job

    • @minciNashu
      @minciNashu 2 роки тому +1

      C++ also introduced similar concepts as standard, and before that there were 3rd party libraries providing some kind of span. See std::string_view and std::span

    • @KayOScode
      @KayOScode 2 роки тому

      @@minciNashu arent those features kind of just bloat though. We have void* and thats all we really need

    • @mikicerise6250
      @mikicerise6250 2 роки тому +6

      My first thought was, I mean, great, but why do I get the impression that C# has a lot of tricks for solving problems caused by C# in the first place? ;) Why not just have a span method on the string that returns a readonly reference to a section of the string?

  • @chrisd961
    @chrisd961 2 роки тому +82

    I'm so impressed by these explanations. Honestly, the best thing to happen for a junior dev, is to find your channel. Great explanations, very helpful videos with in-depth knowledge and analysis. Thank you, please keep doing them!!

  • @sasukesarutobi3862
    @sasukesarutobi3862 2 роки тому +62

    Span is a really under-rated feature, not just for performance, but also my favourite pun in C# - TimeSpan

    • @RichardNobel
      @RichardNobel 2 роки тому +8

      Spantastic pun 😉

    • @RichardNobel
      @RichardNobel 2 роки тому +11

      If you have a dog... it's probably a Spaniel ? 🐕

    • @jeffwilson8246
      @jeffwilson8246 2 роки тому

      Lately the news has been so dull I've tuned into Cspan

    • @tuck1726
      @tuck1726 2 роки тому +1

      @@jeffwilson8246 This whole thread makes as much sense as watching C while only wearing a .
      I don't get the OP. I understand span and TimeSpan, but I don't see the pun. I am the dummy cause 40+ people saw it. Guess I need to get out of my static internal scope of thought.

  • @evolvedant
    @evolvedant 2 роки тому +48

    When I first watched Microsoft themselves explain Span, I was lost and confused. They have a knack for making something sound way more convoluted and complex than needed when they explain new concepts. This video made it all click instantly. Thank you very much, I can't wait to start using Span in my own work.

    • @Faygris
      @Faygris Рік тому +6

      The same is true for their documentation

    • @mrx10001
      @mrx10001 Рік тому +6

      ye microsoft love overcomplicating every single one of their examples.....they need to hire people to teach them to keep things simple. Especially for their documentation.

  • @Manlyman789
    @Manlyman789 2 роки тому +23

    I love these types of videos that explain the standard classes in the framework that help you write more efficent code. These are the types of things I don't stumble upon when researching how to solve a problem. I would love to see more videos like these!

  • @jackkendall6420
    @jackkendall6420 2 роки тому +5

    Great video. This is a topic I've been half-aware of for a while, but seeing it in context helps a lot.

  • @RichardNobel
    @RichardNobel 2 роки тому +7

    Great explanation, Nick, yet again! Thank you for taking the time, in your very informative videos, to show us what's happening "behind the scenes". On the heap, stack, etc. 🙏🏻

  • @AndreMauricio4
    @AndreMauricio4 2 роки тому +1

    hey Nick, I have been watching your videos for a while. Just want to thank you so much for the learnings I got from them. You cannot imagine how helpful you are to people like me. I am using these learnings in a software solution I am developing myself already for a year. Again, thanks!!!
    Andre from Portugal

  • @DepressionAlgorithm
    @DepressionAlgorithm 2 роки тому +28

    Didn't learn anything new in this, but I'm impressed by your presentation. I would have more quickly learned how Span works if this was the first video I saw about it.

  • @snuffsix9598
    @snuffsix9598 2 роки тому

    Your video's are awesome. I very much appreciate the technical in-depth explanations of them. Thanks a ton!

  • @JoeEnos
    @JoeEnos 2 роки тому +1

    As usual, you explain the nuts and bolts, the theory, and the benefits, all better than the documentation and anything else I’ve found.

  • @ristopaasivirta9770
    @ristopaasivirta9770 Рік тому +11

    Thank you for this video.
    As Unity 2021 LTS now supports spans this gave a really good introduction and explanation on how to use them.
    Also the ref struct is valuable information since I have some very short lived structs for triangles and other mesh-related objects that only live for the duration of the method.

  • @binjozoken6055
    @binjozoken6055 2 роки тому +2

    To me, these are the best technical videos on the net, even though about 95 percent of them are over my head. This video, however, was worthy of getting a bowl of popcorn, sitting back and just watching. Thanks a bunch, Nick.

  • @Deathhead68
    @Deathhead68 2 роки тому +2

    Had missed the memo on this one. Thanks, this was really informative!

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

    this is the first explanation of Span that made sense to me.
    thank you!!

  • @Sgro81
    @Sgro81 2 роки тому

    Nothing new under the sun but very well explained. It's useful to have this kind of videos around, proper knowledge should be distributed like this.

  • @Pouya..
    @Pouya.. Рік тому +1

    I'm a senior developer with 10+ years of experience and I learn so much from your videos thanks in advance

  • @pedrocunha4322
    @pedrocunha4322 Рік тому

    I really appreciate your content Nick, helped me a lot improve the way i code :)

  • @Sad-Lemon
    @Sad-Lemon 2 роки тому

    Very nice explanation. Will surely be helpful in my work. Thanks!

  • @coffee22able
    @coffee22able 2 роки тому

    Great Video, thanks for the board explanation, it was awesome!

  • @arlvinmoyo9290
    @arlvinmoyo9290 Рік тому

    Amazing delivery. Thank you!

  • @davemasters
    @davemasters 2 роки тому

    Brilliantly explained, as per usual.

  • @jonowilliams26
    @jonowilliams26 2 роки тому

    Quality content as always !

  • @PontusWittenmark
    @PontusWittenmark 2 роки тому

    Great explanation, Nick 👍

  • @my_temporary_name
    @my_temporary_name 2 роки тому

    Great presentation, thank you!

  • @tzurdo1
    @tzurdo1 2 роки тому +1

    Another great video Nick!
    A small question - is there a way to split Span into array\list of Spans?
    Or actually, what is the best way to do it, without iterating by myself over the Span?

  • @KeithSwanger
    @KeithSwanger 2 роки тому

    Nice explanation. Thank you!

  • @KeesSchollaart
    @KeesSchollaart 2 роки тому +8

    Curious to learn how it deals with byte arrays, compared to working with strings like you demo'd.

  • @misomalu
    @misomalu 2 роки тому

    Excellent video and presentation.

  • @Cassiopeja22
    @Cassiopeja22 2 роки тому

    Excellent explanation, Nick! Thank you very much. :)

  • @user-ss9qg6ke5t
    @user-ss9qg6ke5t 2 роки тому

    You are amazing! Thanks for your explanations

  • @AB-fb1ve
    @AB-fb1ve 2 роки тому

    Span is a powerful structure but has some limitation, what about Memory and what is the difference between them

  • @hirenpatel2236
    @hirenpatel2236 2 роки тому

    Nicely explained!

  • @andreaskarz
    @andreaskarz 2 роки тому

    Wow, I never see before but I will us it in the next projects -- thank you.

  • @oshastitko
    @oshastitko Місяць тому

    Very cool explanation! Thank you!

  • @rainezombi3431
    @rainezombi3431 Рік тому

    Your short brought me here. Good stuff!

  • @milanmladenovic
    @milanmladenovic 2 роки тому

    Hi Nick, great video as usual :)

  • @stuartbooth8232
    @stuartbooth8232 2 роки тому

    Great vid easy to follow thankyou

  • @rifatislamrakesh
    @rifatislamrakesh Рік тому

    Marvelous ! To the point. Respect !

  • @im1in260m
    @im1in260m Рік тому

    Thank you Nick. Your videos are easy to understand, neat and to the point. I looked on your web site at the courses and wanted to know if the Dependency injection was based on a third party app. I was unable to locate a way to contact you there.

  • @xinzhouping
    @xinzhouping 2 роки тому +4

    you should make a follow up video on ref structs if you haven't already.

  • @amirdar
    @amirdar Рік тому

    Great video as always!
    in 2:26 how do you get to that debugger window with the memory tab ?

  • @inxaneofficial7756
    @inxaneofficial7756 Місяць тому

    Good video, very insightful !

  • @HomeSlize
    @HomeSlize 2 роки тому

    good stuff! thanks for sharing.

  • @gbelkin4
    @gbelkin4 2 роки тому

    Brilliant as usally, Nick :-)

  • @tehsimo
    @tehsimo 2 роки тому

    That memory view in rider is awesome

  • @marksmod
    @marksmod Рік тому

    Isn't the ReadonlySpan allocating at least a copy of the string on the heap?
    For example, consider:
    string s = "123";
    ReadOnlySpan span = s;
    Console.WriteLine(int.Parse(span.Slice(0,2)));
    s = "321";
    Console.WriteLine(int.Parse(span.Slice(0,2)));
    produces:
    12
    12
    Oh, since strings are immutable in this language, the second time we assign to »s« we actually perform a second heap allocation, and »span« can happily use the address to which »s« pointed to when »span« was defined. And some smart-pointer-like stuff.

  • @Cruz0e
    @Cruz0e 2 роки тому

    if I understand this well, when you use the span, like:
    Readonly dateAsSpan = _dateAsText;
    you make a new variable called "dateAsSpan" but it's not allocating new memory for the "value" itself (in the heap), but instead just basically stores a reference, similarly when in c (normal C, not ++ or sharp) if I had a function like this
    void double_it(int *j) { j *=2 }
    using "int *j" instead of "int j" (* means that you pass by memory address reference instead of value)

  • @JtendraShahani
    @JtendraShahani Рік тому

    Hi. Thank you for the tutorials. I have learned a lot. I wanted to know how do you get the results inline. Thank you.

  • @tianjinghan1
    @tianjinghan1 Рік тому

    That is a great video, thanks, Nick. May I ask what IDE you are using?

  • @paultaylor2054
    @paultaylor2054 Рік тому

    thank you so much

  • @TheCMajor9th
    @TheCMajor9th 2 роки тому

    great example m8 ty for the presentation

  • @bomite
    @bomite 2 роки тому

    Nice, thanks for that!!!

  • @RichardNobel
    @RichardNobel 2 роки тому +7

    Should we refer to the _Span_ Slice method's start _index_ as... *"Spandex"* ? 😜
    When a (male) programmer is very good at using Span... is he "SpanKing" ? 👑
    (Or even Span heh). 🤴🏻

  • @easycodeunity3d14
    @easycodeunity3d14 2 роки тому

    Thank you very much

  • @minciNashu
    @minciNashu 2 роки тому +2

    When you talk about allocation, it's important to stress that span doesn't really copy the source memory on the stack. The span object itself - containing probably a starting pointer and a max size - is created on the stack.

    • @Maxi-xw1jb
      @Maxi-xw1jb Рік тому

      Very true👆He should explaine it, otherwise it's misleading

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

    The Span.ToString() method would allocate heap memory because strings are immutable, right?

  • @bovineox1111
    @bovineox1111 Рік тому

    Great video. I believe application only stops (completely) for GC when using workstation GC as opposed to server GC - you should compare the differences as they are quite striking and too long to go into here.

  • @rossthemusicandguitarteacher
    @rossthemusicandguitarteacher 2 роки тому

    This is great.

  • @mehdihadeli
    @mehdihadeli 2 роки тому

    Hi, Thanks for your great video,
    Please record a view about diagnostics and tracing in .NET 5.

  • @user-py9cy1sy9u
    @user-py9cy1sy9u 2 роки тому

    Its nice to see that D is benefiting C#

  • @eperez_yt
    @eperez_yt 2 роки тому

    Nice! I learned a new C# feature. Span look like pointers, so I'd have liked to know what would have happened if dateAsText had changed, and make sure if span really works like pointers. (:
    BTW i appreciate your video.

  • @chiragdarji1571
    @chiragdarji1571 Рік тому

    Phenomenal!!!

  • @scabendlin
    @scabendlin Рік тому

    wohoaaaa!!! now i understand a bit more the use span and garbage collectors....thanks nick!!

  • @tobiasj8019
    @tobiasj8019 Рік тому

    Very nice and helpfully 🎉

  • @SuperSpeed52
    @SuperSpeed52 2 роки тому

    This is excellent for string manipulations

  • @MarioRamosMontesinos
    @MarioRamosMontesinos 2 роки тому

    Nice content!

  • @m0ment219
    @m0ment219 2 роки тому

    This is... I can't believe how many times this could've helped me...

  • @yonatankarni8867
    @yonatankarni8867 2 роки тому

    In comparison to C- span sound like a pair of pointers, and slice moves one of them.
    I hope I got the idea of this feature thanks!

  • @digitalpacman
    @digitalpacman 2 роки тому +3

    You should have maybe ran the test with a changing date every time. Strings in c# are instantiated once per instance and re-used. So "foo" in variable a and "foo" in variable b are both the same "foo" in memory. This likely will show a more realistic real world use case.

  • @10199able
    @10199able 2 роки тому +2

    So this is how you use memory tab in Rider!

  • @relaxfunplaynice
    @relaxfunplaynice 2 роки тому

    Is Span.Slice.ToArray() slightly faster than Buffer.BlockCopy or byte array copy using unsafe methods? (seems to be yes)

  • @Aaron31056
    @Aaron31056 Рік тому

    At 10:40, wouldnt it be an offset of 2 instead of 3? Im guessing that its a 0 based offset, so the first value in the string would be index 0 and the third would be 2, so if we want to start reading the month which starts at index 2 the offset would then have to be 2 right? (+2 offset and 2 length).

  • @harag9
    @harag9 2 роки тому +2

    Great vid Nick, and there's me still working in c#7, .NET 4.8 & WinForms... I'm so far behind these days, but good to see new features in c#. I just wish the company I worked for didn't work on 15-20 year old projects...

    • @aminejadid2702
      @aminejadid2702 2 роки тому +2

      don't worry, it's the same for me. Working with .Net framework 4.6.1 ! But i never stop learning new technologies. It is the only way

    • @harag9
      @harag9 2 роки тому +1

      @@aminejadid2702 Same here, but I'm now losing interest doing it in my spare time, so unless I do it at work I lose the new skill.

    • @RealisableSoftware
      @RealisableSoftware 2 роки тому

      Add the System.Memory Nuget package. It's not everything that you get with core, but you get some benefit.

    • @aminejadid2702
      @aminejadid2702 2 роки тому

      @@harag9 You can always find a better job.

    • @horowitzhill6480
      @horowitzhill6480 2 роки тому +1

      .net 4.0 for me at work 🤣

  • @bhavinmistry6262
    @bhavinmistry6262 2 роки тому

    The best👌👌

  • @mehmetedex
    @mehmetedex 2 роки тому

    this is brillant

  • @Dawhun
    @Dawhun 2 роки тому +1

    Hello, great video as usual :)
    Do the .ToString() really needed in the YearAsText() method return ? (end part of the video)

    • @nickchapsas
      @nickchapsas  2 роки тому +1

      It did because at that point it is a ReadOnlySpan not a string and the Console.WriteLine method doesn't have an overload for it.

    • @TylerEich
      @TylerEich 2 роки тому +1

      @@nickchapsas I was wondering about YearAsText, line 21. Would the `ToString()` there still allocate on the heap and then implicitly converts the string to the ReadOnlySpan? I would’ve expected `return yearAsText` without the `ToString()`

    • @metaltyphoon
      @metaltyphoon 2 роки тому

      @@TylerEich No you don’t need to do ToString(). What you would be returning is a readonly ref struct to the caller and in there you can do the ToString()

    • @nickchapsas
      @nickchapsas  2 роки тому

      @@TylerEich Sorry, Yeah not that you pointed out the line I understand what you mean. No that's a mistake. I added a pinned comment to explain that. I missed it because of ReadOnlySpan's implicit operator.

  • @CodingGustavo
    @CodingGustavo 2 роки тому +2

    Does int.Parse accept a Span? or theres a implicit conversion from Span back to string?

    • @nickchapsas
      @nickchapsas  2 роки тому +2

      There is an overload with Span yeah

  • @easycodeunity3d14
    @easycodeunity3d14 2 роки тому

    Awesome!

  • @MultiMinors
    @MultiMinors 2 роки тому +4

    Span? I only know Div

  • @neriacohen5490
    @neriacohen5490 Рік тому

    Amazing

  • @OvRaf
    @OvRaf 10 місяців тому

    perfect

  • @grainfrizz
    @grainfrizz 2 роки тому

    Very nice

  • @VasoPerasLikodric
    @VasoPerasLikodric 2 роки тому +1

    I guess that Parse method have span override. But what is happening if we need to use method which will except string.
    I guess it will have implicit conversion there.

    • @nickchapsas
      @nickchapsas  2 роки тому

      If the method needs a string then the return string will be allocated but you can prevent any potential allocation during the mid-way processing in the method, depending on the workload. Also yeah, int.Parse has a span overload, including many other things that used to accept string.

  • @Guciubla
    @Guciubla 2 роки тому

    Great content!
    how does span compare to string builder in such scenarios?

    • @nickchapsas
      @nickchapsas  2 роки тому +3

      string builder is used to create a string without having to deal with the immutability concerns. Span, even tho it could technically be used for something similar, primarily does the opposite.

  • @SealedKiller
    @SealedKiller 2 роки тому +7

    So it's like a StringView, providing a view into the string pretty much.

    • @igorthelight
      @igorthelight 2 роки тому +2

      True
      For those who are not aware - It's std::string_view (after you #include ) in C++ 17

  • @Radictor44
    @Radictor44 Рік тому +1

    Very good video - has helped to clear up my understanding of Span :)

  • @igorthelight
    @igorthelight 2 роки тому +1

    6:15 - You could access String as an array too!
    But if you would try to change even just one char like that:
    string myString = "Hello!";
    myString [0] = 'h'; // This will not be compiled!
    It still would allocate a whole new string. StringBuilder do not behave like that tho so you could do that:
    var sb = new StringBuilder("Hello!");
    sb[0] = 'h';

    • @Dennis19901
      @Dennis19901 2 роки тому

      The indexer of a string in C# is get only.
      What you wrote is invalid code.

    • @igorthelight
      @igorthelight 2 роки тому

      @@Dennis19901 Thanks! I pointed out that this line can't be compiled.

  • @ivailomanolov6908
    @ivailomanolov6908 2 роки тому

    Good video

  • @Ayomikun
    @Ayomikun 2 роки тому

    Great video as always. Is this only ever useful for strings? Thanks

    • @nickchapsas
      @nickchapsas  2 роки тому +1

      A span resembles an array so it can work as a byte array, int array and so on

  • @rajm1976
    @rajm1976 2 роки тому +1

    In order to make use of the Span 'value' you still need to convert it ToString(), which as you say loses the value in Span, unless the resulting string is a concatination of a bunch of Span.Slice functions. So you could take the fact that Span is basically an Array and then join them together to produce the final result.

    • @nickchapsas
      @nickchapsas  2 роки тому

      As you can see in the pinned comment the ToString() in the method that returns a ReadOnlySpan was a mistake. You don't need it and if you don't use it you don't allocate until the final ToString()

  • @ivandrofly
    @ivandrofly 2 роки тому

    good stuff ;)

  • @AaronAsherRandall
    @AaronAsherRandall 2 роки тому

    Does anyone know the implications this has with C# and Unity and/or if we can use this to our advantage with game optimizations?

  • @EdKolis
    @EdKolis 3 місяці тому

    In your last example you changed the return type but forgot to remove the ToString call from the method, negating the performance benefit. But thanks for explaining Span (and ref struct) in an understandable way; neither of those ever made sense to me until now!

  • @dawidzyrek6481
    @dawidzyrek6481 2 роки тому +26

    **Everyone** Nice video!
    **Me** Wait... A method can return multiple values?!

    • @KoScosss
      @KoScosss 2 роки тому +7

      Tuples!

    • @petrusion2827
      @petrusion2827 2 роки тому +6

      Even before tuples you could use *out* or *ref* parameters to return multiple valuess without the need of making a new type every time. I miss both of those greatly when I have to work in java (among lots of other things).

    • @user-pu4qu9my5j
      @user-pu4qu9my5j 2 роки тому

      Sure. You also can return Monad like F# Descriminated Unions. In nutshell this is the simple value containers, but technically it incapsulated multiple values indeed. Tuples is the famous example of monads.

    • @rajm1976
      @rajm1976 2 роки тому +1

      Yeah but don't. Unless they are related. Like co-ordinates or dates (you would use DateTime instead imo)

    • @rajm1976
      @rajm1976 2 роки тому +1

      @@Qrzychu92 If want to return two unrelated things, they belong in seperate methods entirely. If it was co-ordinates or something like that, then Tuple would be fine.

  • @GuidoSalami1
    @GuidoSalami1 Рік тому

    at 00:03:14 ,if it slows things down, can you ask garbage collection not to free up the garbage until the entire program is done running? i mean, thats what memory is for right? let the entire program finish before doing any sort of expensive garbage collection?

  • @DynotoeTube
    @DynotoeTube 2 роки тому +1

    So I wonder if there is a similar use of SPAN for a more common task of parsing CSV comma delimited strings?

  • @marcomarek7734
    @marcomarek7734 Рік тому

    4:18 benchy was such a cute name 😍😂 didn't see that coming

  • @cyberherbalist
    @cyberherbalist 2 роки тому

    Does the StringBuilder class (which I have been told is more efficient than simply using String) use Span behind the scenes?

    • @nickchapsas
      @nickchapsas  2 роки тому +2

      Some of the StringBuilder implementation has been updated to use that behind the scenes where possible