Найти последний элемент массива при использовании цикла foreach в PHP

Я пишу SQL query creator, используя некоторые параметры. В Java очень легко обнаружить последний элемент массива изнутри цикла for, просто проверив текущую позицию массива с длиной массива.

for(int i=0; i< arr.length;i++){
     boolean isLastElem = i== (arr.length -1) ? true : false;        
}

в PHP они имеют нецелые индексы для доступа к массивам. Поэтому вы должны перебирать массив, используя цикл foreach. Это становится проблематичным, когда вам нужно принять какое-то решение (в моем случае добавить или/и параметр при построении запроса).

Я уверен, что должен быть какой-то стандартный способ сделать это.

Как вы решаете это в PHP?

29 ответов


похоже, вы хотите что-то вроде этого:

$numItems = count($arr);
$i = 0;
foreach($arr as $key=>$value) {
  if(++$i === $numItems) {
    echo "last index!";
  }
}    

это, как говорится, вам не нужно перебирать "массив", используя foreach в php.


вы можете получить значение последнего ключа массива, используя end(array_keys($array)) и сравните его с текущим ключом:

$last_key = end(array_keys($array));
foreach ($array as $key => $value) {
    if ($key == $last_key) {
        // last element
    } else {
        // not last element
    }
}

почему так сложно?

foreach($input as $key => $value) {
    $ret .= "$value";
    if (next($input)==true) $ret .= ",";
}

Это добавит a, за каждым значением, кроме последнего!


когда toEnd достигает 0, это означает, что он находится на последней итерации цикла.

$toEnd = count($arr);
foreach($arr as $key=>$value) {
  if (0 === --$toEnd) {
    echo "last index! $value";
  }
}

последнее значение все еще доступно после цикла, поэтому, если вы просто хотите использовать его для большего количества вещей после цикла, это лучше:

foreach($arr as $key=>$value) {
  //something
}
echo "last index! $key => $value";

если вы не хотите рассматривать последнее значение как специальные внутренние циклы. Это должно быть быстрее, если у вас большие массивы. (Если вы повторно используете массив после цикла внутри той же области, вам нужно "скопировать" массив сначала.)

//If you use this in a large global code without namespaces or functions then you can copy the array like this:
//$array = $originalArrayName; //uncomment to copy an array you may use after this loop

//end($array); $lastKey = key($array); //uncomment if you use the keys
$lastValue = array_pop($array);

//do something special with the last value here before you process all the others?
echo "Last is $lastValue", "\n";

foreach ($array as $key => $value) {
    //do something with all values before the last value
    echo "All except last value: $value", "\n";
}

//do something special with the last value here after you process all the others?
echo "Last is $lastValue", "\n";

и ответить на ваш исходный вопрос "в моем случае, чтобы добавить или / и параметр при построении запроса"; это будет цикл над всеми значениями, А затем объединить их вместе в строку с " и " между ними, но не до первого значения или после последнего значения:

$params = [];
foreach ($array as $value) {
    $params[] = doSomething($value);
}
$parameters = implode(" and ", $params);

ответов уже много, но стоит посмотреть и на итераторы, тем более, что его попросили стандартным способом:

$arr = range(1, 3);

$it = new CachingIterator(new ArrayIterator($arr));
foreach($it as $key => $value)
{
  if (!$it->hasNext()) echo 'Last:';
  echo $value, "\n";
}

вы можете найти что-то, что работает более гибко и для других случаев.


Итак, если Ваш массив имеет уникальные значения массива, то определение последней итерации тривиально:

foreach($array as $element) {
    if ($element === end($array))
        echo 'LAST ELEMENT!';
}

Как вы видите, это работает, если последний элемент появляется только один раз в массиве, иначе вы получаете ложную тревогу. В нем нет, вы должны сравнить ключи (которые уникальны наверняка).

foreach($array as $key => $element) {
    end($array);
    if ($key === key($array))
        echo 'LAST ELEMENT!';
}

также обратите внимание на строгий оператор coparision, что довольно важно в этом случае.


предполагая, что массив хранится в переменной...

foreach($array as $key=>$value) 
{ 
    echo $value;
    if($key != count($array)-1) { echo ", "; }
}

одним из способов может быть обнаружение, имеет ли итератор next. Если к итератору нет следующего, это означает, что вы находитесь в последнем цикле.

foreach ($some_array as $element) {
    if(!next($some_array)) {
         // This is the last $element
    }
}

Если вам нужно что-то сделать для каждого элемента, кроме первого или последнего, и только если в массиве более одного элемента, я предпочитаю следующее решение.

Я знаю, что есть много решений выше и опубликовано месяцев / за год до моего, но это то, что я чувствую, довольно элегантно само по себе. Проверка каждого цикла также является логической проверкой, а не числовой проверкой" i=(count-1)", которая может позволить меньше накладных расходов.

структура цикл может показаться неудобным, но вы можете сравнить его с порядком thead (начало), tfoot (конец), tbody (текущий) в тегах HTML-таблицы.

$first = true;
foreach($array as $key => $value) {
    if ($first) {
        $first = false;
        // Do what you want to do before the first element
        echo "List of key, value pairs:\n";
    } else {
        // Do what you want to do at the end of every element
        // except the last, assuming the list has more than one element
        echo "\n";
    }
    // Do what you want to do for the current element
    echo $key . ' => ' . $value;
}

например, в терминах веб-разработки, если вы хотите добавить border-bottom для каждого элемента, кроме последнего в неупорядоченном списке (ul), то вы можете вместо этого добавить border-top для каждого элемента, кроме первого (CSS: first-child, поддерживаемый IE7+ и Firefox / Webkit поддерживает эту логику, тогда как : last-child не поддерживается IE7).

вы можете свободно использовать переменную $first для каждого вложенного цикла, и все будет работать нормально, так как каждый цикл делает $first false во время первого процесса первой итерации (поэтому перерывы/исключения не вызовут проблем).

$first = true;
foreach($array as $key => $subArray) {
    if ($first) {
        $string = "List of key => value array pairs:\n";
        $first = false;
    } else {
        echo "\n";
    }

    $string .= $key . '=>(';
    $first = true;
    foreach($subArray as $key => $value) {
        if ($first) {
            $first = false;
        } else {
            $string .= ', ';
        }
        $string .= $key . '=>' . $value;
    }
    $string .= ')';
}
echo $string;

пример:

List of key => value array pairs:
key1=>(v1_key1=>v1_val1, v1_key2=>v1_val2)
key2=>(v2_key1=>v2_val1, v2_key2=>v2_val2, v2_key3=>v2_val3)
key3=>(v3_key1=>v3_val1)

Это должен быть простой способ найти последний элемент:

foreach ( $array as $key => $a ) {
    if ( end( array_keys( $array ) ) == $key ) {
        echo "Last element";
     } else {
        echo "Just another element";
     }
}  

ссылки : ссылке


вы все еще можете использовать этот метод с ассоциативными массивами:

$keys = array_keys($array);
for ($i = 0, $l = count($array); $i < $l; ++$i) {
    $key = $array[$i];
    $value = $array[$key];
    $isLastItem = ($i == ($l - 1));
    // do stuff
}

// or this way...

$i = 0;
$l = count($array);
foreach ($array as $key => $value) {
    $isLastItem = ($i == ($l - 1));
    // do stuff
    ++$i;
}

У меня сильное чувство, что в корне этой "проблемы XY" OP хотел просто .


поскольку ваше намерение найти массив EOF-это только для клея. Познакомьтесь с приведенной ниже тактикой. Вам не нужно требовать EOF:

$given_array = array('column1'=>'value1',
                     'column2'=>'value2',
                     'column3'=>'value3');

$glue = '';
foreach($given_array as $column_name=>$value){
    $where .= " $glue $column_name = $value"; //appending the glue
    $glue   = 'AND';
}
echo $where;

o/p:

column1 = value1 AND column2 = value2 AND column3 = value3

похоже, вы хотите что-то вроде этого:

$array = array(
    'First',
    'Second',
    'Third',
    'Last'
);

foreach($array as $key => $value)
{
    if(end($array) === $value)
    {
       echo "last index!" . $value;
    }
}

вы можете сделать count ().

for ($i=0;$i<count(arr);$i++){
    $i == count(arr)-1 ? true : false;
}

или если вы ищете только последний элемент, вы можете использовать end ().

end(arr);

возвращает только последний элемент.

и, как оказалось, Вы можете индексировать массивы php целыми числами. Он совершенно доволен

arr[1];

Как насчет использования "end"? http://php.net/manual/en/function.end.php


вы также можете сделать что-то вроде этого:

end( $elements );
$endKey = key($elements);
foreach ($elements as $key => $value)
{
     if ($key == $endKey) // -- this is the last item
     {
          // do something
     }

     // more code
}

Мне нравится следующее, Поскольку я чувствую, что это довольно аккуратно. Предположим, мы создаем строку с разделителями между всеми элементами: например, a,b, C

$first = true;
foreach ( $items as $item ) {
    $str = ($first)?$first=false:", ".$item;
}

вот еще один способ сделать это:

$arr = range(1, 10);

$end = end($arr);
reset($arr);

while( list($k, $v) = each($arr) )
{
    if( $n == $end )
    {
        echo 'last!';
    }
    else
    {
        echo sprintf('%s ', $v);
    }
}

Если я вас понимаю, то все, что вам нужно, это отменить массив и получить последний элемент с помощью команды pop:

   $rev_array = array_reverse($array);

   echo array_pop($rev_array);

вы также можете попробовать это, чтобы сделать свой запрос... показано здесь с INSERT

<?php
 $week=array('one'=>'monday','two'=>'tuesday','three'=>'wednesday','four'=>'thursday','five'=>'friday','six'=>'saturday','seven'=>'sunday');
 $keys = array_keys($week);
 $string = "INSERT INTO my_table ('";
 $string .= implode("','", $keys);
 $string .= "') VALUES ('";
 $string .= implode("','", $week);
 $string .= "');";
 echo $string;
?>

для сценариев генерации SQL-запросов или всего, что делает другое действие для первого или последнего элементов, намного быстрее (почти в два раза быстрее), чтобы избежать использования ненужных проверок переменных.

текущее принятое решение использует цикл и проверку в цикле, который будет сделан every_single_iteration, правильный (быстрый) способ сделать это следующим образом :

$numItems = count($arr);
$i=0;
$firstitem=$arr[0];
$i++;
while($i<$numItems-1){
    $some_item=$arr[$i];
    $i++;
}
$last_item=$arr[$i];
$i++;

небольшой самодельный тест показал следующее:

тест1: 100000 работает модели морг

время: 1869.3430423737 миллисекунды

test2: 100000 запусков модели, если последний

время: 3235.6359958649 миллисекунд


другой способ - вспомнить предыдущий результат цикла цикла и использовать его в качестве конечного результата:

    $result = $where = "";
    foreach ($conditions as $col => $val) {
        $result = $where .= $this->getAdapter()->quoteInto($col.' = ?', $val);
        $where .=  " AND ";
    }
    return $this->delete($result);

Я лично использую этот вид конструкции, который позволяет легко использовать элементы html и : просто измените равенство для другого свойства...

массив не может содержать ложные элементы, но все остальные элементы, которые приведены в false boolean.

$table = array( 'a' , 'b', 'c');
$it = reset($table);
while( $it !== false ) {
    echo 'all loops';echo $it;
    $nextIt = next($table);
    if ($nextIt === false || $nextIt === $it) {
            echo 'last loop or two identical items';
    }
    $it = $nextIt;
}

вы можете dirctly получить последний индекс по:

$numItems = count($arr);

echo $arr[$numItems-1];


<?php foreach($have_comments as $key => $page_comment): ?>
    <?php echo $page_comment;?>
    <?php if($key+1<count($have_comments)): ?> 
        <?php echo ', '; ?>
    <?php endif;?>
<?php endforeach;?>

вот мое решение: Просто подсчитать массива минус 1 (так как они начинаются в 0).

$lastkey = count($array) - 1;
foreach($array as $k=>$a){
    if($k==$lastkey){
        /*do something*/
    }
}

foreach ($array as $key => $value) {

  $class = ( $key !== count( $array ) -1 ) ? " class='not-last'" : " class='last'";

  echo "<div{$class}>";
  echo "$value['the_title']";
  echo "</div>";

}

ссылка


У меня есть обобщенное решение, которое я использую для общей цели компиляции строки из массива строковых значений. Все, что я делаю, это добавляю необычную строку в конец, а затем заменяю ее.

функция для возврата строки из массива, разделенной, без конечного разделителя:

function returnArraySeparated($aArr, $sSep, $sAdd = "@X@") {

$strReturn = (string) "";


# Compile the set:
foreach ($aArr as $sItem) {
    $strReturn .= $sItem . $sSep;
}

# Easy strip the end:
$strReturn = str_replace($sSep . $sAdd, "", $strReturn . $sAdd);

return $strReturn;
}

ничего особенного, но это работает :)