Ir al contenido principal

Tips semanales #2

Suma de Entradas y Salidas

El problema reside en realizar una suma de dos campos en tablas distintas (entradas y salidas) con un enlace común (artículos), el DER correspondería a lo siguiente:
Modelo de Bodega
Pareciera fácil pero no lo es tanto pues tiene su truco. El código sería el siguiente:

SELECT tipo_articulos_nombre_tipo, articulos_descripcion, 
SUM(cantidad) AS entradas, SUM(cantidad_salida) AS salidas
FROM (
(SELECT
     tipo_articulos.`nombre_tipo` AS tipo_articulos_nombre_tipo,
     articulos.`descripcion` AS articulos_descripcion,
     detalle_entradas.`cantidad` AS cantidad,
 0.0 as cantidad_salida
FROM `articulos` articulos
     LEFT JOIN `detalle_entradas` detalle_entradas ON detalle_entradas.`idarticulo` = articulos.`idarticulo`
     INNER JOIN `entradas` entradas on entradas.identrada=detalle_entradas.`identrada`
     INNER JOIN `tipo_articulos` tipo_articulos ON articulos.`idtipo_articulo` = tipo_articulos.`idtipo_articulo`

ORDER entradas.`fecha_entrada` DESC
)
UNION ALL
(SELECT
     tipo_articulos.`nombre_tipo` AS tipo_articulos_nombre_tipo,
     articulos.`descripcion` AS articulos_descripcion,
     0.0 as cantidad,
     detalle_salidas.`cantidad_despachada` AS cantidad_salida
FROM `articulos` articulos
     INNER JOIN `tipo_articulos` tipo_articulos ON articulos.`idtipo_articulo` = tipo_articulos.`idtipo_articulo`
     LEFT JOIN `detalle_salidas` detalle_salidas  ON detalle_salidas.`idarticulo` = articulos.`idarticulo`
INNER JOIN `salidas` salidas ON detalle_salidas.`idsalida` = salidas.`idsalida`

ORDER BY salidas.`fecha_salida` DESC)
) AS tabla
GROUP BY articulos_descripcion

Me parece que este código es bastante optimizable e incluso puede existir otro enfoque totalmente distinto.
Que coincidan el uso del UNION, subconsulta y agrupación no me da buena espina. Me gustaría saber si es posible hacer esto de otra forma.

Booleanos en PHP

Si quieres convertir un representación en cadena de un boolean, ejem. "yes", "on", "true", "false", etc... a un boolean real, puedes probar esto:

<?php
        $myString = "On";
        $b = filter_var($myString, FILTER_VALIDATE_BOOLEAN);
?>
Esto retornará TRUE para "1", "true", "on" y "yes". De lo contrario retorna FALSE.
 
Escapar los caracteres raros en php para mysql

if(function_exists('mysql_real_escape_string')){
         $_REQUEST = array_map('mysql_real_escape_string', $_REQUEST); 
}

Remover elementos duplicados en un arreglo en PHP: array_unique($array);

$entrada = array("a" => "verde", "rojo", "b" => "verde", "azul", "rojo");
$resultado = array_unique($entrada);
print_r($resultado);
El resultado del ejemplo seria: Array ( [a] => verde [0] => rojo [1] => azul )

 Consulta que devuelve una fila aleatoria de una tabla de MySQL

select * from YOUR_TABLE order by rand() limit 1

Esta consulta funciona bien para tablas con pocos registros (menos de 10,000)  para tablas mas pesadas este enlace provee una mejor forma para seleccionar una fila aleatoria

Comentarios

  1. juan carlos, e tratado de usar la funcion array_map, especificamente en la directiva que das el ejemplo

    if(function_exists('mysql_real_escape_string')){
    $_REQUEST = array_map('mysql_real_escape_string', $_REQUEST);
    }

    pero no me funciona, al verificar la variable que almacena el nuevo arreglo esta vacia. E probado array_map con otras funciones(strtoupper) y funciona bien pero en mysql_real...string nada hermano. talvez me sugeris un camino

    ResponderEliminar
  2. Se me ocurren tres escenarios:
    1. Que el arreglo de entrada ($_REQUEST en este caso)este vacío.
    2. Que la ejecución no esté entrando dentro del if. Lo que significaría que o mysql no esta habilitado o la librería no soporta este método.
    3. Que sí se esté utilizando la función mysql_real_escape_string, pero no exista una conexión abierta, lo que devolvería un false a la variable destino y E_WARNING.
    Intenta descartar estos escenarios.

    ResponderEliminar
  3. en efecto, la solucion tres, es la correcto ayer sabado estuve probando y observe que ese era la razon. upps una pifia. gracias por contestar.

    ResponderEliminar

Publicar un comentario

Entradas populares de este blog

Enumerar filas en una consulta con MySQL

Supongamos que tenemos tablas con la estructura siguiente:
documentos (iddocumento, nombre_documento, url_original, idtipo_documento, idproyecto) proyectos (idproyecto, nombre_proyecto, longitud, unidad_medida) tipo_documentos (idtipo_documento, descripcion_tipo_documento) Tenemos necesidad de hacer una consulta como la siguiente: "Enumerar todos los documentos en la base de datos agrupados por proyecto"

Parece fácil, excepto por el término "enumerar", aquí tienes un truquito para que logres enumerar tus consultas:
SELECT (@rownum:=@rownum+1) AS rownum, nombre_documento, descripcion_tipo_documento, nombre_proyecto FROM (SELECT @rownum:=0) r, documentos AS d INNER JOIN proyectos AS p ON d.idproyecto = p.idproyecto INNER JOIN tipo_documentos AS td ON d.idtipo_documento = td.idtipo_documentoPero que tal si te piden que enumeres los proyectos con sus correspondientes documentos?. Teniendo lo anterior es un poco mas sencillo
SELECT IF(@fila=proyectos.idproyecto, @rownum:…

jQuery DataTables y CodeIgniter

Ajax Source Datatables permite configurar fácilmente el origen de datos de la tabla, para que esta sea generada dinámicamente desde el servidor, así que con CodeIgniter tendríamos el siguiente código

public function page(){ $data['pedidos'] = $this->pedidos_model->get_pedidos($this->input->post('iDisplayStart')); define('AJAX_REQUEST', 1);//truco para que en nginx no muestre el debug $TOTAL = $this->pedidos_model->total(); echo json_encode(array('aaData'=>$data['pedidos'], 'iTotalRecords'=>$TOTAL, 'iTotalDisplayRecords'=>$TOTAL, 'sEcho'=>$this->input->post('sEcho'))); }
Este método producirá algo parecido a esto:

{"iTotalRecords":83099,"iTotalDisplayRecords":83099,"sEcho":"2", "aaData":[{"Id":"85514","num":"86109",&q…

Expresiones Regulares y pruebas en javascript

¿Qué es una expresión regular?
Una expresión regular es una cadena que contiene una combinación de caracteres normales y metacaracteres o metasecuencias especiales. Los caracteres normales coinciden por ellos mismos. Los metacaracteres y metasecuencias son caracteres o secuencias de caracteres que representan ideas como cantidad, posiciones o tipos de caracteres.
Regular Expression Pocket Reference 2nd Ed - Tony Stubblebine - O'Reilly

¿Para qué son útiles las expresiones regulares?
Las expresiones son especialmente útiles para validar información, por ejemplo en formularios de ingreso de datos. Por ejemplo para validar que se ingresó un número de teléfono, puedes usar la siguiente expresión regular.

/^([\+][0-9]{1,3}[ \.\-])?([\(]{1}[0-9]{2,6}[\)])?([0-9 \.\-\/]{3,20})?$/

Parecieran símbolos al azar, pero nada mas lejos de la realidad. Te muestro una tabla básica con los elementos usados para crear expresiones regulares.

Carácter Texto buscado ^Principio de entrada o línea.$Fin de e…