Skip to content

Castle Windsor Array Resolution Gotcha

The shiny new system which I’ve recently been developing makes heavy use of the Chain of Responsibility pattern, and as such a number of service classes take an array of objects in the constructor:

/// <summary>
/// Initializes a new instance of the LeadAllocationService class.
/// </summary>
/// <param name="leadAllocators">Array of Lead Allocators.</param>
public LeadAllocationService(params ILeadAllocator[] leadAllocators)
{
this.LeadAllocators = leadAllocators;
}
view raw CWARG1.cs hosted with ❤ by GitHub

I’m using Castle Windsor for dependency management, so I’ve been fluently registering all instances of ILeadAllocator:

container.Register(
AllTypes
.Of<ILeadAllocator>()
.FromAssemblyNamed("Marshalls.Leads.ApplicationService")
.WithService
.FromInterface(typeof(ILeadAllocator))
.Configure(c => c.LifeStyle.Transient));
view raw CWARG2.cs hosted with ❤ by GitHub

Easy, right? And yet at runtime Windsor surprised me by throwing this exception in my face:

Castle.MicroKernel.Handlers.HandlerException: Can’t create component ‘Marshalls.Leads.ApplicationService.LeadAllocationService’ as it has dependencies to be satisfied.

Marshalls.Leads.ApplicationService.LeadAllocationService is waiting for the following dependencies:

Keys (components with specific keys)

– leadAllocators which was not registered.

Huh?! What gives? Well, a little Googling revealed this post from Castle founder Hamilton Verissimo explaining that by default the Castle MicroKernel expects me to define what should be included in the array. But he goes on to explain that the behaviour I desire can be achieved by registering a custom subresolver with the microkernel. That subresolver has since been included in the Castle Windsor distro, so in actual fact all I needed to do was add the following line of code when configuring my container:

container.Kernel.Resolver.AddSubResolver(
new ArrayResolver(container.Kernel));
view raw CWARG3.cs hosted with ❤ by GitHub

So now, when new implementations of ILeadAllocator are added to the codebase, they are immediately available to the CofR pattern within the LeadAllocationService, with no additional work required by the developer. Hurrah!

Published inTech
Copyright © Ian Fraser Nelson 2025