Usually, when developing small application there is no real justification for using a full-fledged IoC container for managing services.
However, cross-cutting , single instance services are something common that still has to managed. This leaves us with the following options for accessing services:
- Implement the service as singleton.
- Move the service reference between objects.
- Use a service locator.
I don't like the first option as it introduces global variables to the application and the second option is tedious very hard to manage.
This leaves us with the service locator which in general should be the only singleton(static) in the application and provide access to the services according to their interface.
The service locator usually looks like that:
public static class ServiceLocator
{
private static readonly
IDictionary<Type, object> services =
new Dictionary<Type, object>();
private static readonly object locker =
new object();
public static void SetService<T>(T instance)
where T : class
{
if (instance == null)
throw new
ArgumentNullException("instance");
lock (locker)
{
services.Add(typeof(T), instance);
}
}
public static T GetService<T>() where T : class
{
object service;
lock (locker)
{
services.
TryGetValue(typeof(T), out service);
}
return service as T;
}
}
It hold a dictionary of services. The key is the type and the value is the instance. It is quite simple, however I do not like the fact that it has to lock the services dictionary on each access.
When trying to eliminate the need for locking, I remembered this post by Arnon and it inspired my to try this approach:
public static class ServiceLocator
{
private class Resolver<T> where T:class
{
private static T theInstance;
public static void Create(T instance)
{
if (instance == null)
throw new
ArgumentNullException("instance");
Interlocked.
Exchange(ref theInstance, instance);
}
public static T Instance
{
get
{
return theInstance;
}
}
}
public static void SetService<T>(T instance)
where T:class
{
Resolver<T>.Create(instance);
}
public static T GetService<T>() where T:class
{
return Resolver<T>.Instance;
}
}
Pay attention to the Resolver private class. For every type T that is specified as a generic parameter, a new type, Resolver<SomeType> will be created and the static field, theInstance, will be set to the service instance.
What did we achieve here?
In the previous implementation we had to lock the dictionary on every read or write. In the new implementation there is no need to lock for read and the lock for writing was changed to Interlocked which has better performance so we have better concurrency.
No dictionary. Although it is negligible there is no need to search a dictionary. Besides, I always felt that something like service resolution should be something which is related to the infrastructure and should not be managed by a dictionary and I think this implementation better achieves it.
No comments:
Post a Comment