优化foreach数千项

时间:2014-06-18 11:38:37

标签: php memory optimization yii foreach

我在一组 25,000 结果中运行下面的代码。我需要优化,因为我达到了内存限制。

$oldproducts = Oldproduct::model()->findAll(); /*(here i have 25,000 results)*/

foreach($oldproducts as $oldproduct) :
    $criteria = new CDbCriteria;
    $criteria->compare('`someid`', $oldproduct->someid);
    $finds = Newproduct::model()->findAll($criteria);

    if (empty($finds)) {
        $new = new Newproduct;
        $new->someid = $oldproduct->someid;
        $new->save();
    } else {
        foreach($finds as $find) :
            if ($find->price != $oldproduct->price) {
                $find->attributes=array('price' => $oldproduct->price);
                $find->save();
            }
        endforeach;
    }
endforeach;

代码通过 someid 比较两个表的行。如果发现巧合,则更新 price 列,否则会创建新记录。

2 个答案:

答案 0 :(得分:5)

使用CDataProviderIterator

  

...允许对大数据集进行迭代,而无需将整个集合保存在内存中。

首先必须将CDataProvider实例传递给它:

$dataProvider = new CActiveDataProvider("Oldproduct");
$iterator = new CDataProviderIterator($dataProvider);
foreach($iterator as $item) {
    // do stuff
}

答案 1 :(得分:0)

你可以用~5000的块来处理行,而不是让1中的所有行都行!

$cnt = 5000;
$offset = 0;

do {
    $oldproducts = Oldproduct::model()->limit($cnt)->offset($offset)->findAll(); /*(here i have 25,000 results)*/

    foreach($oldproducts as $oldproduct) {
        // your code
    }

    $offset += $cnt;
} while($oldproducts >= $cnt);