Find minutes working alone [closed]

2019-09-27 14:34发布

I have list of times for staff. I need to find out if any of the staff was working alone and how many minutes they were working alone for the day

| staff| start | end   |
|:---  |:---   |:---   |
| 1    | 11:05 | 20:00 | 
| 2    | 11:00 | 17:00 |
| 3    | 19:00 | 03:00 |
| 4    | 13:00 | 20:00 |
| 5    | 19:00 | 03:00 |

With Andrea help, following is the code that gets the first and last person who was working alone with alone minutes, but its not quiet right. Because if there was 3 people with different times that worked alone, it will give problem.

$staff = array(1,2,3,4,5);
$start = array("11:05", "11:00", "19:00", "13:00", "19:00");
$end = array("20:00", "17:00", "03:00", "20:00", "03:05");

array_multisort($start, $end, $staff);

$aloneStart = (strtotime($start[1]) - strtotime($start[0]))/60; // first and second items are the ones that may be working alone at start
$aloneEnd = (strtotime($end[count($end)-1]) - strtotime($end[count($end)-2]))/60; // last and second to last are the ones that may be working alone at end
if($aloneStart > 0){
  $staffAloneStart = $staff[0]; //must be the first who worked alone  
  echo "minutes alone at start: " . $aloneStart . " and it was " . $staffAloneStart . "\n";
} 
if($aloneEnd > 0){
  $staffAloneEnd = $staff[count($end)-1]; // must be the last to end that worked alone  
  echo "minutes alone at end: " . $aloneEnd . " and it was " . $staffAloneEnd . "\n";
} 
$aloneTime = intval($aloneStart)+intval($aloneEnd);
echo "total time alone " . $aloneTime;

with following array, you will see the minutes for first user needs to be more then 5 minutes, because he is working alone more at evening.

$staff = array(1, 2, 3, 4, 5);
$start = array("11:05", "11:10", "19:00", "13:00", "19:00");
$end = array("20:00", "17:00", "03:00", "16:00", "03:00");

2条回答
劫难
2楼-- · 2019-09-27 14:57

Got it!

It took some time but I found a solution.
It does have one extra item in the array but since the duration is zero it shouldn't cause an issue.

$staff = array(1,2,3,4,5);
$start = array("11:05", "11:00", "19:00", "13:00", "19:00");
$end = array("16:00", "17:00", "03:00", "16:00", "03:05");

// Add staff number to end of time ex 11:00 => 11:00#2
For($i=0; $i<count($start);$i++){
    $start[$i] .= "#" . $staff[$i];
    $end[$i] .= "#" . $staff[$i];

}
$t = array_merge($start,$end); // create one long array with all in and out times
sort($t);

// Multisport is needed to get all arrays in time order as reference
array_multisort($start, $end, $staff);

// Find first start time (11:00) and slice array thwre, build string
$test = implode(PHP_EOL,array_slice($t, array_search($start[0], $t)));

// Find the times before first start (night end times) and add them last in string
$test .= PHP_EOL . implode(PHP_EOL,array_slice($t, 0,array_search($start[0], $t)));
$times = explode(PHP_EOL, $test); // explode to make it array again
 // Var_dump($times);

$WhoIsInDaHouse = array("dummy"); // add a dummy variable since 0=false in later if
$j=0;
for($i=0; $i<count($times);$i++){
    $TimePerson = explode("#", $times[$i]);
    $Time = $TimePerson[0];
    $person = $TimePerson[1];


    $inout = array_search($person, $WhoIsInDaHouse); //is person in house and about to leave?
    If($inout != false){ //if person enter work false, if true: key of person leaving in $WhoIsInDaHouse
        //Here $person is leaving work
        Unset($WhoIsInDaHouse[$inout]);

        If(count($WhoIsInDaHouse) == 2){ // someone will now be alone since we have a dummy
            $Alone[$j]["start"] = $Time;
            $Alone[$j]["who"] = array_slice($WhoIsInDaHouse, -1)[0];
        }elseif(count($WhoIsInDaHouse) == 1 && $prevcount == 2){
            // Only dummy left
            $Alone[$j]["end"] = $Time;
            $Alone[$j]["duration"] = strtotime($Alone[$j]["end"])-strtotime($Alone[$j]["start"]);
            $j++;
        }
    }Else{
        // Here person enters work
        $WhoIsInDaHouse[] = $person;

        If(count($WhoIsInDaHouse) == 2){ // someone is entering alone
            $Alone[$j]["start"] = $Time;
            $Alone[$j]["who"] = $person;
        }elseif(count($WhoIsInDaHouse)>2 && $prevcount == 2){ // not alone anymore
            $Alone[$j]["end"] = $Time;
            $Alone[$j]["duration"] = strtotime($Alone[$j]["end"])-strtotime($Alone[$j]["start"]);
            $j++;
        }
    }
    $prevcount = count($WhoIsInDaHouse);
}
Var_dump($Alone);

And see the beauty run https://3v4l.org/dCL2H

It took me a long time to figure out I needed a dummy. Who knew a dummy could be useful?

查看更多
倾城 Initia
3楼-- · 2019-09-27 15:15

One approach would be to create a list of events

Staff   Event Time
1       S     11:05
1       E     20:00
2       S     11:00
2       E     17:00 etc.

Then sort them in time order and iterate through the events.

If a person starts a shift then add one to a staffOnShift variable, if it is an end then subtract one.

If the staffOnShift == 1 && the event is a shift start then add the minutes since the last event to the minutesAlone variable.

Hope this helps you to get started.

Additional explanation

Yes, the start and end need to be treated as separate events so after sorting you would have a list like

Staff   S/E   Time       staffOnShift       minutesAlone
2        S    11:00           1               0
1        S    11:05           2               5
4        S    13:00           3               5
2        E    17:00           2               5
3        S    19:00           3               5
5        S    19:00           4               5
1        E    20:00           3               5
4        E    20:00           2               5
3        E    03:00           1               5
5        E    03:00           0               5 + 0 = 5

Iterate through this list and add when someone starts a shift and subtract when someone’s shift ends, this will give you the number of people on shift at each event, I have put it in column 4.

If staffOnShift == 1 (before the add or subtract), add the minutes since the last event to the minutesAlone variable. (This isnt quite what I said above but the logic table helped to clarify)

With your data the only events to meet those criteria are the second and last. So 5 minutes would be added because there is 1 person on shift at the start. The criteria are met at the end as well but as there are no minutes between the last two events it would not add to the total.

查看更多
登录 后发表回答