How To Write Better Functions In Python

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

КОМЕНТАРІ • 90

  • @Titouan_Jaussan
    @Titouan_Jaussan 4 місяці тому +112

    I was wondering, Could you make some sort of code review type videos ? I know it probably takes more time but at the same time you could also explain a lot of other stuff, like how to handle big sized projects, how to make code more professional on a larger scale ... This could really be cool I think

  • @JeffreyLiss0
    @JeffreyLiss0 4 місяці тому +57

    Eschewing long function names simply makes naming much harder. You end up having to make important distinctions with fewer words, so grammar comes into play. This function, for example, should be named get_discounted_total or get_discounted_price, because that's what it's returning -- not the sum of all discounts to be applied to a product. Playing grammar nazi with code is nobody's idea of a good time, and it's particularly difficult on teams where people have different native languages. Better to make your peace with longer names, especially since any IDE worth the name will offer to autocomplete it for you.

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

      I agree. With modern IDEs there isn't a lot of drawback for long function names.

    • @Just.Be.Kind.please
      @Just.Be.Kind.please 3 місяці тому +1

      While I’m opining from a position of extreme novice in coding, I have absolutely noticed myself using more detail/length in my function names when needed. This helps me and my neurodivergence keep things understandable when going back to review or continue working on a project.
      I imagine it would be just as helpful, if not more, for a group/team.

  • @dingaroo2003
    @dingaroo2003 Місяць тому +1

    Been programming with Python for 6 years, but after going through your video, I am hoping to elevate my Python skills!

  • @drew.esbssy
    @drew.esbssy 4 місяці тому +24

    I like how I came to watch the video thinking it would just be normal stuff i would already know and ended up learning how to create function documentations

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

      i was disappointed the way i expected to, i knew all of this :c

  • @Marc-ElianBegin
    @Marc-ElianBegin 2 місяці тому +1

    Thanks for the video. There are several standards for documentation. Making a review of those with its pros and cons would be nice.

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

    3:23 this is commonly referred to as Dependency Injection (DI)

  • @frozendude707
    @frozendude707 2 місяці тому +1

    7:23 I'm gonna do a sad frikkin "errm, actually" here, so technically it should drop an exception or segfault if the locale is missing or corrupt, in which case you could return a generic time format like %H:%M:%S (there seems to be no built-in support for standard ISO 8601 formats).

  • @marcof1430
    @marcof1430 4 місяці тому +5

    I think documentation has a double side: useful but terrible if it's not updated. And it's easy to forget to update, even in the video the documentation needed to be updated. I prefer comments in the hard parts so I understand what the programmer wanted to achieve or do..

    • @nuynobi
      @nuynobi 3 місяці тому +2

      Valid point about documentation needing to be maintained. That's an argument for keeping docstrings simple and concise (unlike in this video), _not_ and argument for not using them at all. Comments and documentation have different purposes. Both are useful and should be employed where appropriate. The convention is that docstrings explain the _what_ while comments explain the _how._

  • @amitshogun
    @amitshogun 4 місяці тому +5

    This function has two responsibilities. Sum and percentage discount. Better to split / extract to 2 simple generic functions and create another one that combines both. Better code reuse.
    And what about validating percentage value?
    Thank you

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

      there is already a generic function for sum: the sum function. It is also faster, because it's implemented in C. The function he wrote already combines both functionalities. There is no need to write a separate one liner function to apply a percentage, it is literally just a multiplication

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

      ​@@ziul123 True, but his point still stands. Why pass in a list of prices when you could simply pass in the total already summed? And it should be called something like apply_discount(price, rate).

  • @DuncanBooth
    @DuncanBooth 4 місяці тому +11

    I don't like it when people duplicate the parameter types in the docstring. You have the type clearly specified in the function signature so no need to repeat it. (And if you're using tools to process the docstring that don't use the function signature types get better tools.)

    • @nuynobi
      @nuynobi 3 місяці тому +1

      Amen. It's redundant. And then you need to remember to update it in both places if something changes. DRY.

  • @Greenbay-bn3yk
    @Greenbay-bn3yk 4 місяці тому +2

    I learnt a lot of Python techniques from this channel. Thanks Mr. Indently 🙂

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

    I don't like get methods in Python but I like everything else in the video

    • @MrMaxtng
      @MrMaxtng 4 місяці тому +1

      Its a convention, not a real getter as per Java or similar. You could call a function that calculates coefficients for a neural network get_coefficients. Why do you care?

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

    Good tips for any programming language!

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

    I would also recommend that you make the validation as function.

  • @spreadneck2063
    @spreadneck2063 Місяць тому +1

    So you can use : instead of = when defining a variable?

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

      name: str = ‘bob’
      Variable: type = value
      : allows you to assign the type

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

    I would suggest adding one last step: a test for the function. 😊

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

    my money don't jiggle jiggle, it folds

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

    Beautiful!

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

    The overall idea of this is great, but the example has issues.
    1. You use Iterable, but you iterate over it twice, which is unsafe, because an Iterator, which can only be looped once is also an Iterable.
    2. You declare float as the sub-type for prices, but check for int or float later. Just use SupportsFloat as the hint instead and explicitly cast later, e.g. via map(float, prices). This will also have the benefit of doing the type validation for you and raise if needed, instead of even needing to call isinstance.

  • @uruloki
    @uruloki 4 місяці тому +5

    Shouldn’t the discount function be renamed to totalAfterDiscount?

    • @michaelhoffmann2891
      @michaelhoffmann2891 4 місяці тому +1

      totalAfterCalculatedDiscountWithLocalisationAndRebates(), which is overridden in DiscountAndRebateCalculationInstantiationFactory.
      Java.............. 🙄
      But in case, you were serious: Java uses camel case, Python uses snake case. If one of my peers uses the former format in the latter language, I will not approve their PR. I see that a lot among more junior, fresh-out-of-uni coders.

    • @uruloki
      @uruloki 4 місяці тому +1

      @@michaelhoffmann2891 Case is usually shop specific. Good on you for enforcing your standard!

    • @michaelhoffmann2891
      @michaelhoffmann2891 4 місяці тому +1

      @@uruloki True, but I would look *very* askance at a shop that dictated camel case for Python in defiance of PEP.

  • @baptisteb8910
    @baptisteb8910 4 місяці тому +1

    For the function name I would say:
    1. As short as possible, but as long as needed.
    And if the name is too long, then the function might need to be divided.
    Otherwise LGTM !

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

    can you do some tutorial in dart/flutter?

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

    05:46 i'm not sure it's pep but probably you write dosrctring in a wrong way. it starts with a verb. like "get current time" instead of "a function that returns you a current time".

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

    thanks

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

    Why do you need to specify the type of the "now" variable? Even in static type language lile c++ you can write "auto" and let the compiler figure out the variable type. Here - almost any IDE will know that the returned value is "datetime" and will give you same intellisense support without the surplus verbosity.

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

      Compiler wouldn't warn you if the function would start, after some modification, returning strings. If you write it explicitly, then you get proper warning if you modify something and the return type is wrong.

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

      In this case, either the rest of the code will be compatible with the new type which means the new type has the same protocol or is a subclass of datetyme type, or the intellisense will alert me of incorrect/incompatible usage.

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

      ​@@ArtyomKatsapActually, yeah, you're right. I thought you talked about function signatures somehow, but yeah, the variable typing is imo kinda pointless there, the datetime.now() is already a typed function.
      If you were to reassign "now", it would make more sense to type it, but here, it seems purely fluff.

    • @Jason-b9t
      @Jason-b9t 4 місяці тому

      @@gJonii In most cases you do not need to type the variable type. (I wouldn't do it either.)
      But typing variable type is sometimes helpful, like the following code:
      from datetime import datetime
      now = datetime.now()
      today = datetime.date(now) # Get today's date
      if today == now:
      print('today is now')
      else:
      print('today is not now') # Print this line because the type of "now" is different from the type of "today".
      If you type variable type it is easier to understand the type of the variable.

    • @Jason-b9t
      @Jason-b9t 4 місяці тому

      ​@@gJonii If you type variable type it is easier to understand the type of the variable.
      now = datetime.now()
      isoNow = datetime.isoformat(now)
      print(f'{now=}, {isoNow=}')
      if isoNow == now:
      print('isoNow is now')
      else:
      print('isoNow is not now') # Print this line because the type of "isoNow" is different from the type of "now".
      But I usually don't type variable type because it's very troublesome.

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

    Brilliant:)

  • @hi5wifi-s567
    @hi5wifi-s567 День тому

    Nice! Short and concise, reuse able, and….
    but example on validation kind of too long. Why can we write as “ if Price >=0” then return “…” and
    “ if Price != percentage” then return “ Invalid” ??

  • @danielrhouck
    @danielrhouck 3 місяці тому +2

    I think you took the short function names too far. Based on the name `get_time`, I expect the *time* and not a string. I’d do `get_time_locale` or something, even with the return type.

  • @TellyT-i5e
    @TellyT-i5e 4 місяці тому +1

    Redundant docs just clutter up the code unless you are making a library api

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

    Only ever abstract over an EXPRESSION, whatever it's arguments...! Simples (:-)

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

    Seems like :type ...: doc is not reflected in the popup (I mean it doesn't mention anything about the type of the params)

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

    If i put all my functions in 1 file, and the "executable" code in another, is there some general guideline for where I should put fixed variables? like, if i call them in the "executable" file, then put them there, otherwise, put in the function file...?

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

      put them where they are gonna be used most, and where it makes sense for them to exist

    • @ffggvfg4323
      @ffggvfg4323 4 місяці тому +1

      Putting all your functions in one file makes your code hard to read and makes debugging a pain. If your code is clogged up with util functions then putting those in a file can be ok as far as variables please on behalf of everyone that will read your code define them in the main file

    • @DrDeuteron
      @DrDeuteron 4 місяці тому +1

      but the point of dunder name == dunder main is that you don't need to separate the code like that, unless another script also needs the function--but then you're writing a package.
      "fixed variables" is just not pythonic..."module constants" go after imports, before def's.

    • @gJonii
      @gJonii 4 місяці тому +1

      I'd put all the code in one file, until you identify some island of functionality that you can separate into its own file with a smaller API.

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

    also and/or emphasis on what you did:
    1) one point of return (if you must have branches)
    2) don't have branches (except for guard clauses)
    3) Use guard clauses: raise exceptions early.
    Most important:
    Be pure:
    No side-effects. Don't modify arguments! e.g.:
    >>>sorted(list_)
    returns a sorted list w/o modifying the argument. Modify objects only in their one methods:
    >>>list_.sort()
    and don't have a state: same arguments means same result (rand and date time notwithstanding).

  • @smanzoli
    @smanzoli 4 місяці тому +1

    The first example was not a good choice (ok, it's an example only), because as a rule, we should NEVER write a function which only purpose is call another function.
    get_time() is completelly useless since we can simply use directly datetime.now()
    Also, inside the useless function, we create a useless variable now, where we could return datetime.now() directly.
    Second example is very good.
    Great video!

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

    about the type annotations;
    i think they're use- and surely helpful BUT, for me as a not very experienced coder who's still figuring out how to do stuff the best way and which techniques are available, and whilst discovering new modules and their functionalities, it can be somewhat distracting to already start using type annotations as much as possible, so my take at it is to use them as long as it stays simple, so to at least get used to them for now and to have a base to build up on later, specially once my scripts start to become bigger. same for docstrings btw..

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

    what vs code extensions are you using? thanks

  • @curtisharwood6626
    @curtisharwood6626 2 місяці тому +1

    your first tip of not naming things thoroughly is definitely just an opinion, and arguably a bad one.

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

    I don't like calling fractions "percents"; that is a source of self-documenting error or ambiguity.

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

    how would I handle a function if I return several value? would I do that as a list and - - > list[Str, into, df. Dataframe()] for example?

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

      tuples! Or a dict.

    • @tw11tube
      @tw11tube 2 місяці тому +1

      Especially if it's part of a library API, use a NamedTuple or dataclass, so the callers can access members by name.

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

    Which IDE are you using?

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

    Error handling is ok basically, but your example so overloaded

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

    comment

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

    The G you pronounce at the end of words (due to accent) is easy to fix... it's silent, simply do not say it! And bingo, your pronounce will get even better!

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

      I can't hear the G you talked about?

  • @JohnUrbanic-m3q
    @JohnUrbanic-m3q 4 місяці тому +6

    Use Python because it it quick and pithy. Once you start adding in this pseudo-typing, error handling and extensive documentation you lose that, and it is a sign that you should be choosing a different language. Right tool for the job.

    • @michealakinkuotu8637
      @michealakinkuotu8637 4 місяці тому +20

      This is not true. Documentation and error handling is a crucial part of any software, whether it's statically typed or dynamically typed.

    • @DrDeuteron
      @DrDeuteron 4 місяці тому +1

      @@michealakinkuotu8637 but LBYL type checking?

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

      @@DrDeuteron Look before you leap is not left out of the discussion in terms of error handling. While LBYL can be redundant or expensive in terms of performance, it doesn't nullify the fact that if done correctly, it gives you a "safer code" and in some cases, a much cleaner or clearer error handling through provision of meaningful feedback.

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

      @@michealakinkuotu8637 I get the POV, but that gets back to OP's point: this is python: Lighten up!
      The only place I allow branchy code is in error handling. When everything is working, I want minimal cyclomatic complexity.

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

      I do find the excessive hints annoying, the worst is type hinting strongly typed magic methods, like:
      init(self) -> None:
      str(self) -> str:
      I just can't....

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

    If your functions is simple enough and had a proper name, a documentation is not needed. Besides keeping documentation up to date double your work load (rewrite the doc, update the example,... every time you refactor) Most IDE can infer input and output so that's a complete waste of time. Documentation is only useful in a library and only for exposed methods.

  • @baptisteb8910
    @baptisteb8910 4 місяці тому +10

    For the function name I would say:
    1. As short as possible, but as long as needed.
    And if the name is too long, then the function might need to be divided.
    Otherwise LGTM !

    • @michaelhoffmann2891
      @michaelhoffmann2891 4 місяці тому +1

      I agree. If my func/method is getting to be more than a screen (and I use a big screen) or around 50 lines (incl copious whitespace), I refactor. Even if it's a helper method used once, with a leading '_' ( to make it "Python Private(tm)").