This project tries to solve the optimization problem of the vrp, with genetic algorithms. The goal is to minimize the distance made by the car and the number of cars been used, while making the delivers for all the packages, while considering certain constraints.
We found that depending of the version use or if you use pycharm or visual studio an error may appear.
If you get the following error:
- ModuleNotFoundError: No module named 'CIFO2024', in the imports you have to delete CIFO2024 an example:
- Change:
- from CIFO2024.charles.fitness_functions import get_fitness
- To:
- from charles.fitness_functions import get_fitness
Then if the following error appears:
- FileNotFoundError: [Errno 2] No such file or directory: '.\goeke-2018\c202C16.txt'
- Then on \vrp_cifo\CIFO2024\read_data.py"
- change on line 6 file_path = os.path.join('.', 'goeke-2018',instance) to
- file_path = os.path.join('CIFO2024', 'goeke-2018',instance)
- If it does not work try the following:
- file_path = os.path.join('vrp_cifo\CIFO2024', 'goeke-2018',instance)
- Vehicles;
- Pickup and delivery locations;
- Recharging stations;
- Depot.
Each row of the dataset contains the following columns:
-
StringId: Unique identifier of the location;
-
Type: Indicates the function of the location, where:
- D: depot;
- F: recharging station;
- CP: customer pickup location;
- CD: customer delivery location.
-
X, Y: Location coordinates (distances are assumed to be Euclidean);
-
Demand: Specifies the quantity of freight capacity required (positive at pickup, negative at delivery);
-
ReadyTime and DueDate: beginning and end of the time window (waiting is allowed);
-
ServiceTime: denotes the entire time spent at pickup/delivery for loading/unloading operations;
-
PartnerId: Relevant for transportation requests and provides the StringId of the partner of each pickup and delivery location.
-
Besides the locations, the dataset also contains information about the vehicle, such as velocity, freight capacity, and battery capacity.
In order to make our data available to the algorithm we have to represent:
- CPX = Customers Pickup (where X represents the indexed customer);
- CDX = Customer Delivery.
For example: A scenario that uses one vehicle and this car has to deliver to customer 1,2 and 3 can be represented as follows: [[CP1,CP2,CD1,CP3,CD3,CD2]]
.
The route of the different vehicles is defined by the inner lists, within the main list. In other words, we have a list, which contains a group of list each represent different cars and its corresponding routes. For example, we may have [[CP1, CD1, CP2, CD2], [CP3, CD3]]
. Here we have two cars, two different routes, car number is delivering the package to customer 1 and two, and the second car is delivering the packages to customer 3. Additionally, the result will be shown in this format: (2, [[CP1, CD1, CP2, CD2], [] [CP3, C3], [], []], 2000) 2, is the number of cars chosen, and the 3 empty spaces shows that the maximum vehicles available are 5. However, the algorithm chose only two cars as the best possible solution, and its fitness is 2000.
- All routes must start and end at Depot 1;
- All picked up orders must be delivered, to the corresponding customer;
- The order only can be picked up if the time window is ready for the pickup;
- The order must be picked up before delivery;
- A car must not exceed its package capacity;
- A car cannot exceed its battery capacity, if its close to running out of battery it should go to charging station.
The fitness function takes into account the following variables:
- Capacity -
get_fitness_capacity()
; - Time -
get_fitness_time()
.
swap_mutation()
;inversion_mutation()
.
- Single-point crossover -
single_point_xo()
; - Cycle crossover -
cycle_xo()
; - Partially-mapped crossover -
pmx()
; - Constructive crossover -
sequential_constructive_xo()
.
- Tournament Selection;
- Fitness Proportionate Selection.
Our implementation results were evaluated across various configurations. We conducted 100 generations and 20 iterations for each configuration, calculating the median of the best fitness values over multiple runs. To provide a comprehensive analysis, we compared our results on two different instances: one with 106 pickup or delivery locations and another with 16 locations. The configurations that performed best included tournament selection, single-point random crossover, and swap mutation, enhanced with elitism.
We've implemented a random insertion to avoid introducing bias during the creation of the population. However, for introducing some variability and ensuring that information is distributed across the different generations, we've used several crossover and mutation functions. The combination of all the techniques used gave a good solution for the problem.
- Matplotlib;
- Numpy;
- Pandas;
- Seaborn;
- Chardet.
Name | ID |
---|---|
Devora Cavaleiro | 20230794 |
Carlos Rodrigues | 20230543 |
David Guarin | 20230602 |
Burcu Yesilyurt | 20230763 |
Lia Henao | 20230600 |
- Rahmat, R. W., & Zaharuddin, W. M. (2013). Solving the Vehicle Routing Problem using Genetic Algorithm.
- Toth, P., & Vigo, D. (2002). An overview of vehicle routing problems. European Journal of Operational Research, 144(3), 465-474.
- Keskin, M., & Çatay, B. (2018). The electric vehicle routing problem with time windows and recharging stations. Transportation Research Part C: Emerging Technologies, 87, 113-137.
- Michael Schneider, Andreas Stenger, Dominik Goeke (2012). The Electric Vehicle Routing Problem with Time Windows and Recharging Stations.
- Goeke 2018 dataset: http://www.vrp-rep.org/datasets/item/2019-0001.html