Tengo la siguiente matriz:
$array = [ 'z' => 2, 'd' => 1, 'a' => 2, ];
Ahora me gustaría ordenar por el valor (entero) y luego ordenar por la clave en función de si está en esta lista de permitidos:
$allowlist = ['a', 'd'];
Así que hice lo siguiente:
arsort($array); uksort($array, function($a, $b) { return in_array($a, $allowlist) ? -1 : 1; });
Pero eso vuelve:
[d]: 1 [a]: 2 [z]: 2
Lo que realmente quiero es ordenar los valores primero y luego, si hay un desempate, ordenar la clave en función de si está en esa lista de permitidos, lo que debería dar como resultado esto:
[a]: 2 [z]: 2 [d]: 1
Puede usar uksort
, que le dará las claves, y puede pasar $array
a su función de clasificación para obtener acceso al valor.
$array = [ 'a' => 2, 'z' => 2, 'd' => 1 ]; $allowlist = ['a', 'd']; uksort( $array, static function($a, $b) use ($array, $allowlist) { if($array[$a] === $array[$b]) { return in_array($a, $allowlist) ? -1 : 1; } return $array[$b] <=> $array[$a]; } );
Esto se inspiró en esta respuesta: https://stackoverflow.com/a/65315474/231316
Demostración aquí: https://3v4l.org/toZIt
Recomiendo hacer la menor cantidad posible de evaluaciones y la menor cantidad posible de llamadas a funciones.
Hay deliberadamente dos operadores de naves espaciales en mi fragmento; esto significa que las llamadas in_array()
solo se ejecutan cuando es necesario. El operador de la nave espacial procesará matrices de reglas de clasificación, pero esto será menos eficiente porque todas las funciones in_array()
se llamarán en cada iteración. No utilices esta demostración .
Código: ( Demostración )
uksort( $array, fn($a, $b) => $array[$b] <=> $array[$a] // sort values DESC ?: in_array($b, $allowlist) <=> in_array($a, $allowlist) // sort keys with priority given to whitelisted keys ); var_export($array);
Esto ordena por valor descendente y luego rompe los empates al verificar si in_array()
desciende. Al ordenar resultados booleanos, descendiendo pone verdadero antes de falso.
Si está tratando con grandes volúmenes de matrices, le insto a que evite in_array()
. En su lugar, voltee su matriz de búsqueda y use isset()
en su función personalizada.
La sintaxis de reglas en cascada visible de la función personalizada hace que sea fácil de leer y ampliar. Si desea agregar otro desempate para cuando ambas o ninguna de las claves esté en la lista blanca, simplemente agregue otra línea con la nueva regla:
?: $a <=> $b // sort by key ascending (alphabetically)