lunes, 11 de enero de 2016

Guardar Tablas en Memoria caché y utilizarlas en todas las páginas

A menudo, una aplicación puede aumentar el rendimiento si almacena en memoria los datos a los que se tiene acceso con más frecuencia y cuya creación requiere un tiempo de procesamiento significativo. Por ejemplo, si su aplicación procesa grandes cantidades de datos utilizando una lógica compleja y, a continuación, los devuelve en un formato de informe que es consultado frecuentemente por los usuarios, resulta eficiente evitar la creación de dicho informe cada vez que un usuario lo solicita. De manera similar, si su aplicación incluye una página que procesa datos complejos pero que no se actualiza frecuentemente, no resulta eficiente que el servidor vuelva a crear dicha página en cada solicitud.
Cita de Microsoft
Este es un ejemplo de cómo  guardo  la configuración de mi  sitio web,  en  este caso  se trata de una tabla de  mi base de datos que  guarda los campos que son obligatorios por cada página. Existe una interfaz donde el usuario  puede cambiar esta configuración y se actualiza la base de datos y la tabla en la cache. La tabla de cargará una sola vez cuando se inicie la aplicación, se actualizará cuando  haya cambios en la base de datos (Desde la aplicación,  aun NO sé como  hacerlo con  dependencia de SQL). y se destruirá cuando de la memoria cuando  pase cierto  tiempo, o  se cierre o  reinicie la aplicación.



Expiración

Además de utilizar el barrido, ASP.NET quita automáticamente los elementos de la memoria caché cuando expiran. Al agregar un elemento a la memoria caché se puede establecer su expiración de acuerdo con la tabla siguiente.
Tipo de expiración
Descripción
Fecha de expiración variable
Especifica el intervalo de tiempo que debe transcurrir desde el último acceso al elemento para que éste expire. Por ejemplo, puede establecer que un elemento expire 20 minutos después del último acceso en caché.
Expiración absoluta
Especifica que un elemento expira en un momento determinado, sin tener en cuenta la frecuencia con que se ha tenido acceso al mismo. Por ejemplo, puede establecer que un elemento expire a las 6:00 p.m. o al cabo de cuatro horas.



Ejemplos de código de los dos tipos de expiración
Cache.Insert("Cache_ConfiguracionPorPagina", tablaAE, null, System.Web.Caching.Cache.NoAbsoluteExpiration, TimeSpan.FromMinutes(1)); //Sólamente se borra si  pasó el tiempo indicado y no  se accedió a ella

 Cache.Insert("Cache_ConfiguracionPorPaginaSE", tablaSE, null,DateTime.Now.AddMinutes(1),System.Web.Caching.Cache.NoSlidingExpiration); // Se estingue en el tiempo indicado,  se acceda o no a la tabla en cache


Creé la clase Utilidades.cs

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using SistemaExpediente.Clases;
using System.Data.SqlClient;
using System.Collections;
using System.IO;
using System.Data;
using System.Web.UI.WebControls;
using System.Web.Caching;

namespace SistemaExpediente.Clases

{
    //Agregue  la herencia de : System.Web.UI.Page línea para usar la cache
    public class Utilidades //: System.Web.UI.Page
    {
        Conexion con = new Conexion();
        string error = "", sql = "", a = "";
        SqlDataReader r;

        /*********CACHE***********/

        //Si no  existe la tabla la crea, si  existe la reemplaza
        /// <summary>
        /// Inicializa las tablas que se guardarán en  la memoria caché
        /// </summary>
        public void CreateAllTablesCache()
        {
            DataTable tablaAE = new DataTable(); //Absolute Expiration
            DataTable tSN = new DataTable();
            tablaAE = con.GetSelect("SELECT *  FROM dbo.View_TablaValidaciones");
            tSN = con.GetSelect("Select * from ConfiguracionSiNo");
            System.Web.HttpContext.Current.Cache.Insert("Cache_ConfiguracionPorPagina", tablaAE, null, Cache.NoAbsoluteExpiration, TimeSpan.FromMinutes(1)); //Sólamente se borra si  pasó el tiempo indicado y no  se accedió a ella
            System.Web.HttpContext.Current.Cache.Insert("Cache_ConfiguracionSiNo", tSN, null, Cache.NoAbsoluteExpiration, TimeSpan.FromMinutes(1));  //Sólamente se borra si  pasó el tiempo indicado y no  se accedió a ella
            //Cache.Insert("Cache_ConfiguracionPorPaginaSE", tablaSE, null, DateTime.Now.AddMinutes(20), Cache.NoSlidingExpiration); // Se estingue en el tiempo indicado,  se acceda o no a la tabla en cache
        }
        /// <summary>
        /// Si  una tabla caducó  de la memoria caché,  se vuelve a crear con  este método
        /// </summary>
        /// <param name="key">El nombre de la tabla a crear</param>
        /// <returns>Regresa un objeto  DataTable</returns>
        public DataTable CreateTableCache(string key)
        {
            DataTable table = new DataTable(); //Absolute Expiration
            if (key != "")
            {
                if (key == "Cache_ConfiguracionPorPagina") table = con.GetSelect("SELECT *  FROM dbo.View_TablaValidaciones");
                if (key == "Cache_ConfiguracionSiNo") table = con.GetSelect("Select * from ConfiguracionSiNo");
            }
            if(table.Rows.Count>0)
            System.Web.HttpContext.Current.Cache.Insert(key, table, null, Cache.NoAbsoluteExpiration, TimeSpan.FromMinutes(1)); //Sólamente se borra si  pasó el tiempo indicado y no  se accedió a ella    
            return table;
        }

        public void RemoveALLTablesCache()

        {
            if (System.Web.HttpContext.Current != null)
            {
                System.Web.HttpContext.Current.Cache.Remove("Cache_ConfiguracionPorPagina");
                System.Web.HttpContext.Current.Cache.Remove("Cache_ConfiguracionSiNo");
            }
        }
        /// <summary>
        /// Para obtener valores escalares de una DataTable en la memoria cache
        /// </summary>
        /// <param name="table">Nombre de la tabla</param>
        /// <param name="filtro">Parámetros utilizados en el 'where' del select al DataTable para obtener  el  valor</param>
        /// <param name="valor">El valor a regresar, de la fila obtenida </param>
        /// <returns>Un string con el  valor</returns>
        public string TableValue(DataTable table,string [] filtro,string valor)
        {
            string f = "";
            for (int i = 0; i < filtro.Length; i++) f += filtro[i] + " AND "; 
            f=f.Substring(0,f.Length-5);
            return table.Select(f)[0][valor].ToString();
        }
        
        /// <summary>
        /// Validación de campos que son iguales en todo el sistema 
        /// </summary>
        /// <param name="t">DataTable a buscar</param>
        /// <param name="c">Valor de la columna campo</param>
        /// <param name="ctrol">Regresa el control que contiene el error</param>
        /// <param name="conte">Regresa el id del  Div HTML  que contiene el error</param>
        /// <param name="msjVal">Regresa un mensaje de validación a mostrar en una ventana de mensajes para el usuario</param>
        /// <returns>Regresa true si el campo es obligatorio y false si no lo es</returns>
        public bool VlGn(DataTable t, string c, ref string ctrol,ref string conte,ref string msjVal)
        {
            ctrol = TableValue(t, new string[] { "Pagina='TODAS'", "Campo='" + c + "'" }, "idValidacion").ToString().Trim();
            conte = TableValue(t, new string[] { "Pagina='TODAS'", "Campo='" + c + "'" }, "Contenedor").ToString().Trim();
            msjVal = TableValue(t, new string[] { "Pagina='TODAS'", "Campo='" + c + "'" }, "Mensaje").ToString().Trim();
            return Convert.ToBoolean(TableValue(t, new string[] { "Pagina='TODAS'", "Campo='" + c + "'" }, "Caracter"));
        }
       /// <summary>
        /// Regresa id de el  control de la validación
       /// </summary>
       /// <param name="t">DataTable a buscar</param>
       /// <param name="c">Valor de la columna campo</param>
       /// <param name="ctrol">Regresa el control que contiene el error</param>
        /// <returns>Regresa id de el  control de la validación</returns>
        public string idVlGn(DataTable t, string c,ref bool ctrol)
        {
            return TableValue(t, new string[] { "Pagina='TODAS'", "Campo='" + c + "'" },c).ToString().Trim();
        }

        //Campos necesarios para la validación

        /// <summary>
        /// Metodo para realizar la validación de campos.Recibe parámetros referenciados con el control, el DIV contenedor y el mensaje de validación
        /// </summary>
        /// <param name="t">DataTable a buscar</param>
        /// <param name="pagina"></param>
        /// <param name="c">Valor de la columna campo</param>
        /// <param name="ctrol">Regresa el control que contiene el error</param>
        /// <param name="conte">Regresa el id del  Div HTML  que contiene el error</param>
        /// <param name="msjVal">Regresa un mensaje de validación a mostrar en una ventana de mensajes para el usuario</param>
        /// <returns>Regresa true si el campo es obligatorio y  false si no lo es</returns>
        public bool Vl(DataTable t, string pagina, string c, ref string ctrol, ref string conte,ref string msjVal)
        {
            ctrol = TableValue(t, new string[] { "Pagina='" + pagina + "'", "Campo='" + c + "'" }, "idValidacion").ToString().Trim();
            conte = TableValue(t, new string[] { "Pagina='" + pagina + "'", "Campo='" + c + "'" }, "Contenedor").ToString().Trim();
            msjVal = TableValue(t, new string[] { "Pagina='" + pagina + "'", "Campo='" + c + "'" }, "Mensaje").ToString().Trim();
            return Convert.ToBoolean(TableValue(t, new string[] { "Pagina='"+pagina+"'", "Campo='" + c + "'" }, "Caracter"));
        }


        //Regresa la tabla obtenida de la cache 

        /// <summary>
        /// Regresa la tabla obtenida de la cache, si la tabla no existe la agrega a la cache llamando al submétodo CreateTableCache
        /// </summary>
        /// <param name="key">Nombre de la tabla</param>
        /// <returns>Tabla de la memoria caché</returns>
        public DataTable GetTableCache(string key)
        {
            DataTable tabla = new DataTable();
            tabla = (DataTable)System.Web.HttpContext.Current.Cache.Get(key);
            if (tabla == null)  tabla = CreateTableCache(key);
            return tabla;
        }

    }

}

Además inicializo las tablas al comenzar la aplicación en mi  archivo  Global.asax


        void Application_Start(object sender, EventArgs e)

        {
            u.CreateAllTablesCache();
        }

        void Application_End(object sender, EventArgs e)

        {
            u.RemoveALLTablesCache();
        }
Y la utilizo en todas las páginas de este modo.

/************VALIDACION***************/


        string ctrlVal = "", conte = "",msjVal="", p = "NotaIngresoHospital";

 DataTable t =u.GetTableCache("Cache_ConfiguracionPorPagina");
else if (u.Vl(t, p, "ExploracionFisica", ref ctrlVal, ref conte, ref msjVal) && txtExplFisi.Text.Equals("")) return msjVal;

Si  el campo 'ExploracionFisica' de la página 'NotaIngresoHospital' es positivo Y el  textbox  txtExplFisi  se encuentra en  blanco, regresa el mensaje de error y los valores de los controles, que vienen por referencia  y con  estos valores hago mis procedimientos con JavaScript.


Puedes buscar la clase conexión en este mismo blog









No hay comentarios.:

Publicar un comentario