Es mala forma de utilizar this
en los estados de bloqueo, ya que es generalmente fuera de su control, que otra cosa podría estar bloqueando en ese objeto.
Para planificar correctamente las operaciones paralelas, se debe tener especial cuidado para considerar posibles situaciones de interbloqueo, y tener un número desconocido de puntos de entrada de bloqueo lo impide. Por ejemplo, cualquiera con una referencia al objeto puede bloquearlo sin que el diseñador/creador del objeto lo sepa. Esto aumenta la complejidad de las soluciones de subprocesos múltiples y puede afectar su corrección.
Un campo privado suele ser una mejor opción ya que el compilador impondrá restricciones de acceso a él, y encapsulará el mecanismo de bloqueo. El uso de this
viola la encapsulación al exponer al público parte de su implementación de bloqueo. Tampoco está claro que va a adquirir un bloqueo en this
a menos que haya sido documentado. Incluso entonces, depender de la documentación para evitar un problema no es óptimo.
Finalmente, existe la idea errónea de que lock(this)
realmente modifica el objeto pasado como parámetro, y de alguna manera lo hace de solo lectura o inaccesible. Esto es falso. El objeto pasado como parámetro a lock
simplemente sirve como una clave . Si ya se mantiene un bloqueo en esa tecla, no se puede hacer el bloqueo; de lo contrario, el bloqueo está permitido.
Es por eso que es malo utilizar cadenas como las claves en las sentencias lock
, ya que son inmutables y se comparten/se pueden acceder a través de partes de la aplicación. Debería usar una variable privada en su lugar, una instancia de Object
lo hará muy bien.
Ejecute el siguiente código C# como ejemplo.
public class Person
{
public int Age { get; set; }
public string Name { get; set; }
public void LockThis()
{
lock (this)
{
System.Threading.Thread.Sleep(10000);
}
}
}
class Program
{
static void Main(string[] args)
{
var nancy = new Person {Name = "Nancy Drew", Age = 15};
var a = new Thread(nancy.LockThis);
a.Start();
var b = new Thread(Timewarp);
b.Start(nancy);
Thread.Sleep(10);
var anotherNancy = new Person { Name = "Nancy Drew", Age = 50 };
var c = new Thread(NameChange);
c.Start(anotherNancy);
a.Join();
Console.ReadLine();
}
static void Timewarp(object subject)
{
var person = subject as Person;
if (person == null) throw new ArgumentNullException("subject");
// A lock does not make the object read-only.
lock (person.Name)
{
while (person.Age <= 23)
{
// There will be a lock on 'person' due to the LockThis method running in another thread
if (Monitor.TryEnter(person, 10) == false)
{
Console.WriteLine("'this' person is locked!");
}
else Monitor.Exit(person);
person.Age++;
if(person.Age == 18)
{
// Changing the 'person.Name' value doesn't change the lock...
person.Name = "Nancy Smith";
}
Console.WriteLine("{0} is {1} years old.", person.Name, person.Age);
}
}
}
static void NameChange(object subject)
{
var person = subject as Person;
if (person == null) throw new ArgumentNullException("subject");
// You should avoid locking on strings, since they are immutable.
if (Monitor.TryEnter(person.Name, 30) == false)
{
Console.WriteLine("Failed to obtain lock on 50 year old Nancy, because Timewarp(object) locked on string \"Nancy Drew\".");
}
else Monitor.Exit(person.Name);
if (Monitor.TryEnter("Nancy Drew", 30) == false)
{
Console.WriteLine("Failed to obtain lock using 'Nancy Drew' literal, locked by 'person.Name' since both are the same object thanks to inlining!");
}
else Monitor.Exit("Nancy Drew");
if (Monitor.TryEnter(person.Name, 10000))
{
string oldName = person.Name;
person.Name = "Nancy Callahan";
Console.WriteLine("Name changed from '{0}' to '{1}'.", oldName, person.Name);
}
else Monitor.Exit(person.Name);
}
}
consola de salida
'this' person is locked!
Nancy Drew is 16 years old.
'this' person is locked!
Nancy Drew is 17 years old.
Failed to obtain lock on 50 year old Nancy, because Timewarp(object) locked on string "Nancy Drew".
'this' person is locked!
Nancy Smith is 18 years old.
'this' person is locked!
Nancy Smith is 19 years old.
'this' person is locked!
Nancy Smith is 20 years old.
Failed to obtain lock using 'Nancy Drew' literal, locked by 'person.Name' since both are the same object thanks to inlining!
'this' person is locked!
Nancy Smith is 21 years old.
'this' person is locked!
Nancy Smith is 22 years old.
'this' person is locked!
Nancy Smith is 23 years old.
'this' person is locked!
Nancy Smith is 24 years old.
Name changed from 'Nancy Drew' to 'Nancy Callahan'.
que es un código de lotta! –
Imo código demasiado para ilustrar realmente el problema. – Tigraine
Imo el ejemplo es una buena longitud. Es fácil de leer. – Parappa