1
\$\begingroup\$

I am a beginner in Python-3. I was solving a level 2 challenge in Google Foobar. Given an initial amount of items (LAMBs) I have to distribute it among individuals, who are ranked according to seniority, while adhering to the following rules:

  1. The junior most individual gets 1 Lamb
  2. An immediately senior individual can't get more than twice their juniors lambs, otherwise the junior will rebel
  3. A senior individual can't get less lambs than the sum of their 2 previous juniors, otherwise the senior will rebel
  4. In case lambs are leftover and adding a senior does not violate any previous rules, they must be added.

The goal is to find the difference between how many individuals can be paid if we pay the minimum amount to each (stingy) and the maximum amount to each (generous). I noticed that stingy condition resembles a Fibonacci sequence and the generous condition is sum of powers of 2. My code is as follows:

from math import log2
def solution(total_lambs)
 return (stingy(total_lambs) - generous(total_lambs))
def stingy(total_lambs)
 if (total_lambs == 1): # from condition 1 at least 1 individual gets paid
 return 1
 elif (total_lambs == 2): # from condition 2 and 4 the most each can is 1
 return 2
 else:
 ind, total, x0, x1 = 2, 2, 1, 1 # no. individuals, sum of Fib seq, intial Fib nos.
 while (total <= total_lambs):
 x0, x1 = x1, x0 + x1 # advancing the Fib seq
 total = total + x1 # total is the sum of lambs paid
 if (total > total_lambs): 
 return ind
 else:
 ind = ind + 1
def generous(total_lambs):
 if (total_lambs == 1):
 return 1
 if (total_lambs == 2):
 return 2
 else:
 ind = int(log2(capital + 1)) # typecasting to int removes decimals
 if (total_lambs - 2**n + 1 >= 2**(n - 1) + 2**(n - 2)): # condition 2, 3 and 4 
 return ind + 1
 else:
 return ind

Please suggest improvements wherever possible.

Reinderien
71k5 gold badges76 silver badges256 bronze badges
asked Nov 23, 2020 at 12:20
\$\endgroup\$
3
  • 2
    \$\begingroup\$ Great first question! \$\endgroup\$ Commented Nov 23, 2020 at 13:06
  • 3
    \$\begingroup\$ Can you provide a link to the original question? \$\endgroup\$ Commented Nov 23, 2020 at 13:07
  • \$\begingroup\$ @reinderien I unfortunately submitted my response before i saved the original question so i can't copy it verbatim. However, it is available online if you search foobar Lovely lucky LAMBs. \$\endgroup\$ Commented Nov 24, 2020 at 0:58

2 Answers 2

3
\$\begingroup\$

Early-returns

else-after-return is redundant, so your stingy is equivalent to

 if (total_lambs == 1): # from condition 1 at least 1 individual gets paid
 return 1
 if (total_lambs == 2): # from condition 2 and 4 the most each can is 1
 return 2
 ind, total, x0, x1 = 2, 2, 1, 1 # no. individuals, sum of Fib seq, intial Fib nos.
 while (total <= total_lambs):
 x0, x1 = x1, x0 + x1 # advancing the Fib seq
 total = total + x1 # total is the sum of lambs paid
 if (total > total_lambs): 
 return ind
 ind = ind + 1

Return parens

You have a mix of this style:

return 1

and this style:

return (stingy(total_lambs) - generous(total_lambs))

the latter not needing outer parens. Probably best to go with the non-parenthetical style. The same is true of

while (total <= total_lambs):

In-place addition

total = total + x1
ind = ind + 1

can be

total += x1
ind += 1
answered Nov 23, 2020 at 13:57
\$\endgroup\$
4
  • \$\begingroup\$ Thank you. As you mentioned in another comment, what can be a good name for storing Fibonnacci numbers? \$\endgroup\$ Commented Nov 24, 2020 at 1:00
  • \$\begingroup\$ A good name? For which variable? \$\endgroup\$ Commented Nov 24, 2020 at 1:01
  • \$\begingroup\$ Storage of the Fibonacci numbers. \$\endgroup\$ Commented Nov 24, 2020 at 1:09
  • 1
    \$\begingroup\$ Oh, you mean x0 and x1. prev and current maybe? \$\endgroup\$ Commented Nov 24, 2020 at 1:25
2
\$\begingroup\$

apart from the improvements mentioned above, for the stingy function you can also reduce the amount of code by using a list with the last 2 fibonacci members, and integrating the initial conditions into the logic

def stingy(total_lambs):
 paid = 0
 previous = [1, 0]
 while total_lambs > 0:
 paid += 1
 previous = [sum(previous), previous[0]]
 total_lambs -= previous[0]
 return paid
answered Nov 23, 2020 at 16:45
\$\endgroup\$
3
  • 1
    \$\begingroup\$ Ish. Overall this is a good suggestion, but framing this as a list doesn't make sense. Just have two variables. \$\endgroup\$ Commented Nov 23, 2020 at 17:03
  • \$\begingroup\$ In other words, keep the OP's x0, x1, possibly with better names. \$\endgroup\$ Commented Nov 23, 2020 at 17:05
  • \$\begingroup\$ Thanks, i will keep this structure in mind for future coding. I framed my function in this way as the rule 3 in original question had an explanation that it is not applicable if only upto 2 individuals get paid. So I did not think of this elegant way. \$\endgroup\$ Commented Nov 24, 2020 at 1:06

Your Answer

Draft saved
Draft discarded

Sign up or log in

Sign up using Google
Sign up using Email and Password

Post as a guest

Required, but never shown

Post as a guest

Required, but never shown

By clicking "Post Your Answer", you agree to our terms of service and acknowledge you have read our privacy policy.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.