As all perfectly-normal-peopleTM do, I keep a spreadsheet of different foods and cost / what nutrients they contain. Cause if you are not optimizing, you're not living life to it's fullest!
But it always been me rougly eyeballing the measurments and trying to put together an optimal diet by hand. So when I find out about OR-Tools? Caching!
OR-Tools & Methodology
It's Googles contraint solver that they have open sourced. It contains a little bit of everything (graph, routing, etc.) but it's the constraint part I'm after. After all, what is an optimized diet but a list of constraints?
Some parts where easy, getting the basics down went quick! However the constrain for a max number of items was pure pain to implement, as I needed to work around the system to get something it could work with
The Data is quite manual. Livsmedelsverket have a good database but I needed to curate specific objects, as I had to do the prices myself. As well as the base needs are a bit tricky and up to interpretation
I used a paper from Livsmedelsverket näringsrekommendationer in combination with FDA recommendations and my own sound(-ish) judgement to determine what to keep in
Code
So, the good parts. Let's start with the relevant code, there is more for output formating / reading data / etc. but this is the main work chunk
# 1 = 100g, 10 = 10g, 100 = 1g - Other values for precision will NOT work properly (mainly output becoming wrong / ugly)
precision = [1, 10, 100][1]
# If you want to loosen the constraints to some percentage, 1 = 100%
constraintsPercentage = 0.95
# Limits how many items to include
maxItems = 10
# All values are increased to be able round to integers without loss of precision
scalingFactor = 1000
def main():
# Scale upp with scalingFactor and apply constraintsPercentage / precision
for values in rdi:
values[1] = math.floor(values[1] * scalingFactor)
for values in constraints:
values[1] = math.floor(values[1] * scalingFactor * constraintsPercentage)
for values in items:
for index, v in enumerate(values[1:], start=1):
values[index] = math.floor((v / precision) * scalingFactor)
solver = pywraplp.Solver.CreateSolver("SCIP")
# Variables
# Created from constraints variable, min values for each nutrient
x = {}
# 0 or 1 if item is used, to enable maxItems to work
y = {}
for i in range(len(items)):
# Limits max units per item for the solution
max = 1000000
x[i] = solver.IntVar(0, max, "")
y[i] = solver.IntVar(0, 1, "")
solver.Add(x[i] <= y[i] * max)
# Constraints for nutrients
for c in constraints:
i = itemsHeader.index(c[0])
min = c[1]
solver.Add(solver.Sum(items[j][i] * x[j] for j in range(len(x))) >= min)
# Limit to maxItems
solver.Add(solver.Sum(y[i] for i in range(len(y))) <= maxItems)
# Objective
cost = []
for i in range(len(x)):
ic = itemsHeader.index("cost")
cost.append(items[i][ic] * x[i])
solver.Minimize(solver.Sum(cost))
status = solver.Solve()
Not to horrible
Results
What is the best diet?
*DRUMROLL*
Cost is 18.8kr per day
8.3kr for 6.4dl of Milk (3%)
3.3kr for 410g of Wheat Flour
3.0kr for 90g of Broccoli
1.3kr for 70g of Yellow Peas
1.0kr for 30g of Sunflower Seeds
80 öre for 20g of Eggs
80 öre for 0.3dl of Rapeseed oil
40 öre for 40g of Carrots
I'm Swedish, can you tell?
For euro / US just divide by 10 and you get a rough approximation
Which gives you the following macro / micro nutrients
(RDI based on my own age / gender / weight / etc.)
Macronutrients | Amount | RDI% |
---|---|---|
calories | 2577kcal | 103% |
carbohydrate | 366g | 133% |
fiber | 30g | 100% |
fat | 77g | 98% |
protein | 85g | 103% |
Vitamins | Amount | RDI% |
---|---|---|
vitamin A | 611mg | 97% |
vitamin C | 86mg | 96% |
folate | 328mcg | 131% |
vitamin B12 | 3mcg | 95% |
Minerals | Amount | RDI% |
---|---|---|
calcium | 1006mg | 134% |
iron | 10mg | 144% |
magnesium | 361mg | 129% |
potassium | 3156mg | 112% |
zinc | 10mg | 95% |
iodine | 114mcg | 95% |
Summary
Milk is awesome!
Also vegetables good, etc.
It was fun to play around with but don't know if created any bound breaking knowledge
Milk comes surprisingly (to me) high up with quite a bit to consume daily
410g of flour is almost 7dl so quite a bit of bread as well
Bread & Milk is most of the content, but seeing as they are both staples it's not to surprising
I thought potatoes would make it onto the list but they have exploded in price so that kicked them off it
Use a balanced diet, eat a bit of everything, not medical adviced, etc
But it does go to show that you can get everything you need for a bit under 600kr per month (€53)