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_documento Pero 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,

"Abrir carpeta contenedora" en Firefox y KDE 4.3.x lanza Cervisia

Este es un bug conocido desde hace algún tiempo, pero hay un truco que puede solucionarlo: Edita cervisia.desktop y kfmclient-dir.desktop localizado en /usr/share/applications/kde4 y agrega una linea con "OnlyShowIn=KDE;". Despues de actualizar "update-mime-cache" firefox usará dolphin. Mas información: https://bugzilla.mozilla.org/show_bug.cgi?id=266600 Actualización: El proceso al fin y al cabo le falta un paso mas. Cuando volvi a probar abrir un archivo desde la opción de "Abrir carpeta contenedora", me pidió que asociara el archivo a un programa, así que nada mas me tocó buscar donde se encuentra dolphin(/usr/bin/) y marcar la opción recordar asociación Actualización: En OpenSUSE 11.2 el problema fue solucionado.

Tips y enlaces de la semana

json_encode y problemas con acentos. Según la documentación de la función json_encode , esta solo funciona con caracteres codificados en utf-8, así que si trabajamos con caracteres con otra codificación podemos convertirlos con la función utf8_encode. Asi: json_encode(utf8_encode($dato)); Si lo que queremos es pasar un arreglo a json, debemos pasar cada item del arreglo a utf8 y para esto usaremos la función array_map, quedando de la siguiente manera: json_encode(array_map("utf8_encode",$arreglo)); Esta función está disponible desde la versión 5.2 de PHP, asi que si usas una versión anterior intentalo con la versión de json_encode y json_decode para PHP4 Este archivo se usa de la siguiente forma: // create a new instance of Services_JSON require_once('JSON.php'); $json = new Services_JSON(); // convert a complex value to JSON notation $value = array(1, 2, ‘foo’); $output = $json->encode($value); print($output); // accept incoming POST data $input =