Skip to content

ttaiv/taylor-bracelet-optimizer

Repository files navigation

Friendship Bracelet Optimizer

Introduction

This is a program that takes letters and a list of strings and finds the best strings to form using the given letters so that as many letters as possible are used. In other words, it minimizes the leftover letters. The idea of this program was to help in building friendship bracelets for Taylor Swift's Eras tour. My girlfriend is a huge swiftie and wanted to use her available letters as efficiently as possible. This repository contains both a (slow) recursive dynamic programming solution and a (fast) integer linear programming solution for the problem.

The problem

Finding the best bracelet texts is a variation of classic Knapsack problem, which is known to be NP-hard. Specifically, the problem is 0/1 multidimensional knapsack problem

It can be formulated as follows:

Bracelet problem mathematical formulation

The dynamic programming solution

The initial solution used dynamic programming with memoization. However, the running time of this approach exploded as the amount of available letters grew. I suspect the reason for this to be the limited overlap between subproblems – only a few combinations of texts lead to precisely the same letter usage. Adding the memoization to the recursive function actually increased the running time (likely due to the the overhead of storing the letter states). On my machine the running time of the dynamic programming solution with a data set of 61 available letters and 305 bracelet text options is about 1 minute and 15 seconds without and 2 minutes with memoization. With the real letter counts (total of 283 letters) the program did not finish execution in 5 hours. The solution with memoization is in the branch memorization.

The integer linear programming solution

This more efficient method leverages integer linear programming with the help of the python library PuLP that calls CBC solver. The running time with real letter counts was just a few seconds.

Files

  • bracelet_texts.py: Contains a python list of all different bracelet_texts. If you use the program, feel free to modify it to your liking.
  • find_bracelet_texts_ilp.py: Implements a function that finds the best texts using integer linear programming.
  • find_bracelet_texts_recursive.py: Implements the dynamic programming solution. Main branch contains the plain recursive version and the branch memorization the one with memoization.
  • main.py: Contains the main program that loads the Excel file with letter counts, preprocesses the texts, calls the chosen solver function and prints the results.
  • plot.py: Contains a script that plots the counts of the available letters and compares them to the letter counts of the texts.
  • test_find_bracelet_texts.py: Contains a few manually engineered test cases for the solver functions.
  • data/
    • letter_counts_real.xlsx: The letter counts for which I wanted to find the best texts (total of 283 letters)
    • letter_counts_test_small.xlsx: A small test letter set (total of 25 letters)
    • letter_counts_test_small2.xlsx: Another small test letter set (total of 29 letters)
    • letter_counts_test_medium.xlsx: A slightly larger test letter set (total of 61 letters)

About

Script that solves 0/1 multidimensional knapsack problem

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages