我正在使用一些参数编写 SQL 查询创建器。在 Java 中,只需检查当前数组位置和数组长度,就可以很容易地从 for 循环中检测数组的最后一个元素。
for(int i=0; i< arr.length;i++){
boolean isLastElem = i== (arr.length -1) ? true : false;
}
在 PHP 中,它们具有访问数组的非整数索引。因此,您必须使用 foreach 循环遍历数组。当您需要做出一些决定(在我的情况下,在构建查询时附加或/和参数)时,这会成为问题。
我确信必须有一些标准的方法来做到这一点。
你如何在 PHP 中解决这个问题?
听起来你想要这样的东西:
$numItems = count($arr);
$i = 0;
foreach($arr as $key=>$value) {
if(++$i === $numItems) {
echo "last index!";
}
}
话虽如此,您不必在 php.ini 中使用 foreach
迭代“数组”。
您可以使用 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
}
}
$lastKey = array_search(end($array), $array);
注意:这不起作用,因为调用 next() 会推进数组指针,因此您将跳过循环中的所有其他元素
为什么这么复杂?
foreach($input as $key => $value) {
$ret .= "$value";
if (next($input)==true) $ret .= ",";
}
这将在除最后一个值之外的每个值后面添加一个 , !
当 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);
$lastValue = array_pop($array);
谢谢。
已经有很多答案了,但也值得研究一下迭代器,尤其是当它被要求提供一种标准方式时:
$arr = range(1, 3);
$it = new CachingIterator(new ArrayIterator($arr));
foreach($it as $key => $value)
{
if (!$it->hasNext()) echo 'Last:';
echo $value, "\n";
}
您可能会发现某些东西在其他情况下也更灵活。
i=0;
和 ++i;
在 PHP 之类的脚本语言中似乎总是很老套。
一种方法是检测迭代器是否具有 next
。如果没有 next 附加到迭代器,则意味着您处于最后一个循环中。
foreach ($some_array as $element) {
if(!next($some_array)) {
// This is the last $element
}
}
自 PHP 7.3 起:
您可以使用 array_key_last($array)
获取数组最后一个键的值并将其与当前键进行比较:
$last_key = array_key_last($array);
foreach ($array as $key => $value) {
if ($key == $last_key) {
// last element
} else {
// not last element
}
}
从 foreach 数组中获取第一个和最后一个元素
foreach($array as $value) {
if ($value === reset($array)) {
echo 'FIRST ELEMENT!';
}
if ($value === end($array)) {
echo 'LAST ITEM!';
}
}
因此,如果您的数组具有唯一的数组值,那么确定最后一次迭代是微不足道的:
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!';
}
还要注意严格的比较运算符,这在这种情况下非常重要。
您仍然可以将该方法与关联数组一起使用:
$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;
}
假设您将数组存储在变量中...
foreach($array as $key=>$value)
{
echo $value;
if($key != count($array)-1) { echo ", "; }
}
echo implode(",", $array)
不要在最后一个值之后添加逗号:
数组:
$data = ['lorem', 'ipsum', 'dolor', 'sit', 'amet'];
功能:
$result = "";
foreach($data as $value) {
$result .= (next($data)) ? "$value, " : $value;
}
结果:
print $result;
lorem, ipsum, dolor, 坐, amet
如果您需要为除第一个或最后一个元素之外的每个元素执行某些操作,并且仅当数组中有多个元素时,我更喜欢以下解决方案。
我知道上面有很多解决方案,并且比我早几个月/一年发布,但我觉得这本身就相当优雅。检查每个循环也是一个布尔检查,而不是数字“i =(count-1)”检查,这可能会减少开销。
循环的结构可能会觉得别扭,但你可以将其与 HTML 表格标签中的 thead(开始)、tfoot(结束)、tbody(当前)的顺序进行比较。
$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;
}
例如,在 Web 开发术语中,如果您想为无序列表 (ul) 中除最后一个元素之外的每个元素添加一个边框底部,那么您可以为除第一个元素之外的每个元素添加一个边框顶部(CSS: IE7+ 和 Firefox/Webkit 支持的 first-child 支持此逻辑,而 IE7 不支持 :last-child)。
您也可以随意为每个嵌套循环重用 $first 变量,并且一切都会正常工作,因为在第一次迭代的第一个过程中每个循环都会使 $first 错误(因此中断/异常不会导致问题) .
$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";
}
}
参考:Link
我有一种强烈的感觉,在这个“XY 问题”的根源上,OP 只需要 implode()
函数。
因为您寻找 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;
}
}
你可以做一个计数()。
for ($i=0;$i<count(arr);$i++){
$i == count(arr)-1 ? true : false;
}
或者如果你只寻找最后一个元素,你可以使用 end()。
end(arr);
只返回最后一个元素。
而且,事实证明,您可以按整数索引 php 数组。它非常满意
arr[1];
$a = array(0=>'A', 2=>'B', 'aaa'=>'C')
。如果您访问 $a[count($a)-1]
,您会得到什么?
你也可以这样做:
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;
}
这是我的解决方案:只需获取数组的计数,减 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>";
}
如果它是一维数组,您可以这样做以保持简短和甜蜜:
foreach($items as $idx => $item) {
if (!isset($items[$idx+1])) {
print "I am last";
}
}
$array = array("dog", "rabbit", "horse", "rat", "cat");
foreach($array as $index => $animal) {
if ($index === array_key_first($array))
echo $animal; // output: dog
if ($index === array_key_last($array))
echo $animal; // output: cat
}
这是另一种方法:
$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++;
一个自制的小基准测试显示以下内容:
test1:100000 次运行模型 morg
时间: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);
不定期副业成功案例分享
$numItems = count($arr)
技巧是不需要的,它会降低可读性 - 在 PHP 中,每次访问 count($arr) 都不会造成性能损失。原因是项目计数在内部保存为数组标题中的特殊字段,并且不是即时计算的。这个技巧来自其他语言(C,Java?,...)。array_key_last($array)
获取数组的最后一个键的值