2016-11-28 115 views
0

我有陣列過濾多維數組

$rows = [ 
    ['k'=>1, 'dop1'=>'take', 'dop2' => 'a', 'dat' => '25-11-2016'], 
    ['k'=>2, 'dop1'=>'make', 'dop2' => 'b', 'dat' => '26-11-2016'], 
    ['k'=>3, 'dop1'=>'sake', 'dop2' => 'c', 'dat' => '27-11-2016'], 
    ['k'=>4, 'dop1'=>'bake', 'dop2' => 'd', 'dat' => '28-11-2016'], 
    ['k'=>5, 'dop1'=>'dake', 'dop2' => 'e', 'dat' => '29-11-2016'], 
    ['k'=>6, 'dop1'=>'jake', 'dop2' => 'f', 'dat' => '30-11-2016'], 
    ['k'=>7, 'dop1'=>'ake', 'dop2' => 'g', 'dat' => '24-11-2016'] 
]; 

和另一個數組用於過濾

$filters = [ 
    ['dat', '=', '27-11-2016'], 
    ['dop1', '=', "bake"], 
    ['dop1', '=', "sake"], 
    ['dop1', '=', "take"], 
]; 

如果重複陣列濾光器的第一個元素,將它們之間OR表達,否則將是AND表達式。 所以這裏的結果必須是['k'=>3, 'dop1'=>'sake', 'dop2' => 'c', 'dat' => '27-11-2016'],因爲'dat' => '27-11-2016'存在於filters數組中,並且'dop1'=>'sake''dop1'=>'sake'也存在。你能幫我嗎?

UPDATE 這是我的代碼示例,但在註釋的位置存在問題。

$dop = ['dop1', 'dop2', 'dop3', 'dop4']; 
$result = []; 

foreach ($rows as $row) { 
    foreach ($filters as $filter) { 
     if(in_array($filter[0], $dop)){ 
      $new_filter = []; 
      $f = explode("\n", $filter[2]); 
      foreach ($f as $item) { 
       array_push($new_filter, [$filter[0], $filter[1], $item]); 
      } 


      if(count($result) == 0){ 
       foreach ($new_filter as $new) { 
        if($row[$new[0]] == $new[2]){ 
         array_push($result, ['r' => $row, 'filter' => $new, 'validity' => 'valid']); 
        } 
       } 
      } else { 
       foreach ($new_filter as $new) { 
        for($i=0;$i<count($result);$i++){ 
         if($result[$i]['filter'][0] == $new[0] && $result[$i]['filter'][1] == $new[1] && $result[$i]['filter'][2] != $new[2]){ 
          //array_push($result, ['r' => $row, 'filter' => $new, 'validity' => 'valid']); 
         } 
        } 
       } 
       $invalid = 1; 
       $id = 0; 
       foreach ($new_filter as $new) { 
        for($i=0;$i<count($result);$i++){ 
         if($result[$i]['r'][$new[0]] == $new[2]){ 
          $invalid = 0; 
          $id = $i; 
          break 2; 
         } 
        } 
       } 
       if($invalid == 1){ 
        $result[$id]['validity'] = 'invalid'; 
       } 
      } 
     } elseif($filter[0] == 'dat') { 
      if(count($result) == 0){ 
       if($row[$filter[0]] == $filter[2]){ 
        array_push($result, ['r' => $row, 'filter' => $filter, 'validity' => 'valid']); 
       } 
      } else { 
       if($row[$filter[0]] == $filter[2]){ 
        for($i=0;$i<count($result);$i++){ 
         if($result[$i]['r'][$filter[0]] != $filter[2]){ 
          $result[$i]['validity'] = 'invalid'; 
         } 
        } 
       } 
      } 
     } 
    } 
} 
$c = 0; 
foreach ($result as $r) { 
    if($r['validity'] == 'valid'){ 
     $c++; 
    } 
} 
echo $c; 
+0

你可以再試一次來說明OR和AND部分......我不完全相信我明白了。你的例子預期的結果是好的......但請澄清當你是或當你是和。 – WEBjuju

+1

我不能發現你自己的嘗試,所以你自己的代碼來實現這一點。你如何期待我們幫助?請注意,我們在這裏幫助您編寫代碼,而不是爲您編寫代碼。爲此,請聘請付費程序員。 – arkascha

+0

@arkascha抱歉,忘記了我自己的代碼,我更新了問題,請幫助我。 –

回答

0

做了一些工作給你的不好的事情是我的方式需要一個eval這可能是危險的,這取決於你將在值中的字符串...

$rows = [ 
    ['k'=>1, 'dop1'=>'take', 'dop2' => 'a', 'dat' => '25-11-2016'], 
    ['k'=>2, 'dop1'=>'make', 'dop2' => 'b', 'dat' => '26-11-2016'], 
    ['k'=>3, 'dop1'=>'sake', 'dop2' => 'c', 'dat' => '27-11-2016'], 
    ['k'=>4, 'dop1'=>'bake', 'dop2' => 'd', 'dat' => '28-11-2016'], 
    ['k'=>5, 'dop1'=>'dake', 'dop2' => 'e', 'dat' => '29-11-2016'], 
    ['k'=>6, 'dop1'=>'jake', 'dop2' => 'f', 'dat' => '30-11-2016'], 
    ['k'=>7, 'dop1'=>'ake', 'dop2' => 'g', 'dat' => '24-11-2016'] 
]; 

$filters = [ 
    ['dat', '=', '27-11-2016'], 
    ['dop1', '=', "bake"], 
    ['dop1', '=', "sake"], 
    ['dop1', '=', "take"], 
]; 


function marrfilter($filters_arr, $rows) 
{ 
    /* 
    * Building filters to eval 
    */ 
    $buildFilter = function($key, $op, $val) 
    { 
     switch($op) 
     { 
      case '=': 
       $op = '=='; 
       break; 
      case '==': 
      case '===': 
      case '!=': 
      case '!==': 
      case '>': 
      case '>=': 
      case '<': 
      case '<=': 
       break; 

      default: 
       throw new Exception('Unknown operand'); 
     } 

     $val = addslashes($val); 

     return '$val ' . $op . " '" . $val . "'"; 
    }; 

    $filters = array(); 
    foreach($filters_arr as $k => $v) 
    { 
     list($key, $op, $val) = $v; 

     if(!isset($filters[$key])) 
     { 
      $filters[$key] = $buildFilter($key, $op, $val); 
     } 
     else 
     { 
      if(!is_array($filters[$key])) 
      { 
       $filters[$key] = array($filters[$key]); 
      } 

      $filters[$key][] = $buildFilter($key, $op, $val); 
     } 
    } 

    /* 
    * Filter function 
    */ 
    $applyFilter = function($row, $fk, $fv) 
    { 
     if(!array_key_exists($fk, $row)) 
     { 
      return false; 
     } 

     $val = $row[$fk]; 

     if(is_array($fv)) 
     { 
      $test = false; 
      foreach($fv as $ft) 
      { 
       eval('$test = (' . $ft . ');'); 

       if($test) return true; 
      } 

      return false; 
     } 
     else 
     { 
      $test = false; 
      // thats why you realy have to care about what you put in filters and data :) 
      eval('$test = (' . $fv . ');'); 

      if($test) return true; 
     } 

     return false; 
    }; 

    /* 
    * Filtering 
    */ 
    $result = array(); 
    foreach($rows as $row) 
    { 
     $r = array(); 
     foreach($filters as $fk => $fv) 
     { 
      $r[] = $applyFilter($row, $fk, $fv); 
     } 

     $r = array_filter($r); 

     if(count($r) == count($filters)) 
     { 
      $result[] = $row; 
     } 
    } 

    return $result; 
} 

這裏工作很好。

見在線:http://sandbox.onlinephpfunctions.com/code/185658700006fe913e6592a7fc51d4cce0db6a68所有的

首先我建立的過濾器,我可以使用,那麼我將它應用到所有行的數組。就這麼簡單:對

希望它能幫助:)


編輯:

重建功能的一些低劣的部分,除去eval防止真棒安全隱患。

  • 現在buildFilter返回函數返回一個簡單的測試,這取決於您設置哪些運算 (你甚至可以自定義測試呵呵)
  • ApplyFilter修改結果:)

這是新的代碼

$rows = [ 
    ['k'=>1, 'dop1'=>'take', 'dop2' => 'a', 'dat' => '25-11-2016'], 
    ['k'=>2, 'dop1'=>'make', 'dop2' => 'b', 'dat' => '26-11-2016'], 
    ['k'=>3, 'dop1'=>'sake', 'dop2' => 'c', 'dat' => '27-11-2016'], 
    ['k'=>4, 'dop1'=>'bake', 'dop2' => 'd', 'dat' => '28-11-2016'], 
    ['k'=>5, 'dop1'=>'dake', 'dop2' => 'e', 'dat' => '29-11-2016'], 
    ['k'=>6, 'dop1'=>'jake', 'dop2' => 'f', 'dat' => '30-11-2016'], 
    ['k'=>7, 'dop1'=>'ake', 'dop2' => 'g', 'dat' => '24-11-2016'] 
]; 

$filters = [ 
    ['dat', '=', '27-11-2016'], 
    ['dop1', '=', "bake"], 
    ['dop1', '=', "sake"], 
    ['dop1', '=', "take"], 
]; 


function marrfilter($filters_arr, $rows) 
{ 
    /* 
    * Building filters to eval 
    */ 
    $buildFilter = function($op, $val) 
    { 
     return function($data) use ($val, $op) 
     { 
      switch($op) 
      { 
       case '=': 
       case '==': 
        return ($data == $val); 
       case '===': 
        return ($data === $val); 

       case '!=': 
        return ($data != $val); 
       case '!==': 
        return ($data !== $val); 

       case '>': 
        return ($data > $val); 
       case '>=': 
        return ($data >= $val); 

       case '<': 
        return ($data < $val); 
       case '<=': 
        return ($data <= $val); 

       default: 
        throw new Exception('Unknown operand ' . $op); 
      } 
     }; 
    }; 

    $filters = array(); 
    foreach($filters_arr as $k => $v) 
    { 
     list($key, $op, $val) = $v; 

     if(!isset($filters[$key])) 
     { 
      $filters[$key] = $buildFilter($op, $val); 
     } 
     else 
     { 
      if(!is_array($filters[$key])) 
      { 
       $filters[$key] = array($filters[$key]); 
      } 

      $filters[$key][] = $buildFilter($op, $val); 
     } 
    } 

    /* 
    * Filter function 
    */ 
    $applyFilter = function($row, $fk, $fv) 
    { 
     // Key not found in row 
     if(!array_key_exists($fk, $row)) 
     { 
      return false; 
     } 

     // Get row[key] value 
     $val = $row[$fk]; 

     // if we have multiple tests for the filter it's an OR 
     if(is_array($fv)) 
     { 
      foreach($fv as $ft) 
      { 
       if($ft($val) === true) return true; // return true if filter passed 

       // or continue if not 
      } 
     } 
     else 
     { 
      if($fv($val) === true) return true; // return true if filter passed 
     } 

     // if no filter passed, return false 
     return false; 
    }; 

    /* 
    * Filtering 
    */ 
    $result = array(); 
    foreach($rows as $row) 
    { 
     $r = array(); 
     foreach($filters as $fk => $fv) 
     { 
      $r[] = $applyFilter($row, $fk, $fv); 
     } 

     $r = array_filter($r); 

     if(count($r) == count($filters)) 
     { 
      $result[] = $row; 
     } 
    } 

    return $result; 
} 
+0

請僅作爲一個例子,這是不可靠的找到這個代碼在生產中,花了10分鐘來建立它... – Bobot

+0

很酷,謝謝。明天,我會測試(我有很多測試版本)並將其修改爲生產。非常感謝。 –

0

所以無需編寫代碼給你,看看PHP's array_filter()

的僞代碼,你需要看起來像這樣的內容:

var_dump(array_filter($rows, function($row) { 
    global $filtered; 

    $match = 0; 

    foreach ($filtered as $index=>$filter) { 
    // run code here to determine if this is an OR or AND and apply matching to the $row 
    // $row will be like ['k'=>1, 'dop1'=>'take', 'dop2' => 'a', 'dat' => '25-11-2016'], 
    // match based on the index of your $filtered ($index===0 AND, otherwise OR) 
    } 

    return $match; 
});