Цикл по часам в день
Я действительно застрял в коде, и я не знаю, как его решить. Я пытаюсь подсчитать, сколько я зарабатываю в день. Проблема в том, что я зарабатываю разную сумму в зависимости от того, какой час в день это. Fx:
08.00 - 06.00 (Morning) -> 150
06.00 - 10.00 (Evening) -> 200
10.00 - 11.00 (Night) -> 250
Итак, я посчитал, сколько часов я работаю в день с этим кодом:
<p>Choose Your Work TIme</p>
<form method='post'>
<select name="s1" required>
<option>Choose Start Time</option>
<option value="8.45">8.45</option>
<option value="9.00">9.00</option>
<option value="9.15">9.15</option>
<option value="9.30">9.30</option>
<option value="9.45">9.45</option>
<option value="10.00">10.00</option>
<option value="10.15">10.15</option>
<option value="10.30">10.30</option>
<option value="10.45">10.45</option>
<option value="11.00">11.00</option>
<option value="11.15">11.15</option>
<option value="11.30">11.30</option>
<option value="11.45">11.45</option>
<option value="12.00">12.00</option>
<option value="12.15">12.15</option>
<option value="12.30">12.30</option>
<option value="12.45">12.45</option>
<option value="13.00">13.00</option>
<option value="13.15">13.15</option>
<option value="13.30">13.30</option>
<option value="13.45">13.45</option>
<option value="14.00">14.00</option>
<option value="14.15">14.15</option>
<option value="14.30">14.30</option>
<option value="14.45">14.45</option>
<option value="15.00">15.00</option>
<option value="15.15">15.15</option>
<option value="15.30">15.30</option>
<option value="15.45">15.45</option>
<option value="16.00">16.00</option>
<option value="16.15">16.15</option>
<option value="16.30">16.30</option>
<option value="16.45">16.45</option>
<option value="17.00">17.00</option>
<option value="17.15">17.15</option>
<option value="17.30">17.30</option>
<option value="17.45">17.45</option>
<option value="18.00">18.00</option>
<option value="18.15">18.15</option>
<option value="18.30">18.30</option>
<option value="18.45">18.45</option>
<option value="19.00">19.00</option>
<option value="19.15">19.15</option>
<option value="19.30">19.30</option>
<option value="19.45">19.45</option>
<option value="20.00">20.00</option>
<option value="20.15">20.15</option>
<option value="20.30">20.30</option>
<option value="20.45">20.45</option>
<option value="21.00">21.00</option>
<option value="21.15">21.15</option>
<option value="21.30">21.30</option>
<option value="21.45">21.45</option>
<option value="22.00">22.00</option>
<option value="22.15">22.15</option>
<option value="22.30">22.30</option>
<option value="22.45">22.45</option>
<option value="23.00">23.00</option>
<option value="23.15">23.15</option>
<option value="23.30">23.30</option>
</select>
<select name="s2" required>
<option>Choose End Time</option>
<option value="8.45">8.45</option>
<option value="9.00">9.00</option>
<option value="9.15">9.15</option>
<option value="9.30">9.30</option>
<option value="9.45">9.45</option>
<option value="10.00">10.00</option>
<option value="10.15">10.15</option>
<option value="10.30">10.30</option>
<option value="10.45">10.45</option>
<option value="11.00">11.00</option>
<option value="11.15">11.15</option>
<option value="11.30">11.30</option>
<option value="11.45">11.45</option>
<option value="12.00">12.00</option>
<option value="12.15">12.15</option>
<option value="12.30">12.30</option>
<option value="12.45">12.45</option>
<option value="13.00">13.00</option>
<option value="13.15">13.15</option>
<option value="13.30">13.30</option>
<option value="13.45">13.45</option>
<option value="14.00">14.00</option>
<option value="14.15">14.15</option>
<option value="14.30">14.30</option>
<option value="14.45">14.45</option>
<option value="15.00">15.00</option>
<option value="15.15">15.15</option>
<option value="15.30">15.30</option>
<option value="15.45">15.45</option>
<option value="16.00">16.00</option>
<option value="16.15">16.15</option>
<option value="16.30">16.30</option>
<option value="16.45">16.45</option>
<option value="17.00">17.00</option>
<option value="17.15">17.15</option>
<option value="17.30">17.30</option>
<option value="17.45">17.45</option>
<option value="18.00">18.00</option>
<option value="18.15">18.15</option>
<option value="18.30">18.30</option>
<option value="18.45">18.45</option>
<option value="19.00">19.00</option>
<option value="19.15">19.15</option>
<option value="19.30">19.30</option>
<option value="19.45">19.45</option>
<option value="20.00">20.00</option>
<option value="20.15">20.15</option>
<option value="20.30">20.30</option>
<option value="20.45">20.45</option>
<option value="21.00">21.00</option>
<option value="21.15">21.15</option>
<option value="21.30">21.30</option>
<option value="21.45">21.45</option>
<option value="22.00">22.00</option>
<option value="22.15">22.15</option>
<option value="22.30">22.30</option>
<option value="22.45">22.45</option>
<option value="23.00">23.00</option>
<option value="23.15">23.15</option>
<option value="23.30">23.30</option>
</select>
<input type="submit" name="submit" Value="Submit">
</form>
Допустим, я работаю с 10.00 утра до 11.00 ночи. Это значит, что я работаю 13 часов. Но как я могу рассчитать, сколько денег я заработаю за указанный период времени? Я пробовал следующее, но ничего не работает:
<?php
// create timeblocks:
// 8 = from 0.00 untill 8.00
// 18 = from 8.00 untill 18.00
// 24 = from 18.00 untill 0.00
$timeblocks=array(
8*60,
18*60,
24*60
);
//earnings PER 15 MINUTES in the above timeblocks
$money=array(
146/4,
204/4,
244/4
);
$earned=0;
//create a start and end time object
$start=date_create_from_format('Y-m-d H:i:s', $_POST['s1']);
$end=date_create_from_format('Y-m-d H:i:s', $_POST['s2']);
//calculate how many blocks of 15 minutes
$diff=$start->diff($end);
$steps=(($diff->d)*3600+($diff->h)*60+($diff->i))/15;
//loop every 15 minutes
for($i=0;$i<=$steps;$i++){
//start with the start date, and add 15 minutes every time
if($i!==0)$start->add(new DateInterval('PT15M'));
//calculate the time
$time=((int)($start->format('H')*60)+(int)($start->format('i')));
//look in timeblocks what block we are in and get the earnings from that block
for($block=0;$block<3;$block++){
if($time<=$timeblocks[$block]){
$earned+=$money[$block];
echo '$'.$money[$block].' earned at '.$start->format('d-m-Y H:i:s').'<br>';
break;
}
}
}
echo 'TOTAL earned: $'.$earned;
?>
Я надеюсь, что кто-нибудь может мне помочь.
С наилучшими пожеланиями Мадс
4 ответа
Я уверен, что есть лучшие решения, но это работает:-)
[редактировать], чтобы соответствовать новым требованиям ОП
//If you haven't done allready SET your timezone!!
date_default_timezone_set('Europe/Amsterdam');
// create timeblocks:
// 8 = from 0.00 untill 8.00
// 18 = from 8.00 untill 18.00
// 24 = from 18.00 untill 0.00
$timeblocks=array(
8*60,
18*60,
24*60
);
//earnings PER 15 MINUTES in the above timeblocks
$money=array(
100/4,
10/4,
50/4
);
$earned=0;
$_POST['s1']='18.30';
$_POST['s2']='8.30';
//sanitize your post: only numbers and a dot allowed
$s1=preg_replace("/[^0-9|.]/","",$_POST['s1']);
$s2=preg_replace("/[^0-9|.]/","",$_POST['s2']);
if($s1==$s2){
//no working time: do something
//NOTE: this assumes nobody working for 24 hours, because if you start at
//6pm day 1 and end 6pm on the next day (==24 hours) the times are the same
//as well.
}
//if second time smaller then first time, we are in the next day and add 24 hours
$add=0;
if($s1>$s2)$add=24;
//make hours and minutes
$s1=explode('.',substr('0'.$s1,-5));
$s2=explode('.',substr('0'.$s2,-5));
//create a start and end time object
$start=new Datetime('now');
$start->setTime($s1[0],$s1[1]);
$end=new Datetime('now');
$end->setTime($s2[0]+$add,$s2[1]);
//calculate how many blocks of 15 minutes
$diff=$start->diff($end);
$steps=(($diff->d)*3600+($diff->h)*60+($diff->i))/15;
//loop every 15 minutes
for($i=1;$i<=$steps;$i++){
//start with the start date, and add 15 minutes every time
$start->add(new DateInterval('PT15M'));
//calculate the time
$time=((int)($start->format('H')*60)+(int)($start->format('i')));
//look in timeblocks what block we are in and get the earnings from that block
for($block=0;$block<3;$block++){
if($time<=$timeblocks[$block]){
$earned+=$money[$block];
echo '$'.$money[$block].' earned at '.$start->format('d-m-Y H:i:s').'<br>';
break;
}
}
}
echo 'TOTAL earned: $'.$earned;
$begin = new DateTime( 'YYYY-MM-DD h:I' );
$end = new DateTime( 'YYYY-MM-DD h:I' );
$interval = new DateInterval('P1I');
$daterange = new DatePeriod($begin, $interval ,$end);
foreach($daterange as $date){
//each of the hours in between.
//do the checks on the hour here $date will be the variable your interested in
}
Я бы предложил использовать дату и время, когда часы переходят на летнее время и т.д.
Было бы проще изменить ваш HTML-код, заменив значения "точечных минут" на.25, .5 и.75 в value
атрибут каждого option
тег. Например:
<option value="13.75">13:45</option>
Следующий PHP-код будет делать то, что вы ожидаете
Ваш оригинальный код:
<?php
if (isset($_POST['submit'])) {
echo "Started time : " . $_POST['s1'] . "<br/>";
echo "End time : " . $_POST['s2'] . "<br/>" ;
$start_time = $_POST['s1'];
$end_time = $_POST['s2'];
$early_time_salary = 150;
$middle_time_salary = 200;
$late_time_salary = 250;
Код для добавления
$hour0=$end_time-$start_time;
if ($hour0>0) {
$hour1=($end_time>18) ? $end_time-18 : 0;
$hour2=($end_time>22) ? $end_time-22 : 0;
$tot=($hour0*150) + $hour1*50 + $hour2*50;
echo "Total earned: $tot";
} else {
echo "Error : work finished before it starts";
}
}
Я бы подошел к этому так:
- Создайте массив поиска с элементами для каждого изменения оплаты на основе времени.
- Создайте период DateTime для повторения каждые 15 минут между двумя переданными значениями времени.
- По мере повторения каждого 15-минутного интервала подсчитывайте правильную оплату и добавляйте эту сумму к общей сумме.
Код: (Демо)
$_POST['s1'] = '16.15';
$_POST['s2'] = '23.30';
$wagesLookup = [
'18:00' => 146 / 4, // morning
'22:00' => 204 / 4, // evening
'24:00' => 244 / 4 // night
];
$period = new DatePeriod(
DateTime::createFromFormat('H.i', $_POST['s1']),
new DateInterval('PT15M'),
DateTime::createFromFormat('H.i:01', "{$_POST['s2']}:01") // bump ahead to make "inclusive"
);
$total = 0;
foreach ($period as $dt) {
$time = $dt->format('H:i');
foreach ($wagesLookup as $mark => $pay) {
if ($time <= $mark) {
break;
}
}
printf("%s earned %.02f\n", $dt->format('H:i'), $pay);
$total += $pay;
}
echo "Total: $total";
Выход:
16:15 earned 36.50
16:30 earned 36.50
16:45 earned 36.50
17:00 earned 36.50
17:15 earned 36.50
17:30 earned 36.50
17:45 earned 36.50
18:00 earned 36.50
18:15 earned 51.00
18:30 earned 51.00
18:45 earned 51.00
19:00 earned 51.00
19:15 earned 51.00
19:30 earned 51.00
19:45 earned 51.00
20:00 earned 51.00
20:15 earned 51.00
20:30 earned 51.00
20:45 earned 51.00
21:00 earned 51.00
21:15 earned 51.00
21:30 earned 51.00
21:45 earned 51.00
22:00 earned 51.00
22:15 earned 61.00
22:30 earned 61.00
22:45 earned 61.00
23:00 earned 61.00
23:15 earned 61.00
Total: 1413