Comparar e intercambiar, también conocido por las siglas CAS (del inglés Compare And Swap), es una operación atómica usada en el diseño de algoritmos concurrentes para permitir la sincronización de hilos de ejecución sin recurrir a bloqueos de los mismos.[1]
Básicamente consiste en comparar el valor de una variable con un valor esperado y, si el valor es el esperado, entonces intercambia el valor de la variable con el de otra variable. Todo esto realizado de una forma atómica. La atomicidad garantiza que el nuevo valor es calculado basado en información actualizada; si entretanto el valor ha sido actualizado por otro hilo de ejecución, la escritura fallará, El resultado de la operación tiene que indicar si se realizó la sustitución; esto puede ser realizado con una simple respuesta booleana (a esta variante se le llama 'comparar y establecer), o retornando el valor leído de la memoria (no el escrito).[2]
Más formalmente podemos definir el algoritmo de la siguiente forma:[2]
En resumen, cuando varios hilos de ejecución intentan actualizar la misma variable simultáneamente usando CAS, uno gana y actualiza el valor de la variable y el resto pierde. Pero los perdedores no son sancionados con la suspensión del hilo, ellos son libres de volver a intentar realizar la operación o simplemente no hacer nada.[2]
La técnica CAS es optimista ya que procede con la actualización a la espera de tener éxito, sin embargo puede haber fallo, que será detectado, si otro hilo ha actualizado la variable desde que él la examinó por última vez.[2]
Supongamos que V es una localización de memoria donde está almacenado el valor 10. Hay múltiples hilos que quieren incrementar este valor y usar el valor incrementado para otras operaciones. Supongamos los siguientes pasos:[2]
La técnica CAS está soportada por las CPU's modernas. El acceso a estas rutinas de CPU ponen a disposición del sistema una implementación eficiente al más bajo nivel del algoritmo. Esto puede ser aprovechado por lenguajes de alto nivel. Por ejemplo Java desde versión 5 tiene acceso a estas funciones vía clases del paquete java.util.concurrent.atomic.[3]