Buscar este blog

domingo, 26 de mayo de 2013

Obtener la localización de usuarios web con GeoIP

En .Net tenemos la posibilidad de obtener la localización de nuestros usuarios usando las bases de datos y clases que proporciona MaxMind de manera gratuita, hay dos versiones de las bases de datos, para localizar el país, y para encontrar datos hasta el nivel de ciudad, las diferencia es que la base de datos de países es mucho mas pequeña pero en la versión de ciudades también podemos obtener también el país pero además la zona, la ciudad el código postal y otros datos.
Para usarla hay que copiar el archivo de datos a una carpeta accesible por nuestra aplicación, instanciar la clase LookupService y llamar sus métodos basandonos siempre en la ip del cliente.
La única inconveniencia es que las clases no vienen en formato binario, pero es cuestión de minutos compilarlas y añadirlas a nuestros proyectos.
A continuación un ejemplo muy simplificado de su uso en Asp.Net:


public void Detect() {
    LookupService ls
        = new LookupService(Server.MapPath("/GeoLiteCity.dat")
        , LookupService.GEOIP_STANDARD);
    Location loc
        = ls.getLocation(Request.UserHostAddress);
    Response.Write(loc.regionName);
}

sábado, 25 de mayo de 2013

Trucos de rendimiento en linq

Usar linq es muy tentador por la forma en que hace fácil escribir código sobre colecciones de valores ya sea en objetos, xml, sql o cualquier otra forma de agrupar valores, sin embargo hay que tener mucho cuidado con su uso ya que podemos caer en trampas que hacen el código muy lento, por ejemplo: en linq a objetos es muy fácil crear una expresión que busque un objeto con un determinado valor en una propiedad, sin embargo la trampa es que linq recorre toda la colección y compara objeto por objeto con el valor buscado, en colecciones pequeñas esto no es un problema pero si manejas un número de objetos significativo tu programa se puede alentar bastante, una posible solución es usar un árbol e indizarlo usando la propiedad que vas a buscar con lo que el rendimiento puede mejorarse incluso en varios de magnitud.
Otro ejemplo linq a xml, en este caso el problema es que el xml se almacena por completo en memoria, siendo una vez mas un problema con archivos grandes que nos pueden usar hasta gigabytes de memoria para archivos de solo unos megas, en estos casos puede convenir mas usar el xmlreader que lee los elementos uno por uno desde el stream y luego los descarta usando por lo tanto poquísima memoria.

Por supuesto las necesidades de cada programa varían y puedes encontrarte en que no tengas otra solución que usar linq.

jueves, 7 de febrero de 2013

Mostrando ayuda chm en Windows Forms

En windows forms hay una clase especial para mostrar ayuda desde archivos chm para nuestra aplicación esta clase es System.Windows.Forms.Help y es una clase estatica muy fácil de usar basta llamar a uno de los métodos ShowHelp con los parámetros adecuados es decir la dirección del archivo y opcionalmente una dirección de navegación que nos mostrara la ayuda en la posición adecuada.
Para mostrar la ayuda en su inicio:

 Help.ShowHelp(this, "Desktop.chm");

Y para mostrarla en la pestaña de busqueda (sin palabra clave indicada):

 Help.ShowHelp(this, "Desktop.chm", HelpNavigator.Find, "");

jueves, 31 de enero de 2013

Usando c# en xslt

En .Net el soporte de xslt se limita a la version 1 y no parece que esto vaya a cambiar pronto asi que estamos muy limitados en cuanto a lo que se puede hacer: hay pocas funciones y no se pueden declarar funciones nuevas.
Afortunadamente Microsoft incluyo soporte para crear funciones nuevas funciones en cualquier lenguaje de .Net esto es a través del elemento <msxsl:script />
En el ejemplo creamos la función Trim que no tiene equivalente en xslt (aunque normalize-space se parece)
Hay que recordar que esto solo se soporta en m$, si vas a usar tu xslt con otras aplicaciones no va a funcionar.
Para activarlo con la clase  XslCompiledTransform se debe cargar el xslt con un objecto XsltSettings con la propiedad EnableScript en true:
XsltSettings xsltSettings = new XsltSettings();
xsltSettings.EnableScript = true;
XslCompiledTransform transform = new XslCompiledTransform(false);
transform.Load(styleSheet, xsltSettings, new XmlUrlResolver());



<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns:msxsl="urn:schemas-microsoft-com:xslt"
    exclude-result-prefixes="msxsl"
    xmlns:user="urn:my-scripts"
>
    <msxsl:script language="C#" implements-prefix="user">
        <![CDATA[
            public string Trim(string input) {
                return input.Trim();
            }
        ]]>
    </msxsl:script>
    <xsl:output method="xml" indent="yes"/>
 
    <xsl:template match="/">
        <NewRoot>
            <xsl:for-each select="OldValues/OldValue">
                <NewValue>
                    <Id>
                        <xsl:value-of select="user:Trim(Id)"/>
                    </Id>
                </NewValue>
            </xsl:for-each>
        </NewRoot>
    </xsl:template>
</xsl:stylesheet>

domingo, 17 de junio de 2012

Delegados anonimos asincronos

El titulo lo dice todo aquí esta el ejemplo:

Func<bool> async = () => {
    return false;
};
IAsyncResult res = async.BeginInvoke(null, null);
async.EndInvoke(res);
 
Lo único que hay que resaltar es que en el ejemplo use una función que retorna un bool como se observa claramente pero se pueden usar las diferentes variantes de Func y Action para cambiar los parámetros y resultados.

sábado, 16 de junio de 2012

MochaHost un buen hosting para Asp.Net

Si buscan un buen hosting para Asp.Net permitanme recomendarles MochaHost.
Yo llevo un tiempo con este servicio contratado para varios sitios, todos hechos en Asp.Net y me parece muy buena alternativa, el servicio es muy económico y tiene varios puntos a su favor en primer lugar todo es ilimitado: sitios ilimitados, subdominios ilimitados, trafico ilimitado y hasta bases de datos ilimitadas de tamaño ilimitado, incluyendo Sql Server.
Otra cosa es que actualizan la versión de .Net en cuanto aparece una nueva así que puedes empezar a usar nuevas características sin esperar meses para la actualización, esto es algo que casi nadie tiene.
Además el servicio es bueno, casi no falla y el soporte técnico responde rápido, por supuesto no esta libre de problemas, pero bueno yo he usado diferentes hostings y este no tiene tantos problemas como otros que son mas caros y lo limitan a uno en prestaciones.
Bueno sin más les dejo el link a MochaHost:

MochaHost - ASP.NET/MS SQL Hosting from $2.48/month !

domingo, 10 de junio de 2012

Usando dbus en .Net

Si programas para Linux una de las cosas que te serán muy útiles es usar dbus ya sea para implementar comunicación entre tus aplicaciones o para inspeccionar el estado del sistema.
En este ejemplo veremos como usar dbus para saber si el screensaver de gnome esta activo.
Usamos la dll dbus-sharp que se obtiene desde el git de mono, aquí esta la liga:
dbus-sharp en github
Yo use la ultima versión del repositorio no las descargas oficiales que son muy viejas.
Bueno lo mejor del proyecto es que te permite obtener la información con tipos de datos duros, nada de castear objects.
Lo primero que debes hacer es crear una interfaz que declare los miembros que exporta la clave de dbus que vas a usar por ejemplo para el screensaver en la siguiente pagina podemos ver los miembros exportados:
http://people.gnome.org/~mccann/gnome-screensaver/docs/gnome-screensaver.html

Ahí podemos ver que tenemos un método y una señal (evento) para saber si el screensaver esta activo:
GetActive y ActiveChanged
Asi que nuestra interfaz declara los dos miembros un evento y una función:
public delegate void ActiveChangedHandler(bool active);
[Interface("org.gnome.ScreenSaver")]
public interface IScreenSaver {
    event ActiveChangedHandler ActiveChanged;
    bool GetActive();
}
Como podemos ver podemos declarar solo los miembros que necesitamos no es necesario declarar todos los miembros.
Ahora usamos la interface en nuestro codigo, solo es necesario conectarse a dbus y obtener un objeto que implementa la interfaz desde la conexión, no es necesario ningún código para el objeto la dll se encarga de implementar la interfaz automáticamente, después solo resta usar los miembros:
class Program {
    static Connection con = Bus.Session;
    static void Main(string[] args) {
        ObjectPath opath = new ObjectPath("/org/gnome/ScreenSaver");
        IScreenSaver bus = con.GetObject<IScreenSaver>
            ("org.gnome.ScreenSaver", opath);
        for(int i = 0; i < 100; i++) {
            Console.WriteLine(bus.GetActive().ToString() + "\t"
                + i.ToString());
            Thread.Sleep(1000);
        }
    }
}
Sencillisimo y muy útil!
Por supuesto también usar en Windows y otros sistemas operativos, aunque por supuesto dbus es mucho mas común en Linux