Write Faster SQL Queries With Dapper In .NET | Clean Architecture

แชร์
ฝัง
  • เผยแพร่เมื่อ 1 ม.ค. 2025

ความคิดเห็น • 110

  • @MilanJovanovicTech
    @MilanJovanovicTech  2 ปีที่แล้ว

    Want to master Clean Architecture? Go here: bit.ly/3PupkOJ
    Want to unlock Modular Monoliths? Go here: bit.ly/3SXlzSt

  • @ashwanibhardwaj8095
    @ashwanibhardwaj8095 2 ปีที่แล้ว +1

    I am happy that you started this. Actually we need a combination of efcore with dapper contrib and insert data there should be condition for class like write(false) to protect data insert exception for nesting data

  • @Real-Hindu-Us88
    @Real-Hindu-Us88 ปีที่แล้ว +1

    Thanks... Here we are fetching only records. If we try to get more 50-100 records at time then it will more clarity... Which is Faster.

  • @Cornet435
    @Cornet435 ปีที่แล้ว +2

    Hi, I have a quick question. Do I really need DTO models in my clean CQRS application? I mean if I use Command with properties to create and records response to return data is there anywhere place for typical DTO models?

    • @MilanJovanovicTech
      @MilanJovanovicTech  ปีที่แล้ว +2

      If you are fine with exposing your command to the API, you don't need a request DTO

  • @daniildukhovny5238
    @daniildukhovny5238 2 ปีที่แล้ว +2

    SqlKata is a good alternative as well (actually, it uses dapper under the hood to build queries)

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว

      So is the overhead worth it?

    • @fko079
      @fko079 ปีที่แล้ว

      @@MilanJovanovicTech With SqlKata you can make queries which are independent of database provider. In some cases it may be benefitial.

  • @tony-ma
    @tony-ma ปีที่แล้ว +1

    You didn't show the dependency injection for the SQL connection factory, do you inject as singleton, scoped or transient?

    • @MilanJovanovicTech
      @MilanJovanovicTech  ปีที่แล้ว +1

      Could be transient/singleton - since it always returns a new SqlConnection

  • @jcyepes73
    @jcyepes73 4 หลายเดือนก่อน +1

    Very nice video.
    What theme do you use for VS 2022?

  • @amrelsher4746
    @amrelsher4746 ปีที่แล้ว +1

    What is better milan to return in handler result or throw exception if something wrong ?

    • @MilanJovanovicTech
      @MilanJovanovicTech  ปีที่แล้ว

      Both are okay, if you understand the pros and cons. So just stick to one of them.

  • @ReneMoergeli
    @ReneMoergeli 2 ปีที่แล้ว +1

    Wondering the performance difference between Microsoft.Data.Sqlite and System.Data.SQLite.Core

  •  2 ปีที่แล้ว +1

    I usually use the linq2db library instead of dapper. That's pretty good on performance, and less abstract than efcore.

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว +1

      I actually never worked with linq2db, quite interesting

    • @Corsair4Fun
      @Corsair4Fun 2 ปีที่แล้ว +1

      Yes indeed. I had a chance to work with it while doing a project based on nopCommerce. They used to go with EF but switched to linq2db recently. I didn’t play too much with it but it looks promising.

  • @FGomesFabio
    @FGomesFabio ปีที่แล้ว +1

    good study. how could take advantage of this professional architecture with oracle using sql queries with transaction and commit correctly. as well as a clean architecture project in .Net 7.0

  • @krccmsitp2884
    @krccmsitp2884 2 ปีที่แล้ว +1

    Out of curiousty, did you explain your DomainErrors calls in any of your videos? I'd like to rewatch it.

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว +1

      Check out this:
      th-cam.com/video/KgfzM0QWHrQ/w-d-xo.html

    • @krccmsitp2884
      @krccmsitp2884 2 ปีที่แล้ว

      @@MilanJovanovicTech Thanks mate!

  • @vinr
    @vinr ปีที่แล้ว +1

    Why didn't you implement another version of Repository instead of putting all SQL queries and connections into the handler?

    • @MilanJovanovicTech
      @MilanJovanovicTech  ปีที่แล้ว

      Do you think it's wrong to use SQL in handlers?

    • @vinr
      @vinr ปีที่แล้ว

      @@MilanJovanovicTech It's not about using SQL in handlers, purpose of having repository is to have your data access logic inside them, so that underlying data store is changed or data access logic is changed you still can introduce new repository implementation while keeping the repository interface as it is and its dependency. So in this case what you could have done is to introduce new repository implementation with Dapper

  • @ransandu
    @ransandu 2 ปีที่แล้ว +1

    Thanks for compiling this video.
    How does the abstraction you suggested would look like in the infrastructure? would be a repository like or service abstraction?
    I hope that abstraction would help us solving the unit test challenges of the handler?

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว +1

      Either is fine, and yes, it will help with unit testing for sure

  • @onedev7316
    @onedev7316 2 ปีที่แล้ว +1

    Hi Milan, nice walk through. I am getting an issue of "The ConnectionString property has not been initialized." Am i missing something?

  • @mohamed-hassan-
    @mohamed-hassan- 7 หลายเดือนก่อน

    I've used multiple times, thanks to you.. keep the great work lad💪🏻

  • @ethan_hunt
    @ethan_hunt 2 ปีที่แล้ว +1

    what is cancelation token it is all over the places ?

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว

      Used for propagating request cancellation down the call chain

  • @KingOfBlades27
    @KingOfBlades27 7 หลายเดือนก่อน

    Might be a dumb question but why are you making a separate connection factory class? Could we just create a new connection normally in the method? I am probably missing something obvious here.

    • @MilanJovanovicTech
      @MilanJovanovicTech  7 หลายเดือนก่อน +1

      Of course. I mainly create these abstractions for testability. Sometimes I may want to do additional things with the connection, so it makes sense to have it one place.

    • @KingOfBlades27
      @KingOfBlades27 7 หลายเดือนก่อน

      @@MilanJovanovicTech Thanks for the response 💪

  • @mahanog9986
    @mahanog9986 2 หลายเดือนก่อน

    hello milan, why dont we Use dapper in the infrastructure layer ?? so we dont violate the rule that "The application layer should not have access to external resources" ??

  • @bahtiyarozdere9303
    @bahtiyarozdere9303 2 ปีที่แล้ว +2

    Thanks for the content. Good demonstration but the conclusion is misleading.
    What you see 20ms response time in swagger UI includes routing, model binding, serialization etc.
    If the aim is to compare EF to Dapper on speed, using benchmark with multiple types of commands and queries would be a better approach. By querying only one object with first result, there will be no big difference especially if you eyeball the result based on api response time.

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว

      This is not a scientific conclusion by any means. 😁
      But I'm more interested in the full flow that includes the API than just raw data access.

  • @i2um1
    @i2um1 2 ปีที่แล้ว +1

    It would be better to use DbDataSource (or DbProviderFactory at least) instead of SqlConnectionFactory. Btw, Dapper doesn't support positional parameters and DbBatch that are useful for Npgsql.

  • @emwagner
    @emwagner 2 ปีที่แล้ว +1

    If you are manually specifying/writing SQL statements to query the database using Dapper, do you have to worry about SQL Injection issues?

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว +2

      No, Dapper takes care of that as long as you use parameterized queries

  • @crazyfxy
    @crazyfxy ปีที่แล้ว +1

    Kind of weird comparison. First you are not showing the EF Code, so there could be all kind of stuff behind that repository. Second Dapper does not have change tracking and EF does (or did you use AsNoTracking, again we can't see).

  • @whitek6532
    @whitek6532 2 ปีที่แล้ว +1

    what if using ef core raw sql with asnotracking ? Dapper still faster ?

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว

      I didn't measure, but most likely Dapper is slightly faster or they're the same

  • @MrBodyPower
    @MrBodyPower 2 ปีที่แล้ว +1

    How should i register the implementation of ISqlConnectionFactory as a scoped or transient Lifetime?

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว

      Transient or Singleton is perfectly fine.

    • @MrBodyPower
      @MrBodyPower 2 ปีที่แล้ว

      @@MilanJovanovicTech So, can i register that implementation as singleton in my IHost worker which use background service that is processing a file and then save in database?

    • @etechguide7251
      @etechguide7251 2 ปีที่แล้ว +1

      @@MilanJovanovicTech Why Singleton? If there are multiple concurrent requests to API endpoint then how this Singleton connection will handle all those?
      So for Singleton this Sql connection will remain open forever till application closes?

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว

      @@etechguide7251 Well, no... Notice that the SqlConnectionFactory just creates a new connection and returns it. It's up to the consumer to dispose of it.

  • @cbo4277
    @cbo4277 2 ปีที่แล้ว

    Task which package you use for the "Result" keyword, or is it a model in Domain or another layer

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว

      I made my own, but you can take a look at FluentResults or something like that

    • @grumpydeveloper69
      @grumpydeveloper69 2 ปีที่แล้ว

      Like the ErrorOr library by @amantinband😇

  • @sanju-p6k7b
    @sanju-p6k7b 13 วันที่ผ่านมา

    I want to fetch 200K records from table of 50 million records with 50-60 columns in table. My SQL statement looks like SELECT * from table where Id in (1,2) and name in ('a', 'b').
    Currently I am using EFCore for all DB operations in my application. Out of EFCore, Dapper which one is more faster for the above scenario ?

    • @MilanJovanovicTech
      @MilanJovanovicTech  13 วันที่ผ่านมา

      IN clause probably not using the index. Check the Execution Plan and figure out how to optimize the index.

  • @NikolaGolijanin-m8s
    @NikolaGolijanin-m8s 9 หลายเดือนก่อน

    Hi Milan, i have a question about parameters. What is the best way to work with nullable params?
    For example if i want to be able to filter table by username and email, but in case that email is null i want to ignore it and filter it only by username.
    I know its possible to use sql OR but does dapper has some functionality to handle that kind of case?

    • @MilanJovanovicTech
      @MilanJovanovicTech  9 หลายเดือนก่อน +1

      Just use some sort of query builder

  • @garethcraig8902
    @garethcraig8902 ปีที่แล้ว

    Why no sprocs. Imbeded sql is hard to debug in production

  • @codewithguillaume
    @codewithguillaume 2 ปีที่แล้ว +1

    Interesting

  • @techpc5453
    @techpc5453 ปีที่แล้ว +1

  • @PersleyGrande-ek7hl
    @PersleyGrande-ek7hl ปีที่แล้ว +1

    Looking pretty great

  • @joshem32
    @joshem32 2 ปีที่แล้ว +1

    Awesome content as always!!
    Just wonder if is there any way to write prepared statements like in Java?

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว

      I think Dapper does something similar to that internally, with the SQL I specified in the example

  • @volodymyrliashenko1024
    @volodymyrliashenko1024 2 ปีที่แล้ว +1

    Why nobody compare performance of Raw SQL in EF with Dapper?
    That would be a honest comparison.

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว

      Because it's not how you would use EF normally, right?

    • @volodymyrliashenko1024
      @volodymyrliashenko1024 2 ปีที่แล้ว

      @@MilanJovanovicTech yes, but if you just need to run a raw SQL - is required to install one more dependency?
      I hope I will find some time to do different comparisons.
      I like Dapper but I don't have an answer to that question.

  • @ibrahimgirisken8945
    @ibrahimgirisken8945 ปีที่แล้ว

    Hello, I am creating a project using dapper with onion architecture. Since I was using Dapper, I could not adapt the identity and jwt token packages to the project. Can you help with this?

  • @zikkrype
    @zikkrype 2 ปีที่แล้ว +1

    I only now noticed that you have World of Warcraft. I used to play Burning Crusade in year 2010. I've hit 2200 rate points on arena 2v2 back there (I was playing mage). It was one of the best moments in my life

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว

      I was a kid when original Wrath released, just entered high school. Enjoying Wrath classic very much right now. 😁
      I was never really much into PvP but that is an impressive rating buddy!

  • @dilipmevada
    @dilipmevada 5 หลายเดือนก่อน

    Can you please share link to download code for this video.

  • @winchester2581
    @winchester2581 2 ปีที่แล้ว +1

    Such a good video, Milan!
    And, thinking of such usage of Dapper, I always have a question in such use cases: can we just put Dapper stuff into Repository or it's bad idea? Because I think that it would be a great idea, but I have strugglings about mixing EF Core and Dapper together in one class

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว +1

      Thanks a lot! :)
      I've done that before - putting Dapper & EF in same repository. However, I mostly used Dapper for UPDATE/DELETE queries in such cases.

    • @winchester2581
      @winchester2581 2 ปีที่แล้ว

      @@MilanJovanovicTech thanks for the reply, it's worthy! Have a great day

    • @adisilagy
      @adisilagy 2 ปีที่แล้ว +1

      Jumping in, are there any reasons not to use queries within the repository?
      One benefit of putting it in a repository is that it can be used by many handles

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว +2

      @@adisilagy On read side, no.
      But if you read to modfy, you won't have change tracking out of the box as you would with EF.

  • @MarcusKaseder
    @MarcusKaseder 2 ปีที่แล้ว +5

    Nice! Didn't use dapper so far. I really like sql statements but I don't like them in code somehow (same with sql like Linq queries)
    But good to see that EF Core 7 can keep up with Dapper 😁.
    BTW... You broke your clean architecture because of the sql and I broke my smartphone because of that heavy smashing. Please staaaahp
    Good job 🙂

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว +1

      I still think it's Clean Architecture regardless! Dogmatic design be damned 😁
      I also dislike SQL in code, but there's no better way, to be honest.
      And no, I won't stop! I will intensify 🔥

    • @MarcusKaseder
      @MarcusKaseder 2 ปีที่แล้ว

      @@MilanJovanovicTech Yeah I'm with you. Clean Architecture is more an orientation to me. I really like it and the separation. But each project has different requirements that can't always fit into that dogmatic approach. Same with DDD, it's not always suitable, but you can still use clean architecture without domain and put everything into your application layer.
      As long as the team is fine with the "violations", I'm too 😁
      I'll send you my broken phone to fix it if you do 😋

    • @pilotboba
      @pilotboba 2 ปีที่แล้ว +1

      I think this is just an example. In real world you would probably use a repository that is implemented and lives in your infra layer which is where the SQL query would live.

    • @MarcusKaseder
      @MarcusKaseder 2 ปีที่แล้ว

      @@pilotboba I absolutely agree with you! But for the sake of the video title, it was okay.
      The bad thing and also what really hurt me is, that @Milan removed the repository from the handler.
      He has already a repository, and also a infrastructure and even a persistence layer. In a real world, he just should've replaced the code within the repository on one of those layers or decorated a new member repository with a "dapper proxy".

    • @GinBilog-ij2rj
      @GinBilog-ij2rj 7 หลายเดือนก่อน

      Did u try calling stored procedure in dapper?

  • @lodevijk
    @lodevijk 3 หลายเดือนก่อน

    I cannot fathom why would someone willingly use something that will crash if the order of properties in the MemberResponse is swapped, without any indication of an error. And it's a runtime error on top.

    • @MilanJovanovicTech
      @MilanJovanovicTech  3 หลายเดือนก่อน

      It it has a parameterless ctor, it should be fine

  • @stefanivovic
    @stefanivovic 2 ปีที่แล้ว +1

    Great video as always :) can you do video about registering and configuring class library in web app ConfigureServices method of Startup. For example we have some class library in which we have api client calling some external api's. and we need to add it to our web app and configure stuff like url, name itc ....

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว

      You mean an HttpClient?

    • @stefanivovic
      @stefanivovic 2 ปีที่แล้ว

      @@MilanJovanovicTech not HttpClient per se. I am talking about pattern for registering and configuring nuget packages. For example we have class library and we wanna use options patter to enable users of our package to configure some stuff, like we do is startup class when we import nuget package. That was my idea, i think it would be really informative

  • @chrisbaker5284
    @chrisbaker5284 ปีที่แล้ว +1

    Another great video. Just out of curiosity, how would you implement the UOW when using Dapper? I'm not a fan of EF in any of it's versions and I'm not convinced that EF hadn't cached the result from the first call in your test (I'm pretty sure SQL server would have at least cached the execuption plan), so I'm not sure of the validity of the test, but I know that EF has improved over the years, I just remember it when it was a pile of horse manure. Anyway, I tend to just use raw ADO as I'm old school, but have quite often struggled with the UOW, which is much easier to achieve in EF, so any thoughts you have on this would be appreciated.

    • @MilanJovanovicTech
      @MilanJovanovicTech  ปีที่แล้ว

      You gotta implement the Identity Map pattern on your own, too much trouble

  • @OleksiiKorniienko
    @OleksiiKorniienko 2 ปีที่แล้ว +1

    Thanks for your video!
    I didn't get why you said Dapper is 'definitely' faster. In this particular scenario, the AVG and Median produced by EF were better and Min was just 3 ms faster, so in your 'performance testing' EF is the definite winner. (Of course, I'm not taking into consideration the cold start of EF query)
    I've just done the same simple comparison using K6 with my pet project, and I can say that I can't see any difference comparing the same query executing with EF and Dapper (or maybe my PC is just too noise due to big amount of background work to show correct result as dapper might be a little bit faster in theory)
    So the conclusion is not very informative in my opinion, cause I think it is not about the performance of the same queries, it's about the 'linq or code first approach' and 'database first approach'.
    IMHO, I believe this performance boost using Dapper is so miserable compared to the features that are provided by EF. Projects that use 'raw SQL with dapper' or 'procedures with dapper' become unsupportable very easily, so I think EF should be picked as a primary tool in the first place and dapper can be used in some corner cases when we need to write something really complex.
    Also, we are all .Net developers and most of us don't know how to write effective SQL queries.

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว +2

      We should do a proper benchmark

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว +2

      And not knowing how to write SQL is not a valid excuse!

    • @rossimac
      @rossimac 2 ปีที่แล้ว

      @@MilanJovanovicTech I'm a big proponent of EF vs Dapper in my teams, whereas most aren't and they favour Dapper. One of the touted reasons is that with EF, you don't need to know SQL and I think that's such a simplification. Yes, you can build quick projects using EF and see quick results, but when getting near a large-scale production database you absolutely need to know SQL to make sure you aren't killing the database while executing any EF stuffs. You need to know SQL for both approaches if you want to go further than entry-level stuff, basically.
      One thing I have noticed is that intermediate and senior developers in my company do not know EF past the simple stuff and they don't see the value in learning it when they can just use something like Dapper paired with SQL knowledge. Which I think is a shame. The age-old "Dapper is faster" is definitely not as valid as it once was and I much prefer the abstractions that EF gives.

  • @parthorn5985
    @parthorn5985 2 ปีที่แล้ว +1

    One big benefit of using dapper is you can access internal sql functions that is not accessible when using EF.
    I think a combination is good where easy non critical db operations can use EF while on the more performance critical db operations you can opt in for dapper.
    Good video 👍⭐️

    • @MilanJovanovicTech
      @MilanJovanovicTech  2 ปีที่แล้ว

      I agree, I also like to combine them!

    • @volodymyrliashenko1024
      @volodymyrliashenko1024 2 ปีที่แล้ว +1

      Not sure what do you mean internal, but you can map even custom function in EF.

    • @victorchisomnwisu9776
      @victorchisomnwisu9776 2 ปีที่แล้ว +1

      You can access internal functions in sql by creating a static method in the db context and decorating it with [DbFunction(Name="")].
      You use it in your ef core query function when you need it