回答:
簡単に言えば、「2番目の次元」がどこかにある可能性がある場合は、少なくとも暗黙的にループすることなしにそれを行うことはできません。最初のアイテムにする必要がある場合は、
is_array($arr[0]);
しかし、私が見つけることができる最も効率的な一般的な方法は、配列でforeachループを使用して、ヒットが見つかるたびに短絡することです(少なくとも暗黙的なループは、ストレートなfor()より優れています)。
$ more multi.php
<?php
$a = array(1 => 'a',2 => 'b',3 => array(1,2,3));
$b = array(1 => 'a',2 => 'b');
$c = array(1 => 'a',2 => 'b','foo' => array(1,array(2)));
function is_multi($a) {
$rv = array_filter($a,'is_array');
if(count($rv)>0) return true;
return false;
}
function is_multi2($a) {
foreach ($a as $v) {
if (is_array($v)) return true;
}
return false;
}
function is_multi3($a) {
$c = count($a);
for ($i=0;$i<$c;$i++) {
if (is_array($a[$i])) return true;
}
return false;
}
$iters = 500000;
$time = microtime(true);
for ($i = 0; $i < $iters; $i++) {
is_multi($a);
is_multi($b);
is_multi($c);
}
$end = microtime(true);
echo "is_multi took ".($end-$time)." seconds in $iters times\n";
$time = microtime(true);
for ($i = 0; $i < $iters; $i++) {
is_multi2($a);
is_multi2($b);
is_multi2($c);
}
$end = microtime(true);
echo "is_multi2 took ".($end-$time)." seconds in $iters times\n";
$time = microtime(true);
for ($i = 0; $i < $iters; $i++) {
is_multi3($a);
is_multi3($b);
is_multi3($c);
}
$end = microtime(true);
echo "is_multi3 took ".($end-$time)." seconds in $iters times\n";
?>
$ php multi.php
is_multi took 7.53565130424 seconds in 500000 times
is_multi2 took 4.56964588165 seconds in 500000 times
is_multi3 took 9.01706600189 seconds in 500000 times
暗黙的なループですが、一致が見つかるとすぐに短絡できません...
$ more multi.php
<?php
$a = array(1 => 'a',2 => 'b',3 => array(1,2,3));
$b = array(1 => 'a',2 => 'b');
function is_multi($a) {
$rv = array_filter($a,'is_array');
if(count($rv)>0) return true;
return false;
}
var_dump(is_multi($a));
var_dump(is_multi($b));
?>
$ php multi.php
bool(true)
bool(false)
is_multi()
でコードを最適化するreturn count($rv)>0
count()を2回使用します。デフォルトモードで1回、再帰モードで1回。値が一致する場合、配列は多次元ではありません。これは、多次元配列の方が再帰カウントが高いためです。
if (count($array) == count($array, COUNT_RECURSIVE))
{
echo 'array is not multidimensional';
}
else
{
echo 'array is multidimensional';
}
このオプションの2番目の値mode
は、PHP 4.2.0で追加されました。PHPドキュメントから:
オプションのモードパラメータがCOUNT_RECURSIVE(または1)に設定されている場合、count()は配列を再帰的にカウントします。これは、多次元配列のすべての要素を数える場合に特に役立ちます。count()は無限再帰を検出しません。
ただし、この方法では検出されませんarray(array())
。
PHP 4.2.0以降の場合:
function is_multi($array) {
return (count($array) != count($array, 1));
}
array(array())
またはarray(array(), array())
どちらかのために働いていません。通常、内部の配列が空の場合、再帰的なカウントは正しく0を追加するため、通常のカウントと一致します。
あなたは単にこれを実行することができます:
if (count($myarray) !== count($myarray, COUNT_RECURSIVE)) return true;
else return false;
オプションのmodeパラメータがCOUNT_RECURSIVE
(または1)に設定されている場合、count()は配列を再帰的にカウントします。これは、多次元配列のすべての要素を数える場合に特に役立ちます。
同じ場合は、サブレベルがどこにもないことを意味します。簡単で速い!
if(count($tasks_by_date) !== count($tasks_by_date, 1))
!==
サブレベルが存在することを確認するために使用されました。同様の何かを探しているかもしれない理論のために...など。
!==
is_array()
配列の最初の要素が配列である場合、残りの要素もそうであるという前提の下で、最初の要素を確認することができます。
if( is_array(current($arr)) ) { // is multidimensional }
function countdim($array)
{
if (is_array(reset($array)))
$return = countdim(reset($array)) + 1;
else
$return = 1;
return $return;
}
この機能が最も簡単で、最も効率的で、最も速い方法であることがわかるでしょう。
function isMultiArray($a){
foreach($a as $v) if(is_array($v)) return TRUE;
return FALSE;
}
次のようにテストできます。
$a = array(1 => 'a',2 => 'b',3 => array(1,2,3));
$b = array(1 => 'a',2 => 'b');
echo isMultiArray($a) ? 'is multi':'is not multi';
echo '<br />';
echo isMultiArray($b) ? 'is multi':'is not multi';
foreach($a as $v) is_array($v) ? return TRUE : return FALSE;
foreach($a as $v) return is_array($v) ? true : false;
PHP 7以降は、次のようにすることができます。
public function is_multi(array $array):bool
{
return is_array($array[array_key_first($array)]);
}
次のような簡単なチェックも実行できます。
$array = array('yo'=>'dream', 'mydear'=> array('anotherYo'=>'dream'));
$array1 = array('yo'=>'dream', 'mydear'=> 'not_array');
function is_multi_dimensional($array){
$flag = 0;
while(list($k,$value)=each($array)){
if(is_array($value))
$flag = 1;
}
return $flag;
}
echo is_multi_dimensional($array); // returns 1
echo is_multi_dimensional($array1); // returns 0
これでもうまくいく
is_array(current($array));
もし偽の一次元配列の場合、真のマルチ次元配列。
currentはあなたの配列の最初の要素を与え、最初の要素が配列であるかどうかをis_array関数でチェックします。
COUNT_RECURSIVEを使用しない
rsortを使用してからissetを使用する
function is_multi_array( $arr ) {
rsort( $arr );
return isset( $arr[0] ) && is_array( $arr[0] );
}
//Usage
var_dump( is_multi_array( $some_array ) );
私の場合。様々な変なコンディションで立ち往生しました。
1番目のケース= array("data"=> "name");
2番目のケース= array("data"=> array("name"=>"username","fname"=>"fname"));
ただしdata
、値の代わりに配列がある場合、sizeof()またはcount()関数はこの条件では機能しません。次に、チェックするカスタム関数を作成します。
配列の最初のインデックスに値がある場合、「値のみ」を返し
ますが、インデックスに値の代わりに配列がある場合、「配列あり」
を返しますこの方法を使用します
function is_multi($a) {
foreach ($a as $v) {
if (is_array($v))
{
return "has array";
break;
}
break;
}
return 'only value';
}
これは上品だと思います(ユーザー名がわからない別のユーザーへの小道具):
static public function isMulti($array)
{
$result = array_unique(array_map("gettype",$array));
return count($result) == 1 && array_shift($result) == "array";
}
上記のすべての方法は、迅速にロールアウトするには複雑すぎます。配列がフラットの場合、最初の要素をテストすると、int、stringなどのプリミティブが返されます。多次元の場合、配列を返します。ひとことで言えば、この1つのライナーをすばやくきれいに使用できます。
echo is_array(array_shift($myArray));
これがtrueを返す場合、配列は多次元です。それ以外はフラットです。ただ注意してください。配列が異なる次元を持つことは非常にまれです。たとえば、モデルからデータを生成している場合、ループが通ることができる同じタイプの多次元またはフラット構造が常にあります。 そうでない場合は、手作業でカスタム構築しています。つまり、すべてがどこにあるかがわかっていて、ループアルゴリズムを記述する必要なく機能します。
array_shift()
れるため、注意が必要です。それでもワンライナーのかゆみがある場合は、よりよく使用してください。current()
前の回答に加えて、確認したい配列のスキーマに応じて:
function is_multi_array($array=[],$mode='every_key'){
$result = false;
if(is_array($array)){
if($mode=='first_key_only'){
if(is_array(array_shift($array))){
$result = true;
}
}
elseif($mode=='every_key'){
$result = true;
foreach($array as $key => $value){
if(!is_array($value)){
$result = false;
break;
}
}
}
elseif($mode=='at_least_one_key'){
if(count($array)!==count($array, COUNT_RECURSIVE)){
$result = true;
}
}
}
return $result;
}
そのように単純です
$isMulti = !empty(array_filter($array, function($e) {
return is_array($e);
}));
if($array[0]){
//enter your code
}
if (isset($array[0])) { }
。配列のインデックスが0から始まることが確実な場合
if ( array_key_exists(0,$array) ) {
// multidimensional array
} else {
// not a multidimensional array
}
*数値インデックスを持つ配列のみ
ネイティブのprint_r関数は、人間が読める文字列を返します。「配列」インスタンスを数えるだけです。
やってみて...
substr_count(print_r([...array...], true), 'Array') > 1;
$a = array(1 => 'a',2 => 'b',3 => array(1,2,3));
$b = array(1 => 'a',2 => 'b');
$c = array(1 => 'a',2 => 'b','foo' => array(1,array(2)));
$d = array(array());
$e = array(1, array());
$f = array(array(), array());
$g = array("hello", "hi" => "hi there");
$h[] = $g;
var_dump(substr_count(print_r($a, true), 'Array') > 1);
...
//a: bool(true)
//b: bool(false)
//c: bool(true)
//d: bool(true)
//e: bool(true)
//f: bool(true)
//g: bool(false)
//h: bool(true)
私のボックスでは、「is_multiは500000回で0.83681297302246秒かかりました」
礼儀:Ruach HaKodesh
function isMultiArray(array $value)
{
return is_array(reset($value));
}