• Jobs
  • About Us
  • professionals
    • Home
    • Jobs
    • Courses and challenges
  • business
    • Home
    • Post vacancy
    • Our process
    • Pricing
    • Assessments
    • Payroll
    • Blog
    • Sales
    • Salary Calculator

0

320
Views
Elequent whereIn no funciona con matriz cuando la matriz es demasiado grande

Tengo una matriz como esta:

$array = ['1' , '2' ,'100'];

y mi consulta es:

$query = Customer::whereIn('id', $array)->get();

Cuando $array es demasiado grande (1500 elementos y más), no funciona y devuelve un valor nulo.

También aumenté el max_allowed_packet en la configuración de MySQL. pero problema no resuelto.

PHP v7.33, laravel v7.19, MySQL v5.7

over 3 years ago · Santiago Trujillo
3 answers
Answer question

0

Ok, esto me tomó 6 horas seguidas, ¡pero logré encontrar una explicación profunda del problema!

De hecho, el problema no es con elocuente en sí mismo, sino con la combinación de elocuente y la base de datos para optimizar las consultas.

Eloquent prepara todas las consultas con PDO::prepare() . En el caso de una consulta whereIn(), el resultado es algo así como

 PREPARE SELECT * FROM model WHERE id IN (?, ?, ?, ?, ?, ?, ?, ?, ?, .....);

Por su parte, la base de datos (mySQL, mariaDB) tiene una variable llamada in_predicate_conversion_threshold para optimizar específicamente la cláusula IN(). De forma predeterminada, el valor es 1000 , lo que significa que la consulta se convertirá a una versión optimizada que implica subconsultas.

Por lo tanto, vincular más de 999 elementos en una cláusula IN puede generar errores en los que la base de datos devolverá 0 resultados sin errores.

Solución alternativa 1 : use consultas sin formato en el lado de laravel.
Solución alternativa 2 : use declaraciones no preparadas.
Solución alternativa 3 : haga fragmentos para selecciones o actualizaciones.
Solución alternativa 4 : aumente o elimine el umbral para la optimización IN (valor de 0 = sin optimización).

Para el último punto, simplemente ejecute SET in_predicate_conversion_threshold = 0; antes de su consulta, o configúrelo globalmente en el archivo de configuración. (no ejecute la consulta directamente en la línea de comando ya que se trata de modificaciones basadas en sesiones)

over 3 years ago · Santiago Trujillo Report

0

Me gustaría poner un UP sobre este tema porque también lo encontramos en nuestra infraestructura.

El mayor problema no es que Eloquent devuelva nulo o vacío, sino el hecho de que no produce ningún error , por lo que es muy difícil de depurar.

Algunas pruebas adicionales que hice:
  • Model::whereIn() está trabajando con una pequeña lista de identificadores
  • Comience a devolver una matriz vacía antes de 1500 entradas
  • La misma consulta hecha con DB::select() funciona bien
  • La misma consulta realizada pero con la extracción de la consulta y el enlace del generador de consultas funciona bien
  • Igual que el anterior pero con Model::hydrate() funcionando bien
Conclusión :
  • Esto es principalmente un problema con Eloquent en sí mismo, ya que no implica la base de datos subyacente (o los límites de la red)
  • Mi versión de la doctrina: 2.13.8, laravel: v7.30.6. No se puede probar con otras versiones en este momento, es posible que se resuelva en versiones más recientes
  • El informe de error ausente puede ser un error o una configuración incorrecta
  • Solución alternativa: consulta sin formato
over 3 years ago · Santiago Trujillo Report

0

Use whereIntegerInRaw, que se asegura de que la matriz sea solo números enteros, pero omite la preparación

 Customer::whereIntegerInRaw('id', $array)->get();
over 3 years ago · Santiago Trujillo Report
Answer question
Find remote jobs

Discover the new way to find a job!

Top jobs
Top job categories
Business
Post vacancy Pricing Our process Sales
Legal
Terms and conditions Privacy policy
© 2025 PeakU Inc. All Rights Reserved.

Andres GPT

Recommend me some offers
I have an error