Archivo por meses: Junio 2012

Los 5 errores mas comunes cuando se programa en C

He hecho una recopilación de algunos de los errores mas comunes que la gente comete a la hora de programar en C, el lenguaje C aunque es muy poderoso es bastante inflexible a diferencia de otros lenguajes de tipado sencillo como PHP por ejemplo :), sin embargo son obvias las ventajas que nos aporta C frente a los demás lenguajes, podemos comenzar diciendo que C ha sido el padre de varios lenguajes :)!.

1.- Sintaxis de comentarios incompletos

a = b; /* el comentario no termina
c = d; /* c = d se incluye en el comentario */

Por lo tanto c = d no se ejecutara

2.- Declaraciones accidentales / declarar booleanos accidentalmente

f(a = b) c; /* a tomara el valor de b, en lugar de ser comparado */

El valor de b sera asignado a “a”, en lugar de ser comparado, debido a que a != 0 entonces c se ejecutara. Otro caso similar podría ser también

 if( a =! b) c; /* esto es compilado como (a = !b) */

En lugar de (a != b) o (a == !b).

3.- Hacer mal uso de las macros en C

#define assign(a,b) a=(char)b
assign(x,y>>8)

/* Lo anterior es compilado como */ 
x = (char) y >> 8    /* Probablemente no es lo que queríamos */

4.- Archivos de cabecera que no coinciden

Supongamos que tenemos un archivo foo.h que contiene las siguientes instrucciones

struct foo { BOOL a};

Además tenemos otros 2 archivos f1.c y f2.c donde hacemos un include a ese archivo de cabecera.

/* F1.c  contiene*/
#define BOOL char
#include "foo.h"

/* F2.c contiene*/
#define BOOL int
#include "foo.h"

f1.c y f2.c difieren acerca del atributo principal de la estructura foo.

5.- Valores de retorno fantasma

Suponemos que tenemos el siguiente código:

 int foo (a)
 { 
    if (a) return(1); 
 } 
/* Esta mal, ya que en algunos casos no retornara ningún valor */

Este es el típico caso donde no conceptualizamos todos los posibles casos y nos hace falta implementar valores de retorno.

Finalmente no esta de mas decir que con las nuevas arquitecturas de 64 bits, hay nuevas formas de cometer errores cuando programamos en C, especialmente cuando se trabaja con matrices que se acercan o superan los 2 ^ 31 elementos (aunque es bastante raro).

Existen muchos mas errores que podemos cometer, algunos son verdaderamente difíciles de ocasionar pero pueden llegar a suceder, sobre todo los que tienen que ver con el compilador xD.
Para una lista mas detallada de errores pueden visitar el siguiente sitio The Top 10 Ways to get screwed by the “C” programming language

SelectBox dependientes – ciudades y estados

Retomando los post sobre programación web en esta ocasión les comparto un rápido ejemplo sobre selectbox dependientes, mas concretamente el ejemplo de selectbox de ciudades y estados. Básicamente tenemos 2 selectbox, el primero contendrá los estados de la republica mexicana, cuando seleccionamos alguno el segundo selectbox contendrá ahora las ciudades de ese estado, para hacer esto necesitamos una pequeña base de datos con 2 tablas (ciudades y estados .. obvio), después algo de código php que se encargue de hacer la conexión y otro mas que nos devuelva los datos de las consultas vía JSON y por ultimo nuestro html y javascript que se encargara de mostrar el resultado.

Al terminar el tutorial tendremos algo que se vera así 🙂

Puedes descargar los archivos que se utilizara en la practica a continuación.

Comenzamos pues :).

Comenzamos creando el archivo que hará la conexión a la base de datos, yo acostumbro hacer una clase para eso ya que luego puedo hacer extends de la misma y todo es mas fácil.

<?
  global $ID_BD;
  $ID_BD=mysql_connect("127.0.0.1", "root","");
  mysql_select_db("states_cities", $ID_BD);
  mysql_query("SET NAMES 'utf8'", $ID_BD);  

class mysql {
VAR $ID;
function mysql (){

}
  function _query($sql){

  global $ID_BD;
   

    //echo "$sql\n";
      return mysql_query($sql,$ID_BD);
  }
  
  function _registers($sql){
  
  global $ID_BD;
    [email protected]_query($sql, $ID_BD);
    
    //echo $result;
    //echo $sql;
    if ($row=mysql_fetch_object($result)){
        do{
        
        $registros[]= $row;
        
        }while($row=mysql_fetch_object($result));
    }
    else
      return array();
    return $registros;
  }
  
  function _register ($sql){

  global $ID_BD;
    //echo "$sql";
      $result=mysql_query($sql, $ID_BD);
      if ($row=mysql_fetch_object($result))
        return  $row;
  }
  
  function _close(){
    mysql_close($this->ID);
  }
  
  function _last_id(){
    global $ID_BD;
      return  mysql_insert_id($ID_BD);
  }
}  

?>

Después creamos el archivo que se encargara de hacer las consultas y regresar el JSON correspondiente, realizaremos peticiones a este archivo via ajax por medio de jquery.

<?php
require_once 'mysql.php';

class data extends mysql
{
  function getStates()
  {
    $states = $this->_registers("SELECT * FROM states;");
    return json_encode($states);
  }

  function getCities($idState)
  {
    $idState = (int) $idState;
    $cities = $this->_registers("SELECT * FROM cities WHERE id_state = " . $idState . ";");
    return json_encode($cities);
  }
}

  $data = new data();
  if($_GET['action'] == "getStates")
  {
    echo $data->getStates();
  }
  else if($_GET['action'] == "getCities" && isset($_GET['stateID']))
  {
    echo $data->getCities($_GET['stateID']);
  }

?>

Como podemos ver he empaquetado el código en una clase y casi en la parte final del código reviso si existen algunos parámetros pasados por GET para mandar llamar a las funciones correspondientes :), examinemos por ejemplo la función que nos trae los estados:

  function getStates()
  {
    $states = $this->_registers("SELECT * FROM states;");
    return json_encode($states);
  }

Básicamente hacemos un select y el arreglo asociativo resultante (checar la clase mysql para mas información) lo pasamos por json_encode para ser manipulado con javascript, lo mismo pasa con la función que te arroja las ciudades solo que esa tiene una clausula WHERE que te manda traer solo las ciudades de X estado :).

Finalmente creamos nuestro archivo html donde desplegaremos los 2 selectbox, para eso escribimos el siguiente código.

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
  <meta http-equiv="X-UA-Compatible" content="IE=edge" />
  <title>SelectBox dependientes</title>
    <script type="text/JavaScript" src="http://ajax.googleapis.com/ajax/libs/jquery/1.5.1/jquery.min.js"></script>
    <script language="JavaScript">

    Object.size = function(obj) {
      var size = 0, key;
      for (key in obj) {
          if (obj.hasOwnProperty(key)) size++;
      }
      return size;
    };
    $(document).ready(function() {

      //$.ajax({
      $.ajax({
        url: 'data.php?action=getStates',
        dataType: 'json',
        success: function(data) {
          var size = Object.size(data);
          var items = "";
          for(var i = 0; i < size; i++)
          {
            items = items + '<option value="' + data[i].id_state + '">' + data[i].name + '</option>';
            //console.log(data[i].id_state + " : " + data[i].name);
          }  
          $('#states').append(items);
          $('#states').change();
        }
      });

     $('#states').change(function() {
      $.ajax({
        url: 'data.php?action=getCities&stateID=' + $('#states').val(),
        dataType: 'json',
        success: function(data) {
          var size = Object.size(data);
          var items = "";
          for(var i = 0; i < size; i++)
          {
            items = items + '<option value="' + data[i].id_state + '">' + data[i].name + '</option>';
            //console.log(data[i].id_state + " : " + data[i].name);
          }  
          $('#cities').html("");
          $('#cities').append(items);
          $('#cities').change();
        }
      });      
     });

    });      
    </script>
</head>
<body>

<a href="#">Selecciona un estado</a>
<br/>
<select id="states">
</select>
<br/>
<a href="#">Selecciona una ciudad</a>
<br/>
<select id="cities">
</select>
<br/>

<div id="map">
</div>

</body>
</html>

Como pueden ver, al inicio cargamos la librería de JQuery desde google, después hacemos la primera petición al script data.php via ajax para que nos traiga los estados y poder “rellenar el primer selectbox”, después tenemos el siguiente código javascript.

     $('#states').change(function() {
      $.ajax({
        url: 'data.php?action=getCities&stateID=' + $('#states').val(),
        dataType: 'json',
        success: function(data) {
          var size = Object.size(data);
          var items = "";
          for(var i = 0; i < size; i++)
          {
            items = items + '<option value="' + data[i].id_state + '">' + data[i].name + '</option>';
            //console.log(data[i].id_state + " : " + data[i].name);
          }  
          $('#cities').html("");
          $('#cities').append(items);
          $('#cities').change();
        }
      });      
     });


change() nos permite detectar el cambio de selección de un selectbox, entonces utilizamos ese evento para “refrescar” las opción del segundo selectbox, vemos cual es el estado que esta seleccionado, extraemos su valor (es el mismo id que le corresponde en la tabla de estados) y armamos la petición para mandar traer las ciudades :).

Y listo con esto tenemos listo un mini ejemplo de selectbox dependientes, espero les sirva este pequeño snippet :), ya saben cualquier duda pónganla en comentarios.

La verdad es un código bastante sencillo y fácil de implementar (unos 10 mins a lo mucho), con un poco mas de código se podría mostrar la ubicación de la ciudad en google maps o utilizar esto en algún formulario de registro o cualquier otra cosa.

salu2