Documentos de Académico
Documentos de Profesional
Documentos de Cultura
1 Threads
1 Threads
Procesos y hebras
Por qu usar hebras y procesos? .......................................9
Procesos ................................................................................15
Ejecucin de procesos............................................. 15
Finalizacin de procesos ......................................... 19
Monitorizacin de procesos ..................................... 21
Operaciones de E/S................................................. 25
Hebras ....................................................................................28
La clase Thread ....................................................... 28
Ejecucin asncrona de delegados .......................... 34
Hebras en aplicaciones Windows ............................ 36
Control de la ejecucin de una hebra ...................... 39
Interrupcin de la ejecucin de una hebra............... 42
Mecanismos de sincronizacin............................................44
Acceso a recursos compartidos............................... 44
Monitores ................................................................. 45
Cerrojos: La sentencia lock...................................... 47
Otros mecanismos de sincronizacin ...................... 52
Operaciones asncronas .......................................... 54
Referencias ............................................................................58
Procesos y hebras
http://csharp.ikor.org/
10
De cara al usuario
El principal objetivo del uso de hebras o procesos es mejorar el rendimiento del sistema, y
ste no siempre se mide en la cantidad de trabajo que se realiza por unidad de tiempo. De
hecho, el uso de hebras se usa a menudo para reducir el tiempo de respuesta de una
aplicacin. En otras palabras, se suelen utilizar hebras para mejorar la interfaz de usuario.
Aunque pueda resultar sorprendente, el usuario es el principal motivo por el que utilizaremos
hebras en nuestras aplicaciones con cierta asiduidad. Cuando una aplicacin tiene que realizar
alguna tarea larga, su interfaz debera seguir respondiendo a las rdenes que el usuario
efecte. La ventana de la aplicacin debera refrescarse y no quedarse en blanco (como pasa
demasiado a menudo). Los botones existentes para cancelar una operacin deberan cancelar
la operacin de un modo inmediato (y no al cabo de un rato, cuando la operacin ya ha
terminado de todos modos).
Jakob Nielsen, por ejemplo, en su libro Usability Engineering, sugiere que cualquier
operacin que se pueda efectuar en una dcima de segundo puede considerarse inmediata, por
lo que no requerir mayor atencin por nuestra parte al implementar la aplicacin. Cuando
una operacin se puede realizar por completo en un segundo, se puede decir que no
interrumpe demasiado el trabajo del usuario, si bien resulta conveniente utilizar, al menos,
una barra de progreso. Sin embargo, en operaciones que tarden varios segundos, el usuario
difcilmente mantendr su atencin y ser aconsejable que nuestra aplicacin sea capaz de
efectuar la operacin concurrentemente con otras tareas.
En demasiadas ocasiones, las aplicaciones no le prestan atencin debida al usuario. Mientras
estn ocupadas haciendo algo, no responden como debieran a las solicitudes que realiza el
usuario. Con frecuencia, adems, la aplicacin no responde pese a que lo que queramos hacer
no tenga nada que ver con la tarea que nos est obstaculizando el trabajo. Estos bloqueos
temporales se suelen deber a que la aplicacin est programada con una nica hebra y,
mientras dura una operacin costosa, esa hebra bloquea todo lo dems, congelando la interfaz
Fernando Berzal, Francisco J. Cortijo & Juan Carlos Cubero
Procesos y hebras
11
http://csharp.ikor.org/
12
Procesos y hebras
13
Paralelismo real
Un sistema multiprocesador dispone de varias CPUs, por lo cual, si nuestra aplicacin
somos capaces de descomponerla en varias hebras o procesos, el sistema operativo
podr asignar cada una a una de las CPUs del sistema una tarea diferente.
Aunque los multiprocesadores an no son demasiado comunes, un PC convencional
incorpora cierto grado de paralelismo. Se pueden encontrar procesadores
especializados en una tarjeta grfica, coprocesadores matemticos con sus
correspondientes juegos de instrucciones (por ejemplo, las extensiones MMX de Intel)
y procesadores digitales de seales en cualquier tarjeta de sonido. Todos estos
dispositivos permiten realizar tareas de forma paralela, si bien suelen requerir
conocimientos detallados de programacin a bajo nivel.
Aparte de los mltiples coprocesadores mencionados, los microprocesadores actuales
suelen incluir soporte para el uso de hebras. Por ejemplo, algunas versiones del
Pentium 4 de Intel incluyen Simultaneous MultiThreading (SMT), que Intel denomina
comercialmente Hyper-Threading. Con SMT, un nico microprocesador puede
funcionar como si fuesen varios procesadores desde el punto de vista lgico. No se
consigue el mismo rendimiento que con un multiprocesador, pero se mantienen ms
ocupadas las distintas unidades de ejecucin de la CPU, que en cualquier procesador
superescalar ya estn preparadas para ejecutar varias operaciones en paralelo.
Tambin podemos repartir el trabajo entre distintas mquinas de un sistema distribuido
para mejorar el rendimiento de nuestras aplicaciones. Un cluster, por ejemplo, es un
conjunto de ordenadores conectados entre s al que se accede como si se tratase de un
nico ordenador (igual que un multiprocesador incluye varios procesadores
controlador por un nico sistema operativo).
Tanto si disponemos de un multiprocesador como si disponemos de un procesador
SMT o de un cluster, el paralelismo que se obtiene es real, pero su aprovechamiento
requiere el uso de hebras y procesos en nuestras aplicaciones .
14
combinacin de dos actividades: hacer ejercicio y mantener una dieta equilibrada. Ambas
deben realizarse durante el mismo perodo de tiempo, aunque no necesariamente a la vez. No
tendra demasiado sentido hacer ejercicio una temporada sin mantener una dieta equilibrada y
luego dejar de hacer ejercicio para ponernos a dieta.
En situaciones de este tipo, la concurrencia de varias actividades es la solucin adecuada para
un problema. En este sentido, la solucin natural al problema implica concurrencia. Desde un
punto de vista ms informtico, la descomposicin de una aplicacin en varias hebras o
procesos puede resultar muy beneficiosa por varios motivos:
- Se puede simplificar la implementacin de un sistema si una secuencia compleja
de operaciones se puede descomponer en una serie de tareas independientes que
se ejecuten concurrentemente.
- La independencia entre las tareas permite que stas se implementen por separado
y se reduzca el acoplamiento entre las distintas partes del sistema
- Un diseo modular facilita la incorporacin de futuras ampliaciones en nuestras
aplicaciones.
Aviso importante
El objetivo principal del uso de paralelismo es mejorar el rendimiento del sistema.
Salvo que un diseo con hebras y procesos simplifique el trabajo del programador, lo
cual no suele pasar a menudo en aplicaciones de gestin, para qu nos vamos a
engaar, el diseador de una aplicacin concurrente deber analizar hasta qu punto
compensa utilizar hebras y procesos.
Los programas diseados para aprovechar el paralelismo existente entre tareas que se
ejecutan concurrentemente deben mejorar el tiempo de respuesta o la carga de trabajo
que puede soportar el sistema. Esto requiere comprobar que no se est sobrecargando
el sistema al usar hebras y procesos. Un nmero excesivo de hebras o procesos puede
llegar a degradar el rendimiento del sistema si se producen demasiados fallos de
pgina al acceder a memoria o se pierde demasiado tiempo de CPU realizando
cambios de contexto. Adems, el rendimiento de una aplicacin concurrente tambin
puede verse seriamente afectado si no se utilizan adecuadamente los mecanismos de
sincronizacin que son necesarios para permitir que distintas hebras y procesos
accedan a recursos compartidos. Tal vez, en un exceso de celo por nuestra parte,
hebras y procesos pasen bloqueados ms tiempo del estrictamente necesario, con lo
cual no se consigue el paralelismo perseguido.
En cualquier caso, ya hemos visto que el uso de paralelismo, y de hebras en particular,
es ms comn de lo que podra pensarse en un principio (y menos de lo que debera).
Una vez motivado su estudio, slo nos falta ver cmo se pueden utilizar procesos y
hebras en la plataforma .NET, tema al que dedicaremos el resto de este captulo.
Procesos y hebras
15
Procesos
La biblioteca de clases de la plataforma .NET incluye una clase, la clase
System.Diagnostics.Process, que es la que nos permite crear procesos, controlar en
cierto modo su ejecucin e incluso acceder a informacin acerca de los procesos que se estn
ejecutando en el sistema (la misma informacin a la que se puede acceder a travs del
Administrador de Tareas de Windows).
Ejecucin de procesos
La clase Process incluye un mtodo esttico que podemos utilizar para comenzar la
ejecucin de un proceso de distintas formas: el mtodo Process.Start().
El uso del mtodo Process.Start() equivale a la realizacin de una llamada a la funcin
ShellExecute. Esta funcin, perteneciente al API estndar de Windows (Win32), es la
que se utiliza en la familia de sistemas operativos de Microsoft para lanzar un proceso. Este
hecho convierte al mtodo Process.Start() en un mtodo muy verstil que se puede
utilizar de distintas formas, tal como veremos a continuacin:
http://csharp.ikor.org/
16
Tipo de fichero
.bat
.cmd
.com
.cpl
.dll
.exe
Aplicacin
.lnk
.msi
Paquete de instalacin
.pif
.scr
Salvapantallas
Para ejecutar un proceso basta, pues, con indicar el fichero que contiene el punto de entrada a
la aplicacin. Si este fichero est en un directorio incluido en el PATH del sistema, ser
suficiente con indicar el nombre del fichero. No har falta poner su ruta de acceso porque el
PATH es una variable de entorno que le indica al sistema operativo en qu carpetas o
directorios ha de buscar los programas. El ejemplo siguiente muestra cmo podemos abrir una
ventana del Internet Explorer desde nuestra aplicacin de un modo trivial:
System.Diagnostics.Process.Start("iexplore.exe");
Procesos y hebras
17
Adems de permitirnos indicar los parmetros que queremos pasarle a un proceso, podemos
especificar algunas propiedades adicionales sobre la ejecucin de un proceso. Para ello,
podemos recurrir a la clase ProcessStartInfo, que al igual que Process, tambin se
encuentra en el espacio de nombres System.Diagnostics. Mediante esta clase podemos
especificar el directorio en el que se ejecutar el proceso (WorkingDirectory) o el estado
de la ventana que se utilizar al iniciar el proceso (WindowStyle), tal como muestra el
siguiente ejemplo:
using System.Diagnostics;
...
ProcessStartInfo info = new ProcessStartInfo();
info.FileName = "iexplore.exe";
info.Arguments = "http://csharp.ikor.org/";
info.WindowStyle = ProcessWindowStyle.Maximized;
Process.Start(info);
http://csharp.ikor.org/
18
En el ejemplo hemos utilizado una de las ventanas de dilogo tpicas de Windows, al que en
la
plataforma
.NET
se
accede
mediante
la
clase
System.Windows.Forms.OpenFileDialog. Esta clase nos permite seleccionar un
fichero, especificando incluso los filtros que le permitimos usar al usuario para ver
determinados tipos de ficheros. Estos filtros se especifican en el formato
descripcin|plantilla, donde la descripcin corresponde a la cadena de caracteres que le
aparecer al usuario en una desplegable y la plantilla indica el nombre que han de tener los
ficheros del tipo indicado por el filtro.
Igual que existe un componente para mostrar un dilogo que
nos permita seleccionar un fichero (OpenFileDialog),
existen otros dilogos estndar que nos permiten guardar un
fichero (SaveFileDialog), seleccionar una carpeta
(FolderBrowserDialog),
escoger
un
color
(ColorDialog), elegir un tipo de letra (FontDialog),
especificar dnde y cmo queremos imprimir algo
(PrintDialog) o configurar la pgina para un trabajo de
impresin (PageSetupDialog). Todos estos componentes
forman parte de la biblioteca de "dilogos comunes" de
Windows
y
heredan
de
la
clase
System.Windows.Forms.CommonDialog.
Igual que se puede abrir un fichero cualquiera utilizando Process.Start(), siempre y
cuando la extensin del fichero haya sido registrada previamente en el Registro de Windows,
tambin se puede abrir una URL cualquiera. Las URLs comienzan por la especificacin de un
protocolo y los protocolos ms comunes tambin aparecen en el Registro de Windows:
http: y https: para acceder a pginas web, ftp: para acceder a un servidor de archivos,
telnet: para conectarse de forma remota a una mquina, mailto: para enviar un mensaje
de correo electrnico o nntp: para acceder a un servidor de noticias, por poner algunos
ejemplos. Por tanto, para abrir una pgina de Internet desde nuestra aplicacin, podemos usar
directamente su URL en la llamada al mtodo Process.Start():
System.Diagnostics.Process.Start("http://csharp.ikor.org/");
Cuando llamamos a este mtodo usando un nombre de un fichero o una URL, en realidad lo
que hacemos es ejecutar el proceso asociado a abrir ese fichero tal como aparezca en el
Registro de Windows. Podemos ejecutar la aplicacin regedit.exe para ver el contenido
del Registro de Windows en nuestra mquina. Dentro del apartadoMi
PC/HKEY_CLASSES_ROOT aparecen los distintos tipos de archivo registrados. Al registrar
un tipo de archivo, aparte de indicar el programa con el que ha de abrirse, tambin se puede
especificar cmo han de realizarse otras operaciones con el archivo. Es lo que se conoce con
el nombre de "verbos del shell", algunos de los cuales se reproducen a continuacin:
Fernando Berzal, Francisco J. Cortijo & Juan Carlos Cubero
Procesos y hebras
19
Verbo
Descripcin
open
edit
Editar un documento
Imprimir un documento
explore
find
El verbo open corresponde a la accin que se realiza cuando hacemos click con el ratn
sobre el nombre de un fichero, si bien se pueden definir cuantas acciones deseemos. Para
efectuar esas acciones, lo nico que tenemos que hacer es recurrir de nuevo a la clase
ProcessStartInfo:
using System.Diagnostics;
...
ProcessStartInfo info = new ProcessStartInfo();
info.FileName = "Ade.jpg";
info.WorkingDirectory = "f://";
info.Verb = "edit"; // vs. "open" || "print"
Process.Start(info);
Si nos interesase crear nuestro propio tipo de archivos, podemos hacer que el usuario no tenga
que abrir nuestra aplicacin para despus acceder al fichero con el que quiera trabajar.
Windows nos ofrece la posibilidad de utilizar su interfaz orientada a documentos. Slo
tenemos que crear entradas correspondientes en el Registro, lo que podemos hacer desde
nuestra propia aplicacin por medio de las clases Microsoft.Win32.Registry y
Microsoft.Win32.RegistryKey, que son las que nos permiten manipular el Registro
del sistema.
Finalizacin de procesos
Cuando se utiliza el mtodo Process.Start() para iniciar la ejecucin de un proceso,
tambin se debera llamar al mtodo Process.Close() para liberar todos los recursos
asociados al objeto de tipo Process. En los ejemplos del apartado anterior, podramos
liberar los recursos asociados al proceso lanzado siempre y cuando antes nos asegursemos de
que la ejecucin del proceso ha finalizado. Para comprobar que la ejecucin de un proceso ha
finalizado disponemos de varias alternativas, como veremos a continuacin.
http://csharp.ikor.org/
20
Si no nos importase que nuestra aplicacin detuviese su ejecucin hasta que el proceso
lanzado finalizase por completo, nos bastara con llamar al mtodo WaitForExit:
No obstante, no parece muy buena idea detener la ejecucin de nuestra aplicacin de forma
indefinida. El subproceso podra "colgarse"; esto es, entrar en un bucle infinito o se quedarse
esperando la ocurrencia de algn tipo de evento de forma indefinida. Si esto ocurriese, nuestra
aplicacin dejara de responder. Para evitar esta posibilidad, podemos limitar la espera a un
perodo de tiempo mximo. Utilizando de nuevo el mtodo WaitForExit, lo nico que
tenemos que hacer es incluir un parmetro en el que indicamos el nmero de milisegundos
que estamos dispuestos a esperar.
Ejecucin invisible de comandos MS-DOS
Process proceso = new Process()
string
string
proceso.StartInfo.FileName = "cmd.exe";
proceso.StartInfo.Arguments =
"/C dir \""+path+"\" >> \"" + salida +"\" && exit";
proceso.StartInfo.WindowStyle = ProcessWindowStyle.Hidden;
proceso.StartInfo.CreateNoWindow = true;
proceso.Start()
proceso.WaitForExit(1000);
if (!proceso.HasExited()) {
proceso.Kill();
} else {
...
}
// 1 segundo de timeout
// Finalizamos el proceso
El ejemplo muestra cmo podemos lanzar una secuencia de comandos MS-DOS desde
nuestra aplicacin. Si el proceso no termina en el tiempo mximo que le indicamos, lo que
comprobamos llamando al mtodo HasExited(), entonces podemos obligar la finalizacin
de la ejecucin del proceso con una llamada al mtodo Kill().
El mtodo HasExited() tambin puede emplearse para comprobar la finalizacin de la
ejecucin de un proceso sin tener que bloquear nuestra aplicacin con una llamada a
Procesos y hebras
21
WaitForExit(), pero tenemos que ser nosotros los que peridicamente sondeemos el
estado del proceso para comprobar su terminacin.
Existe una tercera opcin para comprobar la finalizacin de la ejecucin de un proceso:
aprovechar el evento Exited incluido en la clase Process. De esta forma, no tenemos que
preocuparnos de bloquear la ejecucin de nuestra aplicacin mientras esperamos ni de
sondear peridicamente el estado del proceso. El evento Exited se produce
automticamente cuando termina la ejecucin del proceso, por lo que podemos escribir un
manejador para este evento que realice las tareas oportunas en el momento de la finalizacin
de la ejecucin del proceso. En el siguiente ejemplo, el mtodo ExitHandler se limita
mostrar un mensaje en el que figura el momento en el que termin la ejecucin del proceso:
proceso.Close();
}
Monitorizacin de procesos
Como un lector observador ya habr notado, la clase Process no slo permite controlar la
ejecucin de un proceso, sino que tambin nos ofrece la posibilidad de acceder a informacin
http://csharp.ikor.org/
22
relativa al proceso. Esta informacin es la misma que podemos obtener del Administrador de
Tareas de Windows.
Tal como se muestra en la imagen, en este apartado construiremos un sencillo monitor de
procesos en el que podamos ver datos relativos a los distintos procesos que se estn
ejecutando en nuestra mquina.
Monitor de procesos
Nuestro monitor ser una sencilla aplicacin Windows con un nico formulario. Slo tenemos
que crear la aplicacin y aadir los componentes que se listan en la siguiente tabla:
Componente
Tipo
Propiedad
Valor
formProcess
Form
Text
listProcesses
ListBox
ScrollAlwaysVisible
true
textInfo
TextBox
Multiline
true
BackColor
Info
Text
WindowClose
ForeColor
ForestGreen
buttonClose
Button
Procesos y hebras
23
Componente
Tipo
Propiedad
Valor
buttonKill
Button
Text
Kill
ForeColor
DarkRed
TextAlign
MiddleCenter
Text
labelWarning
Label
Como se puede ver, la clase Process incluye numerosas propiedades que nos permiten
controlar el estado y los recursos utilizados por un proceso.
Finalmente, implementamos el cdigo necesario para forzar la finalizacin de un proceso
http://csharp.ikor.org/
24
Este pequeo ejemplo nos ha servido para mostrar la informacin que se puede obtener de un
proceso en ejecucin a travs de la clase Process. Como vimos en los apartados anteriores,
Process es tambin la clase utilizada para controlar la ejecucin de los procesos del
sistema. En el apartado siguiente, para completar nuestro recorrido por el manejo de procesos
en la plataforma .NET, veremos una ltima aplicacin de la clase Process: la redireccin de
los flujos de entrada y de salida estndar de un proceso.
Nota
En el cuadro de herramientas de Visual Studio .NET, dentro
del apartado "Componentes", podemos encontrar un
componente denominado Process que nos permite, si as lo
deseamos,
trabajar
con
objetos
de
la
clase
System.Diagnostics.Process de forma "visual",
entre comillas porque tampoco se puede decir que se
automatice demasiado el uso del componente Process al
trabajar con l dentro del diseador de formularios.
Procesos y hebras
25
Operaciones de E/S
En ciertas ocasiones, nos puede interesar que la entrada de un proceso provenga directamente
de la salida de otro proceso diferente. Como se mencion al motivar el uso de hebras y
procesos, enviar la salida de un proceso a un fichero y, despus, leer ese fichero no siempre es
la opcin ms eficiente. Los sistemas operativos suelen proporcionar un mecanismo mejor
para conectar unos procesos con otros a travs de sus canales estndares de entrada y salida de
datos: los "pipes".
Un programa estndar, tanto en Windows como en UNIX, posee por defecto tres canales de
entrada/salida: la entrada estndar StdIn (asociada usualmente al teclado del ordenador), la
salida estndar StdOut y un tercer canal, denominado StdErr, que tambin es de salida y
se emplea, por convencin, para mostrar mensajes de error.
Como no poda ser de otra forma, System.Diagnostics.Process nos permite
redireccionar los tres canales de E/S estndar (StdIn, StdOut y StdErr). Slo tenemos
que volver a usar la clase auxiliar ProcessStartInfo. Fijando a true las propiedades
RedirectStandardInput,
RedirectStandardOutput
y
RedirectStandardError, podemos acceder a las propiedades StandardInput,
StandardOutput y StandardError del proceso. Estas ltimas son objetos de tipo
StreamReader (la entrada estndar) o StreamWriter (las dos salidas), los mismos que
se utilizan en la plataforma .NET para trabajar con ficheros.
Slo tenemos que tener en cuenta un ltimo detalle para poder redireccionar la entrada o la
salida de un proceso en C#. Como ya mencionamos antes, Process.Start() utiliza por
defecto la funcin ShellExec del API Win32. Esta funcin delega en el shell de Windows
la decisin de qu hacer para acceder un fichero. En el caso de un fichero ejecutable, lo que
har ser ejecutarlo. No obstante, cuando utilicemos redirecciones, la propiedad
ProcessStartInfo.UseShellExecute debe estar puesta a false para crear
directamente el proceso a partir del fichero ejecutable que hayamos indicado en la propiedad
FileName y que las redirecciones se hagan efectivas.
Por ejemplo, podemos ejecutar comandos MS-DOS redireccionando la entrada y las salidas
del proceso cmd.exe, el que se utiliza para acceder al "smbolo del sistema" en Windows.
En concreto, el siguiente fragmento de cdigo nos muestra un mensaje con un listado del
contenido del directorio raz de la unidad C:\:
26
process.Start()
StreamWriter sIn = proceso.StandardInput;
StreamReader sOut = proceso.StandardOutput;
StreamReader sErr = proceso.StandardError;
sIn.AutoFlush = true;
sIn.Write("dir c:\" + System.Environment.NewLine);
sIn.Write("exit" + System.Environment.NewLine);
string output = sOut.ReadToEnd();
sIn.Close();
sOut.Close();
sErr.Close();
proceso.Close();
MessageBox.Show(output);
Obviamente, el mecanismo anterior slo nos permite controlar las operaciones de E/S de las
aplicaciones en modo consola tpicas de MS-DOS. Las aplicaciones Windows no funcionan
de la misma forma, sino que consisten, bsicamente, en un proceso que va recibiendo los
eventos generados por el sistema operativo y va realizando las operaciones oportunas en
funcin del evento que se haya producido. Los eventos pueden ser de muchos tipos, desde
mensajes para mover, maximizar o minimizar la ventana de la aplicacin, hasta los mensajes
que se producen cuando se pulsa un botn, se hace click con el ratn o se selecciona un
elemento de una lista.
Cada vez que se pulsa una tecla, tambin se produce un evento, que la aplicacin se encargar
de interpretar en funcin del contexto en el que se produzca. Desde nuestras aplicaciones
tambin se pueden generar eventos de forma artificial, de modo que podemos simular la
pulsacin de teclas por parte del usuario. Esto nos permite conseguir un efecto similar al que
se consigue al redireccionar la entrada estndar de una aplicacin en modo consola. Mediante
la clase SendKeys, del espacio de nombres System.Windows.Forms, podemos
automatizar la pulsacin de teclas. A continuacin, se muestra un pequeo ejemplo en el que
el Bloc de Notas funciona "solo":
Procesos y hebras
27
visualice la ventana
la
llamada
a
la aplicacin pueda
es anlogo al de
Con un pequeo retardo entre pulsacin y pulsacin, podra simularse a una persona tecleando
algo. Este retardo lo podramos conseguir con al mtodo Thread.Sleep(), aunque el
control de la ejecucin de una hebra es algo que dejaremos para el siguiente apartado.
Acceso a recursos nativos de Windows
Con SendKeys se puede simular la pulsacin de cualquier combinacin de teclas. No
obstante, SendKeys se limita a enviar eventos de pulsacin de teclas a la aplicacin
que est activa en cada momento. Y no existe ninguna funcin estndar en .NET que
nos permita establecer la aplicacin activa, pero s en el API nativo de Windows.
El API de Windows incluye una gran cantidad de funciones (no mtodos), del orden de
miles. En el caso que nos ocupa, podemos recurrir a las funciones FindWindow y
SetForegroundWindow del API Win32 para establecer la aplicacin activa. Como
estas funciones no forman parte de la biblioteca de clases .NET, tenemos que acceder a
ellas usando los servicios de interoperabilidad con COM. Estos servicios son los que
nos permiten acceder a recursos nativos del sistema operativo y se pueden encontrar en
el espacio de nombres System.Runtime.InteropServices.
Para poder usar la funcin SetForegroundWindow, por ejemplo, tendremos que
escribir algo parecido a lo siguiente:
using System.Runtime.InteropServices;
...
[DllImport("User32",EntryPoint="SetForegroundWindow")]
private static extern bool SetForegroundWindow(System.IntPtr hWnd);
http://csharp.ikor.org/
28
Hebras
Para dotar de cierto paralelismo a nuestras aplicaciones, no es necesario dividirlas en procesos
completamente independientes. Dentro de un proceso, podemos tener varias hebras
ejecutndose concurrentemente. Con las hebras se gana algo de eficiencia, al resultar ms
eficientes tanto los cambios de contexto como la comunicacin entre hebras. No obstante, al
compartir el mismo espacio de memoria, deberemos ser cuidadosos al acceder a recursos
compartidos, para lo que tendremos que utilizar los mecanismos de sincronizacin que
veremos ms adelante.
En la plataforma .NET, las clases incluidas en el espacio de nombres System.Threading,
junto con la palabra reservada lock en el lenguaje de programacin C#, nos proporcionan
toda la infraestructura necesaria para crear aplicaciones multihebra. De hecho, todas las
aplicaciones .NET son en realidad aplicaciones multihebra. El recolector de basura no es ms
que una hebra que se ejecuta concurrentemente con la hebra principal de la aplicacin y va
liberando la memoria que ya no se usa.
La clase Thread
En este apartado veremos cmo se crean hebras y se trabaja con ellas en la plataforma .NET.
En esta plataforma, una hebra se representa mediante un objeto de la clase Thread, que
forma parte del espacio de nombres System.Threading. En este espacio de nombres
tambin se encuentra el delegado sin parmetros ThreadStart, que se emplea para
especificar el punto de entrada a la hebra (algo as como la funcin main de una hebra). Este
delegado se le pasa como parmetro al constructor de la clase Thread para crear una hebra.
Para comenzar la ejecucin de la hebra, se ha de utilizar el mtodo Start() de la clase
Thread. Este mtodo inicia la ejecucin concurrente del delegado que sirve de punto de
entrada de la hebra. Como este delegado no tiene parmetros, el estado inicial de la hebra hay
que establecerlo previamente. Para establecer su estado se pueden utilizar las propiedades y
mtodos del objeto en que se aloje el delegado. El siguiente fragmento de cdigo muestra
cmo se implementa todo este proceso en C#:
Procesos y hebras
29
El fragmento de cdigo anterior presupone que el objeto que encapsula la hebra, de tipo
Tarea tiene una propiedad Param que se utiliza como parmetro en la ejecucin de la hebra
y un mtodo sin parmetros Ejecutar que sirve de punto de entrada de la hebra.
Al final del fragmento de cdigo anterior incluimos una llamada al mtodo Join() de la
hebra. Este mtodo detiene la ejecucin de la hebra actual hasta que finalice la ejecucin de la
hebra que se lanz con la llamada a Start(). En la prctica, no obstante, Join() no se
utiliza demasiado precisamente porque bloquea la ejecucin de la hebra actual y anula las
ventajas de tener varias hebras ejecutndose en paralelo. Generalmente, slo usaremos
Join() cuando queramos finalizar la ejecucin de nuestra aplicacin. En este caso, antes de
terminar la ejecucin de la hebra principal deberemos, de algn modo, finalizar la ejecucin
de las hebras que hemos lanzado desde la hebra principal de la aplicacin.
Establecimiento de prioridades
Como se mencion en la primera parte de este captulo, en ocasiones resulta aconsejable
asignarles distintas prioridades a las hebras de nuestra aplicacin. Antes de llamar al mtodo
Start(), podemos establecer la prioridad de una hebra mediante la propiedad Priority
de la clase Thread. Esta propiedad es del tipo definido por la enumeracin
ThreadPriority y puede tomar cinco valores diferentes:
Nivel de prioridad
Descripcin
Highest
Prioridad mxima
AboveNormal
Normal
BelowNormal
Lowest
Prioridad mnima
http://csharp.ikor.org/
30
Los valores anteriores nos permiten indicarle al sistema operativo qu hebras han de gozar de
acceso preferente a la CPU del sistema. Hemos de tener cuidado al utilizar esta propiedad
porque afecta significativamente a la forma en que se efectan las tareas concurrentemente. Si
nuestra aplicacin tiene una hebra principal que se encarga de recibir los eventos de la interfaz
de usuario y otra auxiliar que hace algn tipo de clculo, lo normal es que la hebra auxiliar
tenga menos prioridad que la hebra principal, con el nico objetivo de que nuestra aplicacin
responda de la manera ms inmediata posible a las acciones que realice el usuario.
Prioridades de los procesos en Windows
Tambin se pueden establecer prioridades para los distintos procesos de nuestra aplicacin.
En este caso, se debe recurrir a la propiedad PriorityClass de la clase
System.Diagnostics.Process. Esta propiedad tiene que tomar uno de los valores
definidos por la enumeracin ProcessPriorityClass, que es distinta a la enumeracin
ThreadPriority:
Clase de prioridad
Prioridad base
Idle
Normal
High
13
RealTime
24
La prioridad de las hebras de un proceso ser relativa a la clase de prioridad del proceso:
- Las hebras de un procesos de prioridad mxima, de clase RealTime, tiene
preferencia incluso sobre algunos procesos importantes del sistema. Si un
procesos con prioridad RealTime tarda algo de tiempo en realizar su tarea, el
resto del sistema puede dejar de responder adecuadamente.
- En el polo opuesto, un proceso de prioridad Idle slo acceder a la CPU cuando
el sistema est completamente inactivo, igual que el salvapantallas.
En la prctica, la asignacin de prioridades a los procesos del sistema es algo ms compleja
porque el sistema operativo puede aumentar temporalmente la prioridad de un proceso bajo
determinadas circunstancias. Por ejemplo, si ponemos a true la propiedad
PriorityBoostEnabled, que por defecto es false, el sistema operativo le dar mayor
prioridad al proceso cuando la ventana principal del mismo tenga el foco de Windows (esto
es, est en primer plano). En cualquier momento, no obstante, podemos conocer cul era la
prioridad inicial asignada a un proceso mediante la propiedad BasePriority, que es un
nmero entero de slo lectura.
Procesos y hebras
31
Un pequeo ejemplo
Supongamos que nuestra aplicacin, por el motivo que sea, ha de realizar una tarea costosa en
trminos de tiempo. La tarea en cuestin puede consistir en un complejo clculo matemtico,
una larga simulacin, la realizacin de una copia de seguridad, el acceso a un recurso a travs
de la red o cualquier otra cosa que requiera su tiempo. Pero a nosotros nos interesa que,
mientras esa tarea se est efectuando, nuestra aplicacin no deje de responder a las rdenes
del usuario. Aqu es donde entra el juego el uso de hebras.
Si nuestro programa es una aplicacin Windows, tendremos un formulario como el siguiente,
en el cual aparece un botn Button y una barra de progreso ProgressBar que sirva para
indicarle al usuario el avance en la realizacin de la tarea:
Siempre es buena idea proporcionarle al usuario pistas visuales en las que se ve cmo avanza
la realizacin de la tarea. Sin embargo, el uso de otro tipo de pistas, como sonidos, no siempre
es
tan
buena
idea.
Afortunadamente,
el
componente
System.Windows.Forms.ProgressBar un mecanismo adecuado para mantener
visible el estado de la aplicacin. Hemos de tener en cuenta que la "visibilidad" del estado de
la aplicacin es un factor importante en la usabilidad de la misma. Adems, esta propiedad no
es exclusiva de las aplicaciones. En un proyecto de desarrollo de software, por ejemplo,
mantener su estado visible es esencial para poder gestionarlo correctamente.
Volvamos ahora a nuestra aplicacin. La tarea pendiente de realizacin, usualmente, se podr
descomponer en una serie de iteraciones que nos servirn para mantener la visibilidad de su
estado. Esas iteraciones pueden consistir, por ejemplo, en realizar un paso en una simulacin,
transmitir un fragmento de los datos que se han de archivar o procesar un subconjunto de los
registros con los que se ha de trabajar. Idealmente, la funcin que realice la tarea tendr un
parmetro proporcional al esfuerzo requerido:
http://csharp.ikor.org/
32
while (hecho<esfuerzo) {
...
hecho++;
MostrarProgreso(hecho, esfuerzo);
}
}
La forma ingenua de implementar nuestra aplicacin sera hacer que, al pulsar el botn
"Calcular", se invocase a la funcin Calcular que acabamos de definir. Sin embargo, esto
lo nico que conseguira es que nuestra aplicacin se quedase bloqueada mientras dure la
operacin. Al cambiar a otra aplicacin y luego volver a la nuestra, nos podramos encontrar
una desagradable sorpresa:
Una imagen como esta es muy comn, por otro lado, en demasiadas aplicaciones. La ventana
"en blanco" se debe a que la aplicacin Windows consiste, inicialmente, en una hebra que va
recibiendo una secuencia de eventos del sistema operativo. Cuando se produce un evento, se
llama al manejador correspondiente de forma sncrona. Hasta que no se vuelva de esa
llamada, no se pasar a tratar el siguiente evento. Si la hebra principal de la aplicacin se
mantiene ocupada en realizar una tarea larga, como respuesta al evento que se produce al
pulsar el botn "Calcular", esta hebra, como es lgico, no puede atender a ningn otro evento
de los que se puedan producir. En el caso de arriba, el evento Paint del formulario, que
solicita refrescar la imagen del formulario en pantalla, se quedar en cola esperando que
termine la tarea asociada a la pulsacin del botn, con lo que la interfaz de la aplicacin se
queda "congelada".
Fernando Berzal, Francisco J. Cortijo & Juan Carlos Cubero
Procesos y hebras
33
Para crear una hebra, necesitaremos un delegado sin parmetros ThreadStart, que ser el
responsable de llamar al mtodo Calcular. En el manejador del evento que lanza la hebra,
se obtienen los parmetros que sean necesarios y se crear la hebra a partir del delegado
ThreadStart, para el cual tenemos que crear un mtodo auxiliar sin parmetros:
http://csharp.ikor.org/
34
Con esta pequea modificacin de la aplicacin inicial, hemos conseguido que nuestra
aplicacin se convierta en una aplicacin multihebra y su interfaz grfica funcione
correctamente. Al menos, aparentemente.
Una vez declarado el delegado como tipo, podemos crear una instancia de ese tipo a partir del
mtodo Calcular, el que se encarga de realizar la tarea para la cual vamos a crear una
hebra auxiliar:
Procesos y hebras
35
El delegado, como tal, lo podemos usar directamente para invocar al mtodo que encapsula
(igual que se invocan los manejadores de eventos usados en los formularios, los cuales
tambin se registran como delegados para responder a los eventos correspondientes). Por
ejemplo, dado el delegado anterior, podramos llamar a la funcin Calcular a travs del
delegado:
delegado (1000);
Cuando un delegado se usa directamente, como hicimos antes, en realidad se est llamando al
mtodo sncrono Invoke, que es el que se encarga de llamar a la funcin concreta con la que
se instancia el delegado (Calcular, en este caso).
Los otros dos mtodos del delegado, BeginInvoke y EndInvoke, son los que permiten
invocar al delegado de forma asncrona. Llamarlo de forma asncrona equivale a crear una
hebra auxiliar y comenzar la ejecucin del delegado en esa hebra auxiliar. Por tanto, podemos
realizar tareas en hebras independientes con slo crear un delegado y llamar a su mtodo
BeginInvoke:
http://csharp.ikor.org/
36
En principio, todo parece funcionar correctamente, aunque an nos quedan algunos detalles
por pulir, como veremos a continuacin.
En este sentido, la documentacin lo deja bastante claro: Slo hay cuatro mtodos de un
control que se pueden llamar de forma segura desde cualquier hebra. Estos mtodos son
Invoke, BeginInvoke, EndInvoke y CreateGraphics. Cualquier otro mtodo ha
de llamarse a travs de uno de los anteriores.
En la prctica, para modificar las propiedades de un control desde una hebra distinta a aqulla
que lo cre, lo nico que tenemos que hacer es volver a utilizar delegados. En el ejemplo de
antes, la hebra auxiliar delegar en la hebra principal de la aplicacin la tarea de actualizar el
estado de la barra de progreso. Para ello, lo nico que tenemos que hacer es aprovechar el
mtodo Invoke del control, que sirve para llamar de forma sncrona a un delegado desde la
Fernando Berzal, Francisco J. Cortijo & Juan Carlos Cubero
Procesos y hebras
37
El uso de Invoke nos garantiza que accedemos de forma segura a los controles de nuestra
ventana en una aplicacin multihebra. La hebra principal crea una hebra encargada de realizar
una tarea computacionalmente costosa, la hebra auxiliar, y sta le pasa el control a la hebra
principal cada vez que necesita actuar sobre los controles de la interfaz, de la que es
responsable la hebra principal de la aplicacin.
http://csharp.ikor.org/
38
Tener que llamar al mtodo Invoke del control cada vez que queremos garantizar el correcto
funcionamiento de nuestra aplicacin multihebra resulta bastante incmodo y, adems, es
bastante fcil que se nos olvide hacerlo en alguna ocasin. Por tanto, no es mala idea que sea
la propia funcin a la que llamamos la que se encargue de asegurar su correcta ejecucin en
una aplicacin multihebra, llamando ella misma al mtodo Invoke.
En el ejemplo anterior, el mtodo MostrarProgreso lo podemos implementar de la
siguiente forma:
void MostrarProgreso (int actual, int total)
{
if ( progress.InvokeRequired == false ) {
// Hebra correcta
progress.Maximum = total;
progress.Value
= actual;
} else {
// Llamar a la funcin desde la hebra adecuada
ProgressDelegate progress = new ProgressDelegate(MostrarProgreso);
this.Invoke (progress, new object[] { actual, total} );
}
}
Si el mtodo devolviese algn valor, entonces habra que recurrir al uso de EndInvoke, si
bien este no es el caso porque lo nico que hace el mtodo es actualizar la interfaz de usuario
desde el lugar adecuado para ello: la hebra principal de la aplicacin.
Siempre que podamos, el uso de BeginInvoke es preferible al de Invoke porque el
mtodo BeginInvoke evita que se puedan producir bloqueos. Al final de este captulo
veremos otro ejemplo del uso de llamadas asncronas en la plataforma .NET, si bien antes nos
Fernando Berzal, Francisco J. Cortijo & Juan Carlos Cubero
Procesos y hebras
39
falta por ver cmo se puede controlar la ejecucin de las hebras y cmo se emplean
mecanismos de sincronizacin para solucionar el problema del acceso a recursos compartidos
en situaciones ms generales.
La hebra estar en estado Inactivo cuando no est ejecutndose, Activo cuando est
ejecutndose y Cancelando cuando est ejecutndose y el usuario haya solicitado la
finalizacin de su ejecucin.
Un programador con experiencia en el desarrollo de
aplicaciones multihebra con herramientas de Borland como
Delphi o C++Builder se dar cuenta de que lo que vamos a
hacer no es ms que implementar el mecanismo que
suministra la clase TThread con el mtodo Terminate()
y la propiedad Terminated.
http://csharp.ikor.org/
40
Si optamos por controlar la ejecucin de la hebra desde un nico botn de nuestra interfaz de
usuario, este botn variar su comportamiento en funcin del estado actual de la hebra.
}
}
Obsrvese que, mientras la hebra no detenga su ejecucin, tenemos que deshabilitar el botn
con el fin de evitar que, por error, se cree una nueva hebra en el intervalo de tiempo que pasa
desde que el usuario pulsa el botn "Cancelar" hasta que la hebra realmente se detiene.
Para que la hebra detenga su ejecucin, sta ha de comprobar peridicamente que su estado es
distinto de Cancelando, el estado al que llega cuando el usuario le indica que quiere
finalizar su ejecucin. En este caso, el estado de la hebra es una variable compartida entre las
dos hebras. En general, deberemos tener especial cuidado con el acceso a recursos
compartidos y sincronizar el acceso a esos recursos utilizando mecanismos de sincronizacin
como los que veremos al final de este captulo. En este caso, no obstante, no resulta necesario
porque slo usamos ese recurso compartido como indicador. Una hebra establece su valor y la
otra se limita a sondear peridicamente dicho valor. La hebra principal establece el estado de
la hebra auxiliar y la hebra auxiliar simplemente comprueba su estado para decidir si ha de
finalizar su ejecucin o no.
Procesos y hebras
41
try {
while ( (estado!=Estado.Cancelando) && ...) {
...
}
} finally {
this.Invoke(endProgress);
}
}
private void EndProgress ()
{
estado = Estado.Inactivo;
buttonCalc.Text = "Calcular";
buttonCalc.Enabled = true;
}
http://csharp.ikor.org/
42
Procesos y hebras
43
http://csharp.ikor.org/
44
Mecanismos de sincronizacin
El cdigo ejecutado por una hebra debe tener en cuenta la posible existencia de otras hebras
que se ejecuten concurrentemente, especialmente si esa hebra comparte determinados recursos
con otras hebras. Dado que las hebras se ejecutan en el mismo espacio de memoria, dentro del
proceso del que son "subprocesos", hay que tener especial cuidado para evitar que dos hebras
accedan a un recurso compartido al mismo tiempo. Este recurso compartido, usualmente, ser
un objeto al que las distintas hebras acceden travs de un miembro esttico de alguna clase (el
equivalente a las variables globales en orientacin a objetos) o de algn otro punto de acceso
comn.
Si bien, idealmente, podra pensarse que las hebras deberan ser completamente
independientes, en la prctica esto no es as. La ejecucin de una hebra puede depender del
resultado de las tareas que realicen otras hebras. De hecho, lo normal es que tenga que existir
un mnimo de cooperacin en la ejecucin de las hebras y esa cooperacin se realizar a
travs de recursos compartidos. En otras palabras, las distintas hebras de una aplicacin han
de coordinar su ejecucin y los mecanismos de sincronizacin son necesarios para ello.
Nota
El modelo de sincronizacin utilizado en la plataforma .NET
es completamente anlogo al utilizado por el lenguaje de
programacin Java. Afortunadamente para el programador,
que ya no tiene que hacer malabarismos con los distintos
modelos de ejecucin del modelo COM usado en Windows
(salvo que tenga que utilizar los mecanismos de
interoperabilidad con COM, claro est).
Procesos y hebras
45
propiedades, sus invariantes. Esos invariantes se mantienen cuando lo nico que hacemos es
consultar el estado del objeto (una simple operacin de lectura), pero puede que no se
verifiquen siempre cuando llamamos a un mtodo que actualiza el estado del objeto (el
equivalente orientado a objetos de una operacin de escritura). En este ltimo caso, se hace
necesario que la hebra que actualiza el estado del objeto bloquee el acceso a ese objeto, al
menos mientras dure la operacin de actualizacin.
Cuando se bloquea el acceso al objeto, slo una hebra puede manipularlo y as nos
aseguramos de que ninguna otra hebra obtiene algo inconsistente al consultar el estado del
objeto. En otras palabras, las aplicaciones multihebra evitan el acceso concurrente a recursos
compartidos utilizando mecanismos de exclusin mutua. En realidad, slo resulta necesario
garantizar la exclusin mutua a aquellos objetos que pueden cambiar de estado.
Si no se bloquea el acceso a un objeto mientras se ste se modifica, el resultado de la
ejecucin de un programa depender de la forma en que se entrelacen las operaciones de las
distintas hebras. Y, como ya sabemos, eso es algo que no se puede predecir y que depender
de la situacin, por lo que este tipo de errores son difciles de detectar y de reproducir.
El mecanismo ms comn para garantizar la exclusin mutua consiste en el uso de secciones
crticas. Una seccin crtica delimita un fragmento de cdigo al cual slo puede acceder una
hebra en cada momento. En el caso del lenguaje de programacin C#, las secciones crticas
las podemos implementar con monitores o con cerrojos, que son los dos mecanismos de
exclusin mutua que veremos en los prximos dos apartados.
Por un lado, hay que bloquear el acceso concurrente a objetos
compartidos. Por otro lado, hay que tener cuidado de no
bloquear innecesariamente la ejecucin de otras hebras, con el
objetivo de que la aplicacin sea realmente concurrente y no
simplemente secuencial.
Monitores
Como se acaba de mencionar, la exclusin mutua puede conseguirse utilizando distintos
mecanismos que permiten definir secciones crticas, regiones de cdigo a la que slo puede
encontrarse una hebra en cada momento, garantizando as la exclusin mutua en el acceso a
recursos compartidos. En la plataforma .NET, la clase System.Threading.Monitor se
utiliza para implementar uno de esos mecanismos: los monitores.
Los monitores proporcionan un modelo de coordinacin similar a las secciones crticas que se
pueden encontrar en el propio API del sistema operativo Windows. De hecho, su uso en .NET
es muy similar al uso de secciones crticas en Win32:
http://csharp.ikor.org/
46
Secciones crticas en C#
// Entrada en la seccin crtica
Monitor.Enter(this);
try {
// Acciones que requiran un acceso coordinado
...
} finally {
// Salida de la seccin crtica
Monitor.Exit(this);
}
En este caso, hemos creado una seccin crtica asociada al objeto this, de forma que,
mientras una hebra est ejecutando el fragmento de cdigo incluido entre
Monitor.Enter(this) y Monitor.Exit(this), ninguna otra hebra podr acceder a
la seccin crtica asociada a this. En el caso de intentar hacerlo, quedar detenida en
Monitor.Enter(this) hasta que la hebra que est ejecutando la seccin crtica salga de
ella. Obviamente, otras hebras podrn siempre acceder a secciones crticas que estn
asociadas a otros objetos (distintos de this) mientras las secciones crticas a las que intenten
acceder estn libres.
Obsrvese tambin cmo hemos envuelto la seccin crtica en un bloque try..finally
para garantizar que se sale de la seccin crtica aun cuando se produzca una excepcin
mientras se ejecutan las sentencias incluidas en las seccin crtica. Si no hicisemos y saltase
una excepcin, ninguna otra hebra podra acceder a la seccin crtica y, posiblemente, esto
acabara ocasionando un bloqueo de nuestra aplicacin.
Aparte de poder definir secciones crticas, resulta conveniente disponer de algn mecanismo
que permita bloquear la ejecucin de una hebra hasta que se cumpla una condicin. En
algunas plataformas, este mecanismo de coordinacin lo proporcionan las "variables
condicin". En otras, se dispone de semforos. En la plataforma .NET, la clase
System.Threading.Monitor incluye los mtodos estticos Wait, Pulse y
PulseAll que se utilizan en la prctica con la misma finalidad que los semforos o las
variables condicin.
Cuando una hebra llama al mtodo Monitor.Wait(obj), queda a la espera de que otra
hebra invoque al mtodo Monitor.Pulse(obj) sobre el mismo objeto obj para
proseguir su ejecucin. Cuando la hebra llama a Monitor.Wait(obj), la hebra ha de
tener acceso exclusivo al objeto obj. Esto es, la llamada a Monitor.Wait(obj) estar
dentro de una seccin crtica definida sobre el objeto obj. Si no, se producira una excepcin
de tipo SynchronizationLockException.
Procesos y hebras
47
while (!condicin)
Monitor.Wait(obj);
http://csharp.ikor.org/
48
Procesos y hebras
49
Lectores y escritores
El mtodo Monitor.PulseAll(obj), que puede parecer algo intil a primera vista,
resulta bastante prctico cuando se quiere permitir el acceso concurrente a un recurso
compartido para leer pero se quiere garantizar el acceso exclusivo al recurso cuando se va a
modificar. La lectura concurrente de los datos mejora el rendimiento de una aplicacin
multihebra y sin afectar a la correccin del programa. A continuacin se muestra una posible
implementacin de un mecanismo que permita concurrencia entre lectores pero exija
exclusin mutua a los escritores:
class RWLock
{
int lectores = 0;
public void AcquireExclusive()
{
lock (this) {
while (lectores!=0) Monitor.Wait(this);
lectores--;
}
}
public void AcquireShared()
{
lock (this) {
while (lectores<0) Monitor.Wait(this);
lectores++;
}
}
public void ReleaseExclusive()
{
lock (this) {
lectores = 0;
Monitor.PulseAll(this);
}
}
public void ReleaseShared()
{
lock (this) {
lectores--;
if (lectores==0) Monitor.Pulse(this);
}
}
}
50
La nica regla que hemos de tener en cuenta para utilizar mecanismos de exclusin mutua es
la siguiente: en una aplicacin multihebra, el acceso a objetos compartidos cuyo estado puede
variar ha de hacerse siempre de forma protegida. La proteccin la puede proporcionar
directamente el cerrojo asociado al objeto cuyas variables de instancia se modifican. El acceso
a los datos siempre se debe hacer desde la hebra que tenga el cerrojo del objeto para
garantizar la exclusin mutua. El siguiente ejemplo muestra cmo se debe hacer:
Obviamente, el lenguaje no pone ninguna restriccin sobre el objeto que se utiliza como
parmetro de la sentencia lock. De todas formas, para simplificar el mantenimiento de la
aplicacin se procura elegir el que resulte ms obvio. Si se modifican variables de instancia de
un objeto, se usa el cerrojo de ese objeto. Si se modifican variables estticas de una clase,
entonces se utiliza el cerrojo asociado al objeto que representa la clase
(lock(typeof(Account)), por poner un ejemplo). De esta forma, se emplea siempre el
cerrojo asociado al objeto cuyos datos privados se manipulan. Adems, se evitan de esta
forma situaciones como la siguiente:
lock (this) {
i++;
}
...
lock (otro) {
i--;
}
En este ejemplo, la variable i se manipula en dos secciones crticas diferentes, si bien dichas
secciones no garantizan la exclusin mutua en el acceso a i porque estn definidas utilizando
los cerrojos de dos objetos diferentes. En programacin concurrente, algo de disciplina y de
rigor siempre resulta aconsejable, si no necesario.
Fernando Berzal, Francisco J. Cortijo & Juan Carlos Cubero
Procesos y hebras
51
52
H1 no avanzar y H2, pese a tener menor prioridad que H1, s que lo har. El
planificador de CPU puede solucionar el problema eventualmente si todas las
hebras disponibles avanzan en su ejecucin, aunque sea ms lentamente cuando
tienen menor prioridad. De ah la importancia de darle una prioridad alta a las
hebras cuyo tiempo de respuesta haya de ser bajo, y una prioridad baja a las
hebras que realicen tareas muy largas.
Todos estos problemas ponen de manifiesto la dificultad que supone trabajar con aplicaciones
multihebra que comparten datos. Hay que asegurarse de asignar los recursos cuidadosamente
para minimizar las restricciones de acceso en exclusiva a los recursos compartidos. En
muchas ocasiones, por ejemplo, se puede simplificar notablemente la implementacin de las
aplicaciones multihebra si hacemos que cada una de las hebras trabaje de forma
independiente, incluso con copias distintas de los mismos datos si hace falta. Sin datos
comunes, los mecanismos de sincronizacin se hacen innecesarios.
Los mecanismos de sincronizacin anteriores utilizan memoria compartida y son tiles
puntualmente en sistemas centralizados. En un sistema distribuido, sin embargo, la
comunicacin y sincronizacin entre procesos habr de realizarse mediante paso de mensajes,
ya sea mediante denominacin directa (haciendo referencia al proceso con el que deseamos
comunicarnos o canal a travs del cual realizaremos la comunicacin) o a travs de nombres
globales (usando buzones, puertos o seales).
Procesos y hebras
53
Como siempre, al usar este tipo de mecanismos, las llamadas a WaitOne() pueden llevar un
parmetro opcional que indique un tiempo mximo de espera, pasado el cual la hebra reanuda
su ejecucin aunque no se haya cumplido la condicin por la que estaba esperando. De este
modo, podemos asegurarnos de que nuestras aplicaciones no se quedarn indefinidamente a la
espera de un suceso que puede no llegar a producirse.
Adems, la clase base WaitHandle incluye un mtodo llamado WaitAll() que se puede
utilizar para esperar simultneamente a todos los elementos de una matriz, ya sean stos
eventos o variables de exclusin mutua.
Como se puede apreciar, los eventos y las variables de exclusin mutua proporcionan la
misma funcionalidad que los monitores y cerrojos vistos con anterioridad. Su principal
ventaja reside en que se pueden utilizar para sincronizar hebras que estn ejecutndose en
distintos espacios de memoria, convirtindose as en un autntico mecanismo de
comunicacin entre procesos.
Aparte de las ya vistas, el espacio de nombres System.Threading incluye otras dos
clases que se pueden emplear para coordinar la ejecucin de tareas que se han de realizar
concurrentemente:
- Se pueden usar temporizadores gracias a la clase Timer, que proporciona un
mecanismo para ejecutar tareas peridicamente. El sistema operativo se encargar
de llamar al mtodo referenciado por un delegado de tipo TimerCallback cada
vez que expire el temporizador. Ntese que el mtodo llamado por el
temporizador se ejecutar en una hebra independiente a la hebra en la que se cre
http://csharp.ikor.org/
54
el temporizador, por las que deberemos ser cuidadosos si dicho mtodo accede a
datos compartidos con otras hebras.
- La clase Interlocked suministra un mtodo un mecanismo de exclusin
mutua a bajo nivel que protege a las aplicaciones multihebra de los errores que se
pueden producir cuando se efecta un cambio de contexto mientras se est
actualizando el valor de una variable. Sus mtodos estticos Increment() y
Decrement() sirven para incrementar o decrementar el valor de una variable
entera de forma atmica, ya que incluso algo aparentemente tan simple no se
ejecuta de forma atmica en el hardware del ordenador. Esta operacin se realiza,
en realidad, en tres pasos: cargar el dato en un registro del microprocesador,
modificar el valor y volver a almacenarlo en memoria. Si queremos que los tres
pasos se ejecuten como uno slo, en vez de los operadores ++ y --, tendremos
que usar los mtodos estticos Increment() y Decrement(). Esta clase
proporciona, adems, otros dos mtodos estticos que permiten establecer el valor
de una variables de forma atmica, con Exchange(), y reemplazar el valor de
una variable slo si sta es igual a un valor determinado con la funcin
CompareExchange(), lo que puede ser til para comprobar que la variable no
ha sido modificada desde que se accedi a ella por ltima vez.
Como estas pginas demuestran, la gama de mecanismos a nuestra disposicin para garantizar
el acceso en exclusiva a un recurso compartido por varias hebras es bastante amplia. Muchos
de estos mecanismos son equivalentes y slo difieren en su implementacin, por lo que su uso
resulta indistinto en la mayor parte de las situaciones con las que nos podamos encontrar.
Operaciones asncronas
Antes de pasar a estudiar distintos mecanismos de comunicacin entre procesos, cerraremos
este captulo retomando la ejecucin asncrona de delegados en un contexto de especial
inters: la realizacin de operaciones asncronas de entrada/salida.
Como ya hemos visto, el uso de hebras resulta especialmente interesante cuando hay que
realizar tareas lentas, como puede ser el acceso a dispositivos de E/S a travs de llamadas
sncronas a mtodos de las clases incluidas en el espacio de nombres System.IO. Al llamar
a un mtodo de forma sncrona, la ejecucin de nuestra hebra queda bloqueada hasta que el
mtodo finaliza su ejecucin, algo particularmente inoportuno cuando se realiza una costosa
operacin de E/S. Si pudisemos realizar esas llamadas de forma asncrona, nuestra aplicacin
no tiene que detenerse mientras se realiza la operacin de E/S. Como veremos a continuacin,
la plataforma .NET nos permite realizar operaciones asncronas de E/S de forma
completamente anloga a la ejecucin asncrona de delegados que vimos al hablar del uso de
hebras en C#.
En primer lugar, lo que haremos ser crear una estructura de datos auxiliar que usaremos para
ir almacenando los datos que leamos de forma asncrona. Esta estructura de datos auxiliar
Fernando Berzal, Francisco J. Cortijo & Juan Carlos Cubero
Procesos y hebras
55
http://csharp.ikor.org/
56
Procesos y hebras
57
http://csharp.ikor.org/
58
Referencias
La programacin con hebras no es un tema discutido con excesiva asiduidad, si bien pueden
encontrarse bastantes artculos en Internet acerca del uso de hebras y procesos. A
continuacin se mencionan algunos que pueden resultar de inters:
- Andrew D. Birrell: An Introduction to Programming with C# Threads. Microsoft
Corporation, 2003. (basado en un informe publicado por el mismo autor en 1989
cuando ste trabajaba para DEC [Digital Equipment Corporation], que ahora
forma parte de HP tras ser adquirida por Compaq)
- Shawn Cicoria: Proper Threading in Winforms .NET. The Code Project
(CodeProject.com), May 2003.
- Chris Sells: Safe, Simple Multithreading in Windows Forms, Part 1. MSDN, June
28, 2002.
- Chris Sells: Safe, Simple Multithreading in Windows Forms, Part 2. MSDN,
September 2, 2002.
- Chris Sells: Safe, Simple Multithreading in Windows Forms, Part 3. MSDN,
January 23, 2003.
- Ian Griffiths: Give Your .NET-based Application a Fast and Responsive UI with
Multiple Threads. MSDN Magazine, February 2003
- Matthias Steinbart: How to do Application Initialization while showing a
SplashScreen. The Code Project (CodeProject.com), January 2003.
- David Carmona: Programming the Thread Pool in the .NET Framework. MSDN,
June 2002.
- Brian Goetz: Concurrency made simple (sort of). IBM developerWorks,
November 2002.
Muchos de estos artculos suelen estar enfocados para la resolucin de problemas concretos
con los que puede encontrarse un programador. Su utilidad radica precisamente en eso. Dado
que estamos acostumbrados a pensar en programas secuenciales, normalmente no razonamos
igual de bien acerca de procesos y hebras concurrentes. En estas situaciones, una rpida
consulta en el Google puede ahorrarnos muchos quebraderos de cabeza. Un poco de ayuda
siempre viene bien, aunque siempre hemos de mantener nuestro punto de vista crtico: quin
nos asegura que lo que encontremos en Internet funcione correctamente?