mirror of
https://github.com/LCTT/TranslateProject.git
synced 2025-01-13 22:30:37 +08:00
7de6a1166e
sources/tech/20200914 Use Python to solve a charity-s business problem.md
216 lines
13 KiB
Markdown
216 lines
13 KiB
Markdown
[#]: collector: (lujun9972)
|
||
[#]: translator: ( )
|
||
[#]: reviewer: ( )
|
||
[#]: publisher: ( )
|
||
[#]: url: ( )
|
||
[#]: subject: (Use Python to solve a charity's business problem)
|
||
[#]: via: (https://opensource.com/article/20/9/solve-problem-python)
|
||
[#]: author: (Chris Hermansen https://opensource.com/users/clhermansen)
|
||
|
||
Use Python to solve a charity's business problem
|
||
======
|
||
Comparing how different programming languages solve the same problem is
|
||
fun and instructive. Next up, Python.
|
||
![Python programming language logo with question marks][1]
|
||
|
||
In my [first article][2] in this series, I described a problem of dividing bulk supplies into hampers of similar value to distribute to struggling neighbors in your community. I also wrote about how I enjoy solving small problems like this with small programs in various languages and comparing how they do it.
|
||
|
||
In the first article, I solved this problem with the [Groovy][3] programming language. Groovy is like [Python][4] in many ways, but syntactically it's more like C and Java. Therefore, it should be interesting and instructive to create the same solution in Python.
|
||
|
||
### The Python solution
|
||
|
||
In Java, I declare utility classes to hold tuples of data (the new record feature is going to be great for that). In Groovy, I use the language support for maps, and I follow the same approach in Python.
|
||
|
||
Use a list of dictionaries to hold the bulk items picked up from the wholesaler:
|
||
|
||
|
||
```
|
||
packs = [
|
||
{'item':'Rice','brand':'Best Family','units':10,'price':5650,'quantity':1},
|
||
{'item':'Spaghetti','brand':'Best Family','units':1,'price':327,'quantity':10},
|
||
{'item':'Sardines','brand':'Fresh Caught','units':3,'price':2727,'quantity':3},
|
||
{'item':'Chickpeas','brand':'Southern Style','units':2,'price':2600,'quantity':5},
|
||
{'item':'Lentils','brand':'Southern Style','units':2,'price':2378,'quantity':5},
|
||
{'item':'Vegetable oil','brand':'Crafco','units':12,'price':10020,'quantity':1},
|
||
{'item':'UHT milk','brand':'Atlantic','units':6,'price':4560,'quantity':2},
|
||
{'item':'Flour','brand':'Neighbor Mills','units':10,'price':5200,'quantity':1},
|
||
{'item':'Tomato sauce','brand':'Best Family','units':1,'price':190,'quantity':10},
|
||
{'item':'Sugar','brand':'Good Price','units':1,'price':565,'quantity':10},
|
||
{'item':'Tea','brand':'Superior','units':5,'price':2720,'quantity':2},
|
||
{'item':'Coffee','brand':'Colombia Select','units':2,'price':4180,'quantity':5},
|
||
{'item':'Tofu','brand':'Gourmet Choice','units':1,'price':1580,'quantity':10},
|
||
{'item':'Bleach','brand':'Blanchite','units':5,'price':3550,'quantity':2},
|
||
{'item':'Soap','brand':'Sunny Day','units':6,'price':1794,'quantity':2}]
|
||
```
|
||
|
||
There is one bulk pack of 10 bags of rice and 10 bulk packs with one bag each of spaghetti. In the above, the variable `packs` is set to a Python list of dictionaries. This turns out to be very similar to the Groovy approach. A few points worth noting about the difference between Groovy and Python:
|
||
|
||
1. In Python, there is no keyword used to define the variable `packs`; Python expects the first use to set a value.
|
||
2. Python dictionary keys (e.g., `item`, `brand`, `units`, `price`, `quantity`) require quotes to indicate they are strings; Groovy assumes these are strings, but accepts quotes as well.
|
||
3. In Python, the notation `{ … }` indicates a dictionary declaration; Groovy uses the same square brackets as a list, but the structure in both cases must have key-value pairs.
|
||
|
||
|
||
|
||
And, yes, those prices aren't in US dollars.
|
||
|
||
Next, unpack the bulk packages. Unpacking the single bulk package of rice, for example, will yield 10 units of rice; that is, the total number of units yielded is `units * quantity`. The Groovy script uses a handy function called `collectMany` that can be used to flatten out lists of lists. As far as I know, Python doesn't have anything similar, so use two list comprehensions to produce the same result:
|
||
|
||
|
||
```
|
||
units = [[{'item':pack['item'],'brand':pack['brand'],
|
||
'price':(pack['price'] / pack['units'])}] *
|
||
(pack['units'] * pack['quantity']) for pack in packs]
|
||
units = [x for sublist in units for x in sublist]
|
||
```
|
||
|
||
The first list comprehension (assignment to units) builds the list of lists of dictionaries. The second "flattens" that into just a list of dictionaries. Note that both Python and Groovy provide an `*` operator that takes a list on the left and a number `N` on the right and replicates the list `N` times.
|
||
|
||
The final step is to repack the units into the hampers for distribution. As in the Groovy version, you need to get a bit more specific about the ideal hamper value, and you might as well not be overly restrictive when you get down to just a few units left:
|
||
|
||
|
||
```
|
||
valueIdeal = 5000
|
||
valueMax = valueIdeal * 1.1
|
||
```
|
||
|
||
OK! Repack the hampers:
|
||
|
||
|
||
```
|
||
import random
|
||
hamperNumber = 0 # [1]
|
||
while len(units) > 0: # [2]
|
||
hamperNumber += 1
|
||
hamper = []
|
||
value = 0
|
||
canAdd = True # [2.1]
|
||
while canAdd: # [2.2]
|
||
u = random.randint(0,len(units)-1) # [2.2.1]
|
||
canAdd = False # [2.2.2]
|
||
o = 0 # [2.2.3]
|
||
while o < len(units): # [2.2.4]
|
||
uo = (u + o) % len(units)
|
||
unit = units[uo]
|
||
unitPrice = unit['price'] # [2.2.4.1]
|
||
if len(units) < 3 or not (unit in hamper) and (value + unitPrice) < valueMax:
|
||
# [2.2.4.2]
|
||
hamper.append(unit)
|
||
value += unitPrice
|
||
units.pop(u) # [2.2.4.3]
|
||
canAdd = len(units) > 0
|
||
break # [2.2.4.4]
|
||
o += 1 # [2.2.4.5]
|
||
# [2.2.5]
|
||
print('')
|
||
print('Hamper',hamperNumber,'value',value)
|
||
for item in hamper:
|
||
print('%-25s%-25s%7.2f' % (item['item'],item['brand'],item['price'])) # [2.3]
|
||
print('Remaining units',len(units)) # [2.4]
|
||
```
|
||
|
||
Some clarification, with numbers in brackets in the comments above (e.g., _[1]_) corresponding to the clarifications below:
|
||
|
||
* 1\. Import Python's random number generator facilities and initialize the hamper number.
|
||
* 2\. This `while` loop will redistribute units into hampers as long as there are more available:
|
||
* 2.1 Increment the hamper number, get a new empty hamper (a list of units), and set its value to 0; start off assuming you can add more items to the hamper.
|
||
* 2.2 This `while` loop will add as many units to the hamper as possible (the Groovy code used a `for` loop, but Python's `for` loops expect to iterate over something, while Groovy has the more traditional C form of `for` loop):
|
||
* 2.2.1 Get a random number between zero and the number of remaining units minus 1.
|
||
* 2.2.2 Assume you can't find more units to add.
|
||
* 2.2.3 Create a variable to be used for the offset from the starting point where you're looking for items to put in the hamper.
|
||
* 2.2.4 Starting at the randomly chosen index, this `while` loop will try to find a unit that can be added to the hamper (once again, note that the Python `for` loop probably isn't suitable here since the length of the list will change during processing).
|
||
* 2.2.4.1. Figure out which unit to look at (random starting point + offset) and get its price.
|
||
* 2.2.4.2 You can add this unit to the hamper if there are only a few left or if the value of the hamper isn't too high once the unit is added.
|
||
* 2.2.4.3 Add the unit to the hamper, increment the hamper value by the unit price, remove the unit from the available units list.
|
||
* 2.2.4.4 As long as there are units left, you can add more, so break out of this loop to keep looking.
|
||
* 2.2.4.5 Increment the offset.
|
||
* 2.2.5 On exit from this `while` loop, if you inspected every remaining unit and could not find one to add to the hamper, the hamper is complete; otherwise, you found one and can continue looking for more.
|
||
* 2.3 Print out the contents of the hamper.
|
||
* 2.4 Print out the remaining units info.
|
||
|
||
|
||
|
||
When you run this code, the output looks quite similar to the output from the Groovy program:
|
||
|
||
|
||
```
|
||
Hamper 1 value 5304.0
|
||
UHT milk Atlantic 760.00
|
||
Tomato sauce Best Family 190.00
|
||
Rice Best Family 565.00
|
||
Coffee Colombia Select 2090.00
|
||
Sugar Good Price 565.00
|
||
Vegetable oil Crafco 835.00
|
||
Soap Sunny Day 299.00
|
||
Remaining units 148
|
||
|
||
Hamper 2 value 5428.0
|
||
Tea Superior 544.00
|
||
Lentils Southern Style 1189.00
|
||
Flour Neighbor Mills 520.00
|
||
Tofu Gourmet Choice 1580.00
|
||
Vegetable oil Crafco 835.00
|
||
UHT milk Atlantic 760.00
|
||
Remaining units 142
|
||
|
||
Hamper 3 value 5424.0
|
||
Soap Sunny Day 299.00
|
||
Chickpeas Southern Style 1300.00
|
||
Sardines Fresh Caught 909.00
|
||
Rice Best Family 565.00
|
||
Vegetable oil Crafco 835.00
|
||
Spaghetti Best Family 327.00
|
||
Lentils Southern Style 1189.00
|
||
Remaining units 135
|
||
|
||
…
|
||
|
||
Hamper 21 value 5145.0
|
||
Tomato sauce Best Family 190.00
|
||
Tea Superior 544.00
|
||
Chickpeas Southern Style 1300.00
|
||
Spaghetti Best Family 327.00
|
||
UHT milk Atlantic 760.00
|
||
Vegetable oil Crafco 835.00
|
||
Lentils Southern Style 1189.00
|
||
Remaining units 4
|
||
|
||
Hamper 22 value 2874.0
|
||
Sardines Fresh Caught 909.00
|
||
Vegetable oil Crafco 835.00
|
||
Rice Best Family 565.00
|
||
Rice Best Family 565.00
|
||
Remaining units 0
|
||
```
|
||
|
||
The last hamper is abbreviated in contents and value.
|
||
|
||
### Closing thoughts
|
||
|
||
At a glance, there isn't a whole lot of difference between the Python and Groovy versions of this program. Both have a similar set of constructs that make handling lists and dictionaries very straightforward. Neither requires a lot of "boilerplate code" or other "ceremonial" actions.
|
||
|
||
Also, as in the Groovy example, there is some fiddly business about being able to add units to the hamper. Basically, you pick a random position in the list of units and, starting at that position, iterate through the list until you either find a unit whose price allows it to be included or until you exhaust the list. Also, when there are only a few items left, you just toss them into the last hamper.
|
||
|
||
Another issue worth mentioning: This isn't a particularly efficient approach. Removing elements from lists, being careless about repeated expressions, and a few other things make this less suitable for a huge redistribution problem. Still, it runs in a blink on my old machine.
|
||
|
||
If you are shuddering at my use of `while` loops and mutating the data in this code, you probably wish I made it more functional. I couldn't think of a way to use map and reduce features in Python in conjunction with a random selection of units for repackaging. Can you?
|
||
|
||
In the next article, I'll re-do this in Java just to see how much less effort Groovy and Python are, and future articles will cover Julia and Go.
|
||
|
||
--------------------------------------------------------------------------------
|
||
|
||
via: https://opensource.com/article/20/9/solve-problem-python
|
||
|
||
作者:[Chris Hermansen][a]
|
||
选题:[lujun9972][b]
|
||
译者:[译者ID](https://github.com/译者ID)
|
||
校对:[校对者ID](https://github.com/校对者ID)
|
||
|
||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||
|
||
[a]: https://opensource.com/users/clhermansen
|
||
[b]: https://github.com/lujun9972
|
||
[1]: https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/python_programming_question.png?itok=cOeJW-8r (Python programming language logo with question marks)
|
||
[2]: https://opensource.com/article/20/8/solving-problem-groovy
|
||
[3]: https://groovy-lang.org/
|
||
[4]: https://www.python.org/
|