Как передать значения конструктору в моей службе wcf?
Я хотел бы передать значения в конструкторе класса, который реализует службу.
однако ServiceHost только позволяет мне передать имя типа для создания, а не какие аргументы передать его контрастору.
Я хотел бы иметь возможность пройти на заводе, который создает мой объект обслуживания.
то, что я нашел до сих пор:
- поведение инъекции зависимостей WCF что больше, чем то, что я ищу для моих нужд и кажется слишком сложным.
8 ответов
вам нужно будет реализовать комбинацию custom ServiceHostFactory
, ServiceHost
и IInstanceProvider
.
данный сервис с этой подписью конструктора:
public MyService(IDependency dep)
вот пример, который может запустить MyService:
public class MyServiceHostFactory : ServiceHostFactory
{
private readonly IDependency dep;
public MyServiceHostFactory()
{
this.dep = new MyClass();
}
protected override ServiceHost CreateServiceHost(Type serviceType,
Uri[] baseAddresses)
{
return new MyServiceHost(this.dep, serviceType, baseAddresses);
}
}
public class MyServiceHost : ServiceHost
{
public MyServiceHost(IDependency dep, Type serviceType, params Uri[] baseAddresses)
: base(serviceType, baseAddresses)
{
if (dep == null)
{
throw new ArgumentNullException("dep");
}
foreach (var cd in this.ImplementedContracts.Values)
{
cd.Behaviors.Add(new MyInstanceProvider(dep));
}
}
}
public class MyInstanceProvider : IInstanceProvider, IContractBehavior
{
private readonly IDependency dep;
public MyInstanceProvider(IDependency dep)
{
if (dep == null)
{
throw new ArgumentNullException("dep");
}
this.dep = dep;
}
#region IInstanceProvider Members
public object GetInstance(InstanceContext instanceContext, Message message)
{
return this.GetInstance(instanceContext);
}
public object GetInstance(InstanceContext instanceContext)
{
return new MyService(this.dep);
}
public void ReleaseInstance(InstanceContext instanceContext, object instance)
{
var disposable = instance as IDisposable;
if (disposable != null)
{
disposable.Dispose();
}
}
#endregion
#region IContractBehavior Members
public void AddBindingParameters(ContractDescription contractDescription, ServiceEndpoint endpoint, BindingParameterCollection bindingParameters)
{
}
public void ApplyClientBehavior(ContractDescription contractDescription, ServiceEndpoint endpoint, ClientRuntime clientRuntime)
{
}
public void ApplyDispatchBehavior(ContractDescription contractDescription, ServiceEndpoint endpoint, DispatchRuntime dispatchRuntime)
{
dispatchRuntime.InstanceProvider = this;
}
public void Validate(ContractDescription contractDescription, ServiceEndpoint endpoint)
{
}
#endregion
}
зарегистрируйте MyServiceHostFactory в своем MyService.svc файл, или использовать MyServiceHost непосредственно в коде для сценариев самостоятельного размещения.
вы можете легко обобщить этот подход, и на самом деле некоторые контейнеры DI уже сделали это для вы (cue: Windsor'S WCF Facility).
вы можете просто создать и экземпляр Service
и передайте этот экземпляр в ServiceHost
"объект". Единственное, что вам нужно сделать, это добавить [ServiceBehaviour]
атрибут для вашей службы и пометить все возвращаемые объекты с
ответ Марка с IInstanceProvider
является правильным.
вместо использования пользовательского ServiceHostFactory вы также можете использовать пользовательский атрибут (скажем MyInstanceProviderBehaviorAttribute
). Выведите его из Attribute
, сделать ее реализации IServiceBehavior
и реализовать IServiceBehavior.ApplyDispatchBehavior
методом
// YourInstanceProvider implements IInstanceProvider
var instanceProvider = new YourInstanceProvider(<yourargs>);
foreach (ChannelDispatcher dispatcher in serviceHostBase.ChannelDispatchers)
{
foreach (var epDispatcher in dispatcher.Endpoints)
{
// this registers your custom IInstanceProvider
epDispatcher.DispatchRuntime.InstanceProvider = instanceProvider;
}
}
затем примените атрибут к классу реализации службы
[ServiceBehavior]
[MyInstanceProviderBehavior(<params as you want>)]
public class MyService : IMyContract
третий вариант: вы также можете применить поведение службы с помощью файла конфигурации.
Я работал с ответом Марка, но (по крайней мере, для моего сценария) это было бесполезно сложно. Один из ServiceHost
конструкторы принимают экземпляр службы, который можно передать непосредственно из ServiceHostFactory
реализация.
чтобы отбросить пример Марка, это выглядело бы так:
public class MyServiceHostFactory : ServiceHostFactory
{
private readonly IDependency _dep;
public MyServiceHostFactory()
{
_dep = new MyClass();
}
protected override ServiceHost CreateServiceHost(Type serviceType,
Uri[] baseAddresses)
{
var instance = new MyService(_dep);
return new MyServiceHost(instance, serviceType, baseAddresses);
}
}
public class MyServiceHost : ServiceHost
{
public MyServiceHost(MyService instance, Type serviceType, params Uri[] baseAddresses)
: base(instance, baseAddresses)
{
}
}
винт ... я смешал инъекции зависимостей и шаблоны локатора служб (но в основном это все еще инъекция зависимостей, и это даже происходит в конструкторе, что означает, что вы можете иметь только для чтения состояние).
public class MyService : IMyService
{
private readonly Dependencies _dependencies;
// set this before creating service host. this can use your IOC container or whatever.
// if you don't like the mutability shown here (IoC containers are usually immutable after being configured)
// you can use some sort of write-once object
// or more advanced approach like authenticated access
public static Func<Dependencies> GetDependencies { get; set; }
public class Dependencies
{
// whatever your service needs here.
public Thing1 Thing1 {get;}
public Thing2 Thing2 {get;}
public Dependencies(Thing1 thing1, Thing2 thing2)
{
Thing1 = thing1;
Thing2 = thing2;
}
}
public MyService ()
{
_dependencies = GetDependencies(); // this will blow up at run time in the exact same way your IoC container will if it hasn't been properly configured up front. NO DIFFERENCE
}
}
зависимости сервиса четко указаны в договоре его вложенности Dependencies
класса. Если вы используете контейнер IoC (тот, который еще не исправил беспорядок WCF для вас), вы можете настроить его для создания Dependencies
экземпляр вместо услуга. Таким образом, вы получаете теплое нечеткое чувство, которое ваш контейнер дает вам, а также не нужно прыгать через слишком много обручей, наложенных WCF.
Я не собираюсь терять сон из-за такого подхода. Как и все остальные. В конце концов, вы контейнер IoC-это большая, жирная, статическая коллекция делегатов, которая создает материал для вас. Что еще добавить?
Это было очень полезное решение, особенно для тех, кто новичок в WCF верстальщик. Я хотел бы опубликовать небольшой совет для всех пользователей,которые могут использовать это для службы IIS. MyServiceHost должен наследовать WebServiceHost, не только ServiceHost.
public class MyServiceHost : WebServiceHost
{
public MyServiceHost(MyService instance, Type serviceType, params Uri[] baseAddresses)
: base(instance, baseAddresses)
{
}
}
это создаст все необходимые привязки и т. д. Для ваших конечных точек в IIS.
Я использую статические переменные своего типа. Не уверен, что это лучший способ, но он работает для меня:
public class MyServer
{
public static string CustomerDisplayName;
...
}
когда я создаю экземпляр хоста службы, я делаю следующее:
protected override void OnStart(string[] args)
{
MyServer.CustomerDisplayName = "Test customer";
...
selfHost = new ServiceHost(typeof(MyServer), baseAddress);
....
}
мы столкнулись с этой же проблемой и решил ее следующим образом. Это простое решение.
в Visual Studio просто создайте обычное приложение-службу WCF и удалите его интерфейс. Оставь .cs-файл на месте (просто переименуйте его) и откройте этот cs-файл и замените имя интерфейса своим исходным именем класса, который реализует логику службы (таким образом, класс службы использует наследование и заменяет вашу фактическую реализацию). Добавить по умолчанию конструктор, который вызывает конструкторы базового класса, например:
public class Service1 : MyLogicNamespace.MyService
{
public Service1() : base(new MyDependency1(), new MyDependency2()) {}
}
базовый класс MyService-это фактическая реализация службы. Этот базовый класс не должен иметь конструктор без параметров, а только конструкторы с параметрами, которые принимают зависимости.
служба должна использовать этот класс вместо исходного MyService.
Это простое решение и работает как шарм : - D