<?xml version="1.0" encoding="UTF-8"?>
<rss xmlns:content="http://purl.org/rss/1.0/modules/content/" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns:taxo="http://purl.org/rss/1.0/modules/taxonomy/" version="2.0">
  <channel>
    <title>topic Re: Using Proc Optmodel for Set Covering Problem in Mathematical Optimization, Discrete-Event Simulation, and OR</title>
    <link>https://communities.sas.com/t5/Mathematical-Optimization/Using-Proc-Optmodel-for-Set-Covering-Problem/m-p/730797#M3381</link>
    <description>&lt;P&gt;Please see if the following does what you want:&lt;/P&gt;
&lt;PRE&gt;&lt;CODE class=" language-sas"&gt;proc optmodel;
   set &amp;lt;num,num&amp;gt; ORIGINS_DESTINATIONS;
   read data have into ORIGINS_DESTINATIONS=[originid destinationid];

   set ORIGINS;
   num capacity {ORIGINS};
   read data have into ORIGINS=[originid] capacity;

   set DESTINATIONS;
   num demand {DESTINATIONS};
   read data have into DESTINATIONS=[destinationid] demand=pop10;

   var IsOpen {ORIGINS} binary;
   var Serve {ORIGINS_DESTINATIONS} binary;

   con Cover {d in DESTINATIONS}:
      sum {&amp;lt;o,(d)&amp;gt; in ORIGINS_DESTINATIONS} Serve[o,d] &amp;gt;= 1;
   con CapacityCon {o in ORIGINS}:
      sum {&amp;lt;(o),d&amp;gt; in ORIGINS_DESTINATIONS} demand[d] * Serve[o,d] &amp;lt;= capacity[o] * IsOpen[o];

   /* primary objective: minimize number of open origins */
   min NumOpen = sum {o in ORIGINS} IsOpen[o];

   /* secondary objective: maximize total coverage of demands */
   max TotalCoverage = sum {&amp;lt;o,d&amp;gt; in ORIGINS_DESTINATIONS} demand[d] * Serve[o,d];

   solve obj NumOpen;
   print NumOpen TotalCoverage;
   print Serve;
   print IsOpen {o in ORIGINS} (sum {&amp;lt;(o),d&amp;gt; in ORIGINS_DESTINATIONS} demand[d] * Serve[o,d]);

   num minNumOpen;
   minNumOpen = NumOpen.sol;
   con ObjectiveCut:
      NumOpen &amp;lt;= minNumOpen;

   solve obj TotalCoverage;
   print NumOpen TotalCoverage;
   print Serve;
   print IsOpen {o in ORIGINS} (sum {&amp;lt;(o),d&amp;gt; in ORIGINS_DESTINATIONS} demand[d] * Serve[o,d]);
quit;&lt;/CODE&gt;&lt;/PRE&gt;
&lt;DIV class="branch"&gt;
&lt;DIV&gt;
&lt;DIV align="center"&gt;
&lt;TABLE class="table" summary="Procedure Optmodel: Print Table" frame="box" rules="all" cellspacing="0" cellpadding="5"&gt;
&lt;THEAD&gt;
&lt;TR&gt;
&lt;TH class="r b header" scope="col"&gt;NumOpen&lt;/TH&gt;
&lt;TH class="r b header" scope="col"&gt;TotalCoverage&lt;/TH&gt;
&lt;/TR&gt;
&lt;/THEAD&gt;
&lt;TBODY&gt;
&lt;TR&gt;
&lt;TD class="r data"&gt;2&lt;/TD&gt;
&lt;TD class="r data"&gt;41&lt;/TD&gt;
&lt;/TR&gt;
&lt;/TBODY&gt;
&lt;/TABLE&gt;
&lt;/DIV&gt;
&lt;/DIV&gt;
&lt;BR /&gt;&lt;A name="IDX1204" target="_blank"&gt;&lt;/A&gt;
&lt;DIV&gt;
&lt;DIV align="center"&gt;
&lt;TABLE class="table" summary="Procedure Optmodel: Serve" frame="box" rules="all" cellspacing="0" cellpadding="5"&gt;&lt;COLGROUP&gt; &lt;COL /&gt;&lt;/COLGROUP&gt; &lt;COLGROUP&gt; &lt;COL /&gt; &lt;COL /&gt; &lt;COL /&gt; &lt;COL /&gt;&lt;/COLGROUP&gt;
&lt;THEAD&gt;
&lt;TR&gt;
&lt;TH class="c b header" colspan="5" scope="colgroup"&gt;Serve&lt;/TH&gt;
&lt;/TR&gt;
&lt;TR&gt;
&lt;TH class="c headerempty" scope="col"&gt;&amp;nbsp;&lt;/TH&gt;
&lt;TH class="r b header" scope="col"&gt;1&lt;/TH&gt;
&lt;TH class="r b header" scope="col"&gt;2&lt;/TH&gt;
&lt;TH class="r b header" scope="col"&gt;3&lt;/TH&gt;
&lt;TH class="r b header" scope="col"&gt;4&lt;/TH&gt;
&lt;/TR&gt;
&lt;/THEAD&gt;
&lt;TBODY&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;1&lt;/TH&gt;
&lt;TD class="r data"&gt;1&lt;/TD&gt;
&lt;TD class="r data"&gt;1&lt;/TD&gt;
&lt;TD class="r data"&gt;&amp;nbsp;&lt;/TD&gt;
&lt;TD class="r data"&gt;&amp;nbsp;&lt;/TD&gt;
&lt;/TR&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;2&lt;/TH&gt;
&lt;TD class="r data"&gt;&amp;nbsp;&lt;/TD&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;TD class="r data"&gt;1&lt;/TD&gt;
&lt;TD class="r data"&gt;1&lt;/TD&gt;
&lt;/TR&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;3&lt;/TH&gt;
&lt;TD class="r data"&gt;&amp;nbsp;&lt;/TD&gt;
&lt;TD class="r data"&gt;&amp;nbsp;&lt;/TD&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;/TR&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;4&lt;/TH&gt;
&lt;TD class="r data"&gt;&amp;nbsp;&lt;/TD&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;TD class="r data"&gt;&amp;nbsp;&lt;/TD&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;/TR&gt;
&lt;/TBODY&gt;
&lt;/TABLE&gt;
&lt;/DIV&gt;
&lt;/DIV&gt;
&lt;BR /&gt;&lt;A name="IDX1205" target="_blank"&gt;&lt;/A&gt;
&lt;DIV&gt;
&lt;DIV align="center"&gt;
&lt;TABLE class="table" summary="Procedure Optmodel: Print Table" frame="box" rules="all" cellspacing="0" cellpadding="5"&gt;&lt;COLGROUP&gt; &lt;COL /&gt;&lt;/COLGROUP&gt; &lt;COLGROUP&gt; &lt;COL /&gt; &lt;COL /&gt;&lt;/COLGROUP&gt;
&lt;THEAD&gt;
&lt;TR&gt;
&lt;TH class="l b header" scope="col"&gt;[1]&lt;/TH&gt;
&lt;TH class="r b header" scope="col"&gt;IsOpen&lt;/TH&gt;
&lt;TH class="c headerempty" scope="col"&gt;&amp;nbsp;&lt;/TH&gt;
&lt;/TR&gt;
&lt;/THEAD&gt;
&lt;TBODY&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;1&lt;/TH&gt;
&lt;TD class="r data"&gt;1&lt;/TD&gt;
&lt;TD class="r data"&gt;20&lt;/TD&gt;
&lt;/TR&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;2&lt;/TH&gt;
&lt;TD class="r data"&gt;1&lt;/TD&gt;
&lt;TD class="r data"&gt;21&lt;/TD&gt;
&lt;/TR&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;3&lt;/TH&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;/TR&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;4&lt;/TH&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;/TR&gt;
&lt;/TBODY&gt;
&lt;/TABLE&gt;
&lt;/DIV&gt;
&lt;/DIV&gt;
&lt;/DIV&gt;</description>
    <pubDate>Thu, 01 Apr 2021 18:02:45 GMT</pubDate>
    <dc:creator>RobPratt</dc:creator>
    <dc:date>2021-04-01T18:02:45Z</dc:date>
    <item>
      <title>Using Proc Optmodel for Set Covering Problem</title>
      <link>https://communities.sas.com/t5/Mathematical-Optimization/Using-Proc-Optmodel-for-Set-Covering-Problem/m-p/730680#M3380</link>
      <description>&lt;P&gt;Hi all,&lt;/P&gt;&lt;P&gt;&amp;nbsp;&lt;/P&gt;&lt;P&gt;I am working on a set covering project where I am trying to locate facilities within a 25-minute distance from demand points. All destinations within 25-minutes of an origin were generated in GIS software external to SAS.&amp;nbsp;&lt;/P&gt;&lt;P&gt;&amp;nbsp;&lt;/P&gt;&lt;P&gt;So really, right now what I have is: Facility IDs, Destination ID, Destination Population, Capacity.&lt;/P&gt;&lt;P&gt;Simplified version would look like:&lt;/P&gt;&lt;P&gt;&amp;nbsp;&lt;/P&gt;&lt;P&gt;&lt;SPAN&gt;&lt;STRONG&gt;data&lt;/STRONG&gt;&lt;/SPAN&gt; have;&lt;/P&gt;&lt;P&gt;&lt;SPAN class="Apple-converted-space"&gt;&amp;nbsp;&amp;nbsp; &lt;/SPAN&gt;&lt;SPAN&gt;input&lt;/SPAN&gt; originid destinationid pop10 capacity;&lt;/P&gt;&lt;P&gt;&lt;SPAN&gt;&lt;SPAN class="Apple-converted-space"&gt;&amp;nbsp;&amp;nbsp; &lt;/SPAN&gt;&lt;/SPAN&gt;datalines&lt;SPAN&gt;;&lt;/SPAN&gt;&lt;/P&gt;&lt;P&gt;1 1 15 22&lt;/P&gt;&lt;P&gt;1 2 5 22&lt;/P&gt;&lt;P&gt;2 2 5 22&lt;/P&gt;&lt;P&gt;2 3 17 22&lt;/P&gt;&lt;P&gt;2 4 4 22&lt;/P&gt;&lt;P&gt;3 3 17 22&lt;/P&gt;&lt;P&gt;3 4 4 22&lt;/P&gt;&lt;P&gt;4 2 5 22&lt;/P&gt;&lt;P&gt;4 4 4 22&lt;/P&gt;&lt;P&gt;;&lt;/P&gt;&lt;P&gt;&amp;nbsp;&lt;/P&gt;&lt;P&gt;What I want would be for origin 1 and 3 to be selected since that covers all points 1-4 without exceeding the capacity of 22, while excluding origin id 2, since placement there would exceed 22, and 4, since 1 and 3 serve more while still covering each demand point. Overlap is okay.&amp;nbsp;&lt;/P&gt;&lt;P&gt;&amp;nbsp;&lt;/P&gt;&lt;P&gt;At the end of the day, I'm starting to think this may be a data prep problem on my end. This is how I've been conceptualizing it, but I'm not sure if my have data set is telling the same story to the computer:&lt;/P&gt;&lt;P&gt;&lt;span class="lia-inline-image-display-wrapper lia-image-align-inline" image-alt="mhflah_0-1617283378638.png" style="width: 400px;"&gt;&lt;img src="https://communities.sas.com/t5/image/serverpage/image-id/56783i11AA342090E86D85/image-size/medium?v=v2&amp;amp;px=400" role="button" title="mhflah_0-1617283378638.png" alt="mhflah_0-1617283378638.png" /&gt;&lt;/span&gt;&lt;/P&gt;&lt;P&gt;&amp;nbsp;&lt;/P&gt;&lt;P&gt;Thank you for your time.&amp;nbsp;&lt;/P&gt;&lt;P&gt;&amp;nbsp;&lt;/P&gt;&lt;P&gt;&amp;nbsp;&lt;/P&gt;</description>
      <pubDate>Thu, 01 Apr 2021 13:27:43 GMT</pubDate>
      <guid>https://communities.sas.com/t5/Mathematical-Optimization/Using-Proc-Optmodel-for-Set-Covering-Problem/m-p/730680#M3380</guid>
      <dc:creator>mhflah</dc:creator>
      <dc:date>2021-04-01T13:27:43Z</dc:date>
    </item>
    <item>
      <title>Re: Using Proc Optmodel for Set Covering Problem</title>
      <link>https://communities.sas.com/t5/Mathematical-Optimization/Using-Proc-Optmodel-for-Set-Covering-Problem/m-p/730797#M3381</link>
      <description>&lt;P&gt;Please see if the following does what you want:&lt;/P&gt;
&lt;PRE&gt;&lt;CODE class=" language-sas"&gt;proc optmodel;
   set &amp;lt;num,num&amp;gt; ORIGINS_DESTINATIONS;
   read data have into ORIGINS_DESTINATIONS=[originid destinationid];

   set ORIGINS;
   num capacity {ORIGINS};
   read data have into ORIGINS=[originid] capacity;

   set DESTINATIONS;
   num demand {DESTINATIONS};
   read data have into DESTINATIONS=[destinationid] demand=pop10;

   var IsOpen {ORIGINS} binary;
   var Serve {ORIGINS_DESTINATIONS} binary;

   con Cover {d in DESTINATIONS}:
      sum {&amp;lt;o,(d)&amp;gt; in ORIGINS_DESTINATIONS} Serve[o,d] &amp;gt;= 1;
   con CapacityCon {o in ORIGINS}:
      sum {&amp;lt;(o),d&amp;gt; in ORIGINS_DESTINATIONS} demand[d] * Serve[o,d] &amp;lt;= capacity[o] * IsOpen[o];

   /* primary objective: minimize number of open origins */
   min NumOpen = sum {o in ORIGINS} IsOpen[o];

   /* secondary objective: maximize total coverage of demands */
   max TotalCoverage = sum {&amp;lt;o,d&amp;gt; in ORIGINS_DESTINATIONS} demand[d] * Serve[o,d];

   solve obj NumOpen;
   print NumOpen TotalCoverage;
   print Serve;
   print IsOpen {o in ORIGINS} (sum {&amp;lt;(o),d&amp;gt; in ORIGINS_DESTINATIONS} demand[d] * Serve[o,d]);

   num minNumOpen;
   minNumOpen = NumOpen.sol;
   con ObjectiveCut:
      NumOpen &amp;lt;= minNumOpen;

   solve obj TotalCoverage;
   print NumOpen TotalCoverage;
   print Serve;
   print IsOpen {o in ORIGINS} (sum {&amp;lt;(o),d&amp;gt; in ORIGINS_DESTINATIONS} demand[d] * Serve[o,d]);
quit;&lt;/CODE&gt;&lt;/PRE&gt;
&lt;DIV class="branch"&gt;
&lt;DIV&gt;
&lt;DIV align="center"&gt;
&lt;TABLE class="table" summary="Procedure Optmodel: Print Table" frame="box" rules="all" cellspacing="0" cellpadding="5"&gt;
&lt;THEAD&gt;
&lt;TR&gt;
&lt;TH class="r b header" scope="col"&gt;NumOpen&lt;/TH&gt;
&lt;TH class="r b header" scope="col"&gt;TotalCoverage&lt;/TH&gt;
&lt;/TR&gt;
&lt;/THEAD&gt;
&lt;TBODY&gt;
&lt;TR&gt;
&lt;TD class="r data"&gt;2&lt;/TD&gt;
&lt;TD class="r data"&gt;41&lt;/TD&gt;
&lt;/TR&gt;
&lt;/TBODY&gt;
&lt;/TABLE&gt;
&lt;/DIV&gt;
&lt;/DIV&gt;
&lt;BR /&gt;&lt;A name="IDX1204" target="_blank"&gt;&lt;/A&gt;
&lt;DIV&gt;
&lt;DIV align="center"&gt;
&lt;TABLE class="table" summary="Procedure Optmodel: Serve" frame="box" rules="all" cellspacing="0" cellpadding="5"&gt;&lt;COLGROUP&gt; &lt;COL /&gt;&lt;/COLGROUP&gt; &lt;COLGROUP&gt; &lt;COL /&gt; &lt;COL /&gt; &lt;COL /&gt; &lt;COL /&gt;&lt;/COLGROUP&gt;
&lt;THEAD&gt;
&lt;TR&gt;
&lt;TH class="c b header" colspan="5" scope="colgroup"&gt;Serve&lt;/TH&gt;
&lt;/TR&gt;
&lt;TR&gt;
&lt;TH class="c headerempty" scope="col"&gt;&amp;nbsp;&lt;/TH&gt;
&lt;TH class="r b header" scope="col"&gt;1&lt;/TH&gt;
&lt;TH class="r b header" scope="col"&gt;2&lt;/TH&gt;
&lt;TH class="r b header" scope="col"&gt;3&lt;/TH&gt;
&lt;TH class="r b header" scope="col"&gt;4&lt;/TH&gt;
&lt;/TR&gt;
&lt;/THEAD&gt;
&lt;TBODY&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;1&lt;/TH&gt;
&lt;TD class="r data"&gt;1&lt;/TD&gt;
&lt;TD class="r data"&gt;1&lt;/TD&gt;
&lt;TD class="r data"&gt;&amp;nbsp;&lt;/TD&gt;
&lt;TD class="r data"&gt;&amp;nbsp;&lt;/TD&gt;
&lt;/TR&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;2&lt;/TH&gt;
&lt;TD class="r data"&gt;&amp;nbsp;&lt;/TD&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;TD class="r data"&gt;1&lt;/TD&gt;
&lt;TD class="r data"&gt;1&lt;/TD&gt;
&lt;/TR&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;3&lt;/TH&gt;
&lt;TD class="r data"&gt;&amp;nbsp;&lt;/TD&gt;
&lt;TD class="r data"&gt;&amp;nbsp;&lt;/TD&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;/TR&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;4&lt;/TH&gt;
&lt;TD class="r data"&gt;&amp;nbsp;&lt;/TD&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;TD class="r data"&gt;&amp;nbsp;&lt;/TD&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;/TR&gt;
&lt;/TBODY&gt;
&lt;/TABLE&gt;
&lt;/DIV&gt;
&lt;/DIV&gt;
&lt;BR /&gt;&lt;A name="IDX1205" target="_blank"&gt;&lt;/A&gt;
&lt;DIV&gt;
&lt;DIV align="center"&gt;
&lt;TABLE class="table" summary="Procedure Optmodel: Print Table" frame="box" rules="all" cellspacing="0" cellpadding="5"&gt;&lt;COLGROUP&gt; &lt;COL /&gt;&lt;/COLGROUP&gt; &lt;COLGROUP&gt; &lt;COL /&gt; &lt;COL /&gt;&lt;/COLGROUP&gt;
&lt;THEAD&gt;
&lt;TR&gt;
&lt;TH class="l b header" scope="col"&gt;[1]&lt;/TH&gt;
&lt;TH class="r b header" scope="col"&gt;IsOpen&lt;/TH&gt;
&lt;TH class="c headerempty" scope="col"&gt;&amp;nbsp;&lt;/TH&gt;
&lt;/TR&gt;
&lt;/THEAD&gt;
&lt;TBODY&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;1&lt;/TH&gt;
&lt;TD class="r data"&gt;1&lt;/TD&gt;
&lt;TD class="r data"&gt;20&lt;/TD&gt;
&lt;/TR&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;2&lt;/TH&gt;
&lt;TD class="r data"&gt;1&lt;/TD&gt;
&lt;TD class="r data"&gt;21&lt;/TD&gt;
&lt;/TR&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;3&lt;/TH&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;/TR&gt;
&lt;TR&gt;
&lt;TH class="l rowheader" scope="row"&gt;4&lt;/TH&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;TD class="r data"&gt;0&lt;/TD&gt;
&lt;/TR&gt;
&lt;/TBODY&gt;
&lt;/TABLE&gt;
&lt;/DIV&gt;
&lt;/DIV&gt;
&lt;/DIV&gt;</description>
      <pubDate>Thu, 01 Apr 2021 18:02:45 GMT</pubDate>
      <guid>https://communities.sas.com/t5/Mathematical-Optimization/Using-Proc-Optmodel-for-Set-Covering-Problem/m-p/730797#M3381</guid>
      <dc:creator>RobPratt</dc:creator>
      <dc:date>2021-04-01T18:02:45Z</dc:date>
    </item>
    <item>
      <title>Re: Using Proc Optmodel for Set Covering Problem</title>
      <link>https://communities.sas.com/t5/Mathematical-Optimization/Using-Proc-Optmodel-for-Set-Covering-Problem/m-p/730839#M3382</link>
      <description>This is exactly what I was trying to do.&lt;BR /&gt;&lt;BR /&gt;Thank you, Rob!</description>
      <pubDate>Thu, 01 Apr 2021 20:14:01 GMT</pubDate>
      <guid>https://communities.sas.com/t5/Mathematical-Optimization/Using-Proc-Optmodel-for-Set-Covering-Problem/m-p/730839#M3382</guid>
      <dc:creator>mhflah</dc:creator>
      <dc:date>2021-04-01T20:14:01Z</dc:date>
    </item>
  </channel>
</rss>

