By Dr. Elisabeth Rodríguez-Heck
Mathematical programming is an extremely powerful and flexible technology. Even the most traditional types of mathematical models—like Linear Programs (LP), Mixed-Integer Programs (MIP), and Quadratic Programs (QP)—allow you to solve many of the world’s most complex problems across manufacturing, electrical power distribution, finance, scheduling, supply chain management, transport and logistics, and more.
You may find, however, that for some business requirements, an LP, MIP, or QP alone might not be enough to capture the complexities of your problem or your organization’s processes. That’s why our Gurobi R&D team has built several advanced features that may help you out. Let’s take a look at four of them:
Real-world optimization problems often have two or more competing—or even conflicting—objectives. For example, a hospital manager may want to minimize the number of total shifts for hospital nurses while maximizing worker satisfaction. Or an investor may want to create a portfolio that maximizes profit while minimizing risk.
In such cases, Gurobi’s Multiple Objectives feature may be just what you are looking for. The Multiple Objectives API allows you to express several objectives and seamlessly integrate them with the rest of your code, helping to keep your codebase easy to understand and maintain.
Gurobi can handle the trade-offs of multiple objectives using one of two approaches—or a combination of both:
Sometimes, using a multi-objective hierarchy can help address numerical issues in problems with large objective function coefficients. For example, a minimization problem may contain two clearly distinct sets of objective function coefficients—with one group of coefficients ranging between 0 and 100 and another group ranging between 100,000 and 100,500. Often, the objective terms with the large coefficients actually correspond to a second “hidden” objective function, which is modeled as a penalty. For example, in the nurse scheduling problem, the primary objective may be to minimize shifts, but a second penalized objective may account for shift balance among the nurses. Unbalanced solutions are penalized (with a factor of 100,000) and are, therefore, (hopefully) avoided by the minimization due to the large coefficient. However, such a large spread of coefficients can sometimes cause numerical issues that can be avoided by modeling the problem as a hierarchical multi-objective optimization instead.
Take a look at our multiobj.py example to learn more about this feature.
Life is uncertain—and so is the input data of many business problems. You may have a mathematical model capturing all the details of your problem, but you would still need to understand what happens to the optimized production if the forecasted demand is 10% lower than anticipated. Or you may be planning the construction of a building and want to know what happens to the building costs if phase one of the project is a day late, or two days late, or even five days late.
Gurobi’s Multiple Scenarios feature allows you to specify a set of different scenarios and compute optimal solutions for all these scenarios. While it is possible to implement and solve the scenarios as separate models, using this feature allows you to solve the problems faster than if they were completely separate models, because all the scenarios are solved as one big model using binary variables to distinguish the scenarios.
While, at first, it may look like the multiple scenarios feature is quite restricted, there are some tips and tricks to circumvent these restrictions. A hard restriction, though, is that this feature cannot be used in combination with the multiple objectives feature.
Explore our multiscenario.py and sensitivity.py examples to learn more about this feature.
You have finished building an impeccable model capturing all of the business details. After fixing some bugs, you spend some time improving your model and maybe even tuning the solver parameters to obtain the best performance. Finally, your model produces an optimal solution in a reasonable time. Congratulations, your job here is done!
…or maybe not?
Very often, the hardest part is not the actual modeling or coding itself, but it’s getting your customers or users to accept and implement the solution that the optimization provided.
How is this possible? Can’t they see this is optimal?
Well… sometimes it may be difficult to accept a solution that comes out of a “black box.” Especially if your users are not familiar with mathematical optimization, it may be difficult to believe that it is actually possible to find an optimal solution to such a complex business problem.
In such cases, it may be very useful to have more than one solution available and provide this to the end-user so that they can compare the solutions and understand why the optimal solution is actually optimal. This approach helps to demonstrate the value of the optimal solution and gives decision-makers a greater feeling of control.
With Gurobi, you can define a Solution Pool, which lets the solver report multiple solutions automatically after a single run. There are different ways to set up the solution pool—such as (1) storing all solutions found in the regular optimization without any additional computation effort or (2) storing the n best solutions in addition to the optimal one, at the cost of additional computations.
Having multiple solutions can also be helpful in cases where your business problem has some implicit elements—like preferences or other aspects—that may be difficult to quantify. In this case, you may prefer to select a solution from the pool, even if it’s different from the optimal one.
Finally, modelers often identify modeling errors or missing constraints by looking at the solutions. While an optimal solution can be helpful for this, it is possible that having other solutions can further help.
Our poolsearch.py example is a good place to get started with this feature.
Dealing with infeasibility can be very frustrating. Especially when working with large-scale models, it can be difficult to understand which constraints are problematic. Sometimes the mathematical translation of the constraints is perfectly sound, but the business requirements must be revised. Gurobi provides two features to help you deal with infeasibility.
The first feature is the Irreducible Inconsistent Subsystem (IIS) computation, which can help you understand why your model is infeasible. In short, an IIS is an infeasible subset of constraints and variable bounds of your original model, such that removing a single constraint or bound makes it feasible. Writing an IIS into a human-understandable model format can help you detect the culprit(s) by manually inspecting the set of conflicting constraints.
Since IISs are meant to be interpreted by a human, Gurobi tries to provide a small IIS. However, this is sometimes not possible, because computing an IIS is an extremely difficult problem itself, often involving the resolution of many MIPs. In cases where the IIS is not helpful, computing a Feasibility Relaxation might come in handy to help you understand which changes you have to make to your model to recover feasibility. The feasopt.py example illustrates the use of the Feasibility Relaxation feature.
Often enough, the business requirements make perfect sense and their translation into mathematical constraints is correct, but Gurobi still says that your model is infeasible. In the majority of such cases, numerical issues are to blame. A symptom of numerical issues can be that optimization says that the model is infeasible, but the IIS computation throws an error saying that an IIS cannot be computed on a feasible model. Even though parameter settings can help resolve the symptoms of numerical issues, addressing the root cause of the numerical issues is usually the recommended and most robust solution.
To dig even deeper into Gurobi’s advanced modeling features—and discover more “hidden gems”—check out our one-hour, on-demand webinar presentation: “Hidden Gems – Useful Gurobi Features You Might Not Know.”
GUROBI NEWSLETTER
Latest news and releases
Choose the evaluation license that fits you best, and start working with our Expert Team for technical guidance and support.
Request free trial hours, so you can see how quickly and easily a model can be solved on the cloud.
Cookie | Duration | Description |
---|---|---|
_biz_flagsA | 1 year | A Cloudflare cookie set to record users’ settings as well as for authentication and analytics. |
_biz_pendingA | 1 year | A Cloudflare cookie set to record users’ settings as well as for authentication and analytics. |
_biz_sid | 30 minutes | This cookie is set by Bizible, to store the user's session id. |
ARRAffinity | session | ARRAffinity cookie is set by Azure app service, and allows the service to choose the right instance established by a user to deliver subsequent requests made by that user. |
ARRAffinitySameSite | session | This cookie is set by Windows Azure cloud, and is used for load balancing to make sure the visitor page requests are routed to the same server in any browsing session. |
BIGipServersj02web-nginx-app_https | session | NGINX cookie |
cookielawinfo-checkbox-advertisement | 1 year | Set by the GDPR Cookie Consent plugin, this cookie is used to record the user consent for the cookies in the "Advertisement" category . |
cookielawinfo-checkbox-analytics | 11 months | This cookie is set by GDPR Cookie Consent plugin. The cookie is used to store the user consent for the cookies in the category "Analytics". |
cookielawinfo-checkbox-functional | 11 months | The cookie is set by GDPR cookie consent to record the user consent for the cookies in the category "Functional". |
cookielawinfo-checkbox-necessary | 11 months | This cookie is set by GDPR Cookie Consent plugin. The cookies is used to store the user consent for the cookies in the category "Necessary". |
cookielawinfo-checkbox-others | 11 months | This cookie is set by GDPR Cookie Consent plugin. The cookie is used to store the user consent for the cookies in the category "Other. |
cookielawinfo-checkbox-performance | 11 months | This cookie is set by GDPR Cookie Consent plugin. The cookie is used to store the user consent for the cookies in the category "Performance". |
CookieLawInfoConsent | 1 year | Records the default button state of the corresponding category & the status of CCPA. It works only in coordination with the primary cookie. |
elementor | never | This cookie is used by the website's WordPress theme. It allows the website owner to implement or change the website's content in real-time. |
JSESSIONID | session | New Relic uses this cookie to store a session identifier so that New Relic can monitor session counts for an application. |
viewed_cookie_policy | 11 months | The cookie is set by the GDPR Cookie Consent plugin and is used to store whether or not user has consented to the use of cookies. It does not store any personal data. |
Cookie | Duration | Description |
---|---|---|
__cf_bm | 30 minutes | This cookie, set by Cloudflare, is used to support Cloudflare Bot Management. |
_biz_nA | 1 year | Bizible sets this cookie to remember users’ settings as well as for authentication and analytics. |
_biz_uid | 1 year | This cookie is set by Bizible, to store user id on the current domain. |
_hjAbsoluteSessionInProgress | 30 minutes | Hotjar sets this cookie to detect a user's first pageview session, which is a True/False flag set by the cookie. |
_mkto_trk | 2 years | This cookie is set by Marketo. This allows a website to track visitor behavior on the sites on which the cookie is installed and to link a visitor to the recipient of an email marketing campaign, to measure campaign effectiveness. Tracking is performed anonymously until a user self-identifies by submitting a form. |
bcookie | 1 year | LinkedIn sets this cookie from LinkedIn share buttons and ad tags to recognize browser ID. |
bscookie | 1 year | LinkedIn sets this cookie to store performed actions on the website. |
doc_langsBB | 1 year | Documentation system cookie |
doc_version | 1 year | Documentation system cookie |
lang | session | LinkedIn sets this cookie to remember a user's language setting. |
lidc | 1 day | LinkedIn sets the lidc cookie to facilitate data center selection. |
UserMatchHistory | 1 month | LinkedIn sets this cookie for LinkedIn Ads ID syncing. |
whova_client_id | 10 years | Event agenda system cookie |
Cookie | Duration | Description |
---|---|---|
_gat_UA-5909815-1 | 1 minute | A variation of the _gat cookie set by Google Analytics and Google Tag Manager to allow website owners to track visitor behaviour and measure site performance. The pattern element in the name contains the unique identity number of the account or website it relates to. |
Cookie | Duration | Description |
---|---|---|
_an_uid | 7 days | 6Sense Cookie |
_BUID | 1 year | This cookie, set by Bizible, is a universal user id to identify the same user across multiple clients’ domains. |
_ga | 2 years | The _ga cookie, installed by Google Analytics, calculates visitor, session and campaign data and also keeps track of site usage for the site's analytics report. The cookie stores information anonymously and assigns a randomly generated number to recognize unique visitors. |
_ga_* | 1 year 1 month 4 days | Google Analytics sets this cookie to store and count page views. |
_gat_UA-* | 1 minute | Google Analytics sets this cookie for user behaviour tracking. |
_gcl_au | 3 months | Provided by Google Tag Manager to experiment advertisement efficiency of websites using their services. |
_gd_session | 4 hours | This cookie is used for collecting information on users visit to the website. It collects data such as total number of visits, average time spent on the website and the pages loaded. |
_gd_visitor | 2 years | This cookie is used for collecting information on the users visit such as number of visits, average time spent on the website and the pages loaded for displaying targeted ads. |
_gid | 1 day | Installed by Google Analytics, _gid cookie stores information on how visitors use a website, while also creating an analytics report of the website's performance. Some of the data that are collected include the number of visitors, their source, and the pages they visit anonymously. |
_hjFirstSeen | 30 minutes | Hotjar sets this cookie to identify a new user’s first session. It stores the true/false value, indicating whether it was the first time Hotjar saw this user. |
_hjIncludedInSessionSample_* | 2 minutes | Hotjar cookie that is set to determine if a user is included in the data sampling defined by a site's daily session limit. |
_hjRecordingEnabled | never | Hotjar sets this cookie when a Recording starts and is read when the recording module is initialized, to see if the user is already in a recording in a particular session. |
_hjRecordingLastActivity | never | Hotjar sets this cookie when a user recording starts and when data is sent through the WebSocket. |
_hjSession_* | 30 minutes | Hotjar cookie that is set when a user first lands on a page with the Hotjar script. It is used to persist the Hotjar User ID, unique to that site on the browser. This ensures that behavior in subsequent visits to the same site will be attributed to the same user ID. |
_hjSessionUser_* | 1 year | Hotjar cookie that is set when a user first lands on a page with the Hotjar script. It is used to persist the Hotjar User ID, unique to that site on the browser. This ensures that behavior in subsequent visits to the same site will be attributed to the same user ID. |
_hjTLDTest | session | To determine the most generic cookie path that has to be used instead of the page hostname, Hotjar sets the _hjTLDTest cookie to store different URL substring alternatives until it fails. |
6suuid | 2 years | 6Sense Cookie |
AnalyticsSyncHistory | 1 month | LinkedIn cookie |
BE_CLA3 | 1 year 1 month 4 days | BrightEdge sets this cookie to enable data aggregation, analysis and report creation to assess marketing effectiveness and provide solutions for SEO, SEM and website performance. |
CONSENT | 2 years | YouTube sets this cookie via embedded youtube-videos and registers anonymous statistical data. |
dj | 10 years | DemandJump cookie |
djaimid.a28e | 2 years | DemandJump cookiean |
djaimses.a28e | 30 minutes | DemandJump cookie |
li_gc | 5 months 27 days | LinkedIn Cookie |
ln_or | 1 day | LinkedIn Cookie |
vuid | 2 years | Vimeo installs this cookie to collect tracking information by setting a unique ID to embed videos to the website. |
Cookie | Duration | Description |
---|---|---|
__adroll | 1 year 1 month | This cookie is set by AdRoll to identify users across visits and devices. It is used by real-time bidding for advertisers to display relevant advertisements. |
__adroll_fpc | 1 year | AdRoll sets this cookie to target users with advertisements based on their browsing behaviour. |
__adroll_shared | 1 year 1 month | Adroll sets this cookie to collect information on users across different websites for relevant advertising. |
__ar_v4 | 1 year | This cookie is set under the domain DoubleClick, to place ads that point to the website in Google search results and to track conversion rates for these ads. |
_fbp | 3 months | This cookie is set by Facebook to display advertisements when either on Facebook or on a digital platform powered by Facebook advertising, after visiting the website. |
_te_ | session | Adroll cookie |
fr | 3 months | Facebook sets this cookie to show relevant advertisements to users by tracking user behaviour across the web, on sites that have Facebook pixel or Facebook social plugin. |
IDE | 1 year 24 days | Google DoubleClick IDE cookies are used to store information about how the user uses the website to present them with relevant ads and according to the user profile. |
li_sugr | 3 months | LinkedIn sets this cookie to collect user behaviour data to optimise the website and make advertisements on the website more relevant. |
test_cookie | 15 minutes | The test_cookie is set by doubleclick.net and is used to determine if the user's browser supports cookies. |
VISITOR_INFO1_LIVE | 5 months 27 days | A cookie set by YouTube to measure bandwidth that determines whether the user gets the new or old player interface. |
YSC | session | YSC cookie is set by Youtube and is used to track the views of embedded videos on Youtube pages. |
yt-remote-connected-devices | never | YouTube sets this cookie to store the video preferences of the user using embedded YouTube video. |
yt-remote-device-id | never | YouTube sets this cookie to store the video preferences of the user using embedded YouTube video. |
yt.innertube::nextId | never | This cookie, set by YouTube, registers a unique ID to store data on what videos from YouTube the user has seen. |
yt.innertube::requests | never | This cookie, set by YouTube, registers a unique ID to store data on what videos from YouTube the user has seen. |