PHP sayı aralıkları Hesaplamaları

3 Cevap php

ve her şeyden önce, sorumu okumak için zaman ayırdığınız için teşekkür ederiz.

Ben bir senaryo yazmaya çalışıyorum, ve ben çözmek zor bulma yaşıyorum bir sorun geldim. I numaraları (örneğin, 1000 ve 2000) bir çift çalışıyorum, ve bir çift sayı bir dizi vardır:

$pairs = array(
    array(800, 1100),
    array(1500, 1600),
    array(1900, 2100)
)

Ben bulmaya çalışıyorum ne, 1000 ve 2000 yılları arasında sayı çiftleri kapsamadığı aralıkları nasıl olduğunu. Bu örnekte 1000-1100 dizi (800, 1100) ile kaplıdır, 1500-1600 dizi (kapsamındadır 1500, 1600) ve 1900-2000 1101-1499 ve 1599-1899 kapsayacak şekilde sol ile beni bırakır,) dizi (1900, 2100 ile kaplıdır. Ben yeterince açık oluyorum umuyoruz.

Ne ben merak ediyorum bana PHP getiri değişkeni $ çiftleri kapsamadığı aralıkları bir dizi yapmak nasıl olduğunu. Bu örnekte dönecekti:

array(
    array(1101, 1499),
    array(1599, 1899)
)

Ne bunu yapmak için en iyi yol olacağını herhangi bir fikrin var mı?

Şimdiden teşekkür ederim.

3 Cevap

Peki, öncelikle sorunu tanımlamak gerekir:

  1. Çiftleri sıralanır?
  2. Çiftleri üstüste musunuz?
  3. (Bu durum gibi görünüyor), belirli bir aralık için eksik aralıkları bulmak istiyorsunuz?

Çiftleri sınıflandırılmaktadır değilseniz, önce bunları sıralamak:

usort($pairs, 'cmp_pair');

function cmp_pair($a, $b) {
  if ($a[0] == $b[0]) {
    if ($a[1] == $b[1]) {
      return 0;
    } else {
      return $a[1] < $b[1] ? -1 : 1;
    }
  } else {
    return $a[0] < $b[0] ? -1 : 1;
  }
}

Örtüşen aralıkları izin verilirse, bir örtüşmeyen dizi çiftlerinin listesini dönüşümü. İşte bunu yapmak için nasıl bir öneri:

$prev = false;
$newpairs = array();
foreach ($pairs as $pair) {
  if ($prev) {
    // this also handles the case of merging two ranges
    // eg 100-199 with 200 to 250 to 100-250
    if ($prev[1] >= $pair[0]-1) {
      $prev = array($prev[0], max($prev[1], $pair[1]));
    } else {
      $newpairs[] = $prev;
    }
  }
  $prev = $pair;
}
$pairs = $newpairs;

Sorun da sıralanmış bir dizi var gibi biraz daha basit hale gelir böylece şimdi herhangi örtüşen çift olmamalıdır.

function missing($start, $end, $pairs) {
  $missing = array();
  $prev = false;
  foreach ($pairs as $pair) {
    // if the current pair starts above the end, we're done
    if ($pair[0] > $end) {
      break;
    }

    // we can ignore any pairs that end before the start
    if ($pair[1] < $start) {
      continue;
    }

    // if the pair encompasses the whole range, nothing is missing
    if ($pair[0] <= $start && $pair[1] >= $end) {
      break;
    }

    // if this is our first overlapping pair and it starts above
    // the start we can backfill the missing range
    if ($pair[0] > $start && !$missing) {
      $missing[] = array($start, $pair[0]);
    }

    // compare this pair to the previous one (if there is one) and
    // fill in the missing range
    if ($prev) {
      $missing[] = array($prev[1]+1, $pair[0]-1);
    }

    // set the previous
    $prev = $pair;
  }

  // if this never got set the whole range is missing
  if (!$prev) {
    $missing[] = array($start, $end);

  // if the last overlapping range ended before the end then
  // we are missing a range from the end of it to the end of
  // of the relevant range
  } else if ($prev[1] < $end) {
    $missing[] = array($prev[1]+1, $end);
  }

  // done!
  return $missing;
}

Umut olur.

Ben böyle bir şey yapacağını:

begin = 1000
end   = 2000
uncovered = ()
foreach pairs as pair
  if (pair[0] > begin)
    push (uncovered, begin, pair[0])
    begin = pair[1]
  end if
end foreach

This is only an idea, but here is the point: Consider you have a big segment (from 1000 to 2000) and small one. You want to get each segments of the big one that are not covered by the small one. Imagine you have a pen!

Başlangıcını init. Sahip olduğunuz her "küçük bir kesimi" üzerinde yineleme. YOU (kesinlikle) başlangıcından sonra ise, o zaman orada bir "delik", yani mevcut segmentinin başında başlar ziyade ezberlemek gerekir.

Umarım bu yardımcı olur, ve bu doğru olduğunu!

// your original arrays of integers
$pairs = array(
    array(800, 1100),
    array(1500, 1600),
    array(1900, 2100)
);

// first, normalize the whole thing into a giant list of integers that
// are included in the array pairs, combine and sort numerically
$numbers_in_pairs = array();
foreach($pairs as $set) {
    $numbers_in_pairs = array_merge($numbers_in_pairs, range($set[0], $set[1]));
}
sort($numbers_in_pairs);

// find the min
$min = $numbers_in_pairs[0];

// find the max
$max = $numbers_in_pairs[count($numbers_in_pairs)-1];

Find the array difference

// create an array of all numbers inclusive between the min and max
$all_numbers = range($min, $max);

// the numbers NOT included in the set can be found by doing array_diff
// between the two arrays, we need to sort this to assure no errors when
// we iterate over it to get the maxes and mins
$not_in_set = array_diff($all_numbers, $numbers_in_pairs);
sort($not_in_set);

Metadata about the sets we'll use later:

// gather metadata about the numbers that are not inside the set
// $not_in_set_meta['min'] = lowest integer
// $not_in_set_meta['max'] = highest integer
// $not_in_set_meta['mins'] = min boundary integer
// $not_in_set_meta['maxes'] = max boundary integer
$not_in_set_meta = array();
for($i=0;$i<count($not_in_set);$i++) {
    if ($i == 0) {
    	$not_in_set_meta['min'] = $not_in_set[$i];
    	$not_in_set_meta['mins'][] = $not_in_set[$i];
    } else if ($i == count($not_in_set)-1 ) {
    	$not_in_set_meta['max'] = $not_in_set[$i];
    	$not_in_set_meta['maxes'][] = $not_in_set[$i];
    } else {
    	// in the event that a number stands alone
    	// that it can be BOTH the min and the max
    	if (($not_in_set[$i+1] - $not_in_set[$i]) > 1) {
    		$not_in_set_meta['maxes'][] = $not_in_set[$i];
    	}
    	if (($not_in_set[$i] - $not_in_set[$i-1]) > 1) {
    		$not_in_set_meta['mins'][] = $not_in_set[$i];
    	}
    }
}

Final output:

// The final variable which we'll dump the ranges not covered into:
$non_sets = array();

while(count($not_in_set_meta['mins']) > 0 && count($not_in_set_meta['maxes'])) {
    $non_sets[] = array(array_shift($not_in_set_meta['mins']), 
                        array_shift($not_in_set_meta['maxes']));
}
// print it out:
print var_export($non_sets);

Result:

array (
  0 => 
  array (
    0 => 1101,
    1 => 1499,
  ),
  1 => 
  array (
    0 => 1601,
    1 => 1899,
  ),
)

?>