AutoStart / Pre-warm功能在IIS 7.5 / WCF服务中不起作用怎么办?

内容来源于 Stack Overflow,并遵循CC BY-SA 3.0许可协议进行翻译与使用

  • 回答 (2)
  • 关注 (0)
  • 查看 (48)

为了从头开始测试IIS / WCF实现的许多令人头疼的问题,我构建了HelloWorld服务,并在这里遍历了(很好地)客户端。我为net.tcp添加了端点,并且该服务IIS 7.5在其自己ApplicationPool调用的硬件中在(在Windows 7上)下的两种绑定下端对端地正常工作。

我正在努力工作的是宣布的AutoStart和Preload(或“预热缓存”)功能。我已经非常仔细地按照这里这里所列出的指示(非常相似,但总是有很好的第二意见)。这意味着我

1)设置应用程序池startMode

<applicationPools> 
     <!-- ... -->
     <add name="HW" managedRuntimeVersion="v4.0" startMode="AlwaysRunning" /> 
</applicationPools>

2)启用serviceAutoStart并设置一个指向我的指针serviceAutoStartProvider

<site name="HW" id="2">
    <application path="/" applicationPool="HW" serviceAutoStartEnabled="true" serviceAutoStartProvider="PreWarmMyCache" />
    <!-- ... -->
</site>

3)并命名为所述提供者,其中GetType().AssemblyQualifiedName该类别的全部列在下面

<serviceAutoStartProviders> 
    <add name="PreWarmMyCache" type="MyWCFServices.Preloader, HelloWorldServer, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null" /> 
</serviceAutoStartProviders>
using System;

namespace MyWCFServices
{
    public class Preloader : System.Web.Hosting.IProcessHostPreloadClient
    {
        public void Preload(string[] parameters)
        {
            System.IO.StreamWriter sw = new System.IO.StreamWriter(@"C:\temp\PreloadTest.txt");
            sw.WriteLine("Preload executed {0:G}", DateTime.Now);
            sw.Close();
        }
    }
}

我试图重置该Preload方法中的相对App_Data文件夹路径(另一个SO答案建议),没关系。另外,我学习了w3wp.exe一个简单浏览到本地主机的过程。该流程消耗了巨大的17MB内存来提供其单个微小的OperationContract,而提供零预加载值的价格。17MB的ColdDeadCache。

提问于
用户回答回答于

这是一个稍微不同的方法:

  1. 使用Windows Server AppFabric进行服务自动启动
  2. 使用WCF基础结构来执行自定义启动代码

回复1:Appfabric AutoStart功能应该可以直接使用(假设没有使用MVC的ServiceRoute来注册您的服务,它们必须在Web.config的serviceActivations部分或使用物理*.svc文件中指定。

注2:要将自定义启动代码注入WCF管道,可以使用如下属性:

using System;
using System.ServiceModel;
using System.ServiceModel.Description;

namespace WCF.Extensions
{
    /// <summary>
    /// Allows to specify a static activation method to be called one the ServiceHost for this service has been opened.
    /// </summary>
    [AttributeUsage(AttributeTargets.Class, AllowMultiple = true, Inherited = false)]
    public class ServiceActivatorAttribute : Attribute, IServiceBehavior
    {
        /// <summary>
        /// Initializes a new instance of the ServiceActivatorAttribute class.
        /// </summary>
        public ServiceActivatorAttribute(Type activatorType, string methodToCall)
        {
            if (activatorType == null) throw new ArgumentNullException("activatorType");
            if (String.IsNullOrEmpty(methodToCall)) throw new ArgumentNullException("methodToCall");

            ActivatorType = activatorType;
            MethodToCall = methodToCall;
        }

        /// <summary>
        /// The class containing the activation method.
        /// </summary>
        public Type ActivatorType { get; private set; }

        /// <summary>
        /// The name of the activation method. Must be 'public static void' and with no parameters.
        /// </summary>
        public string MethodToCall { get; private set; }


        private System.Reflection.MethodInfo activationMethod;

        #region IServiceBehavior
        void IServiceBehavior.AddBindingParameters(ServiceDescription serviceDescription, ServiceHostBase serviceHostBase, System.Collections.ObjectModel.Collection<ServiceEndpoint> endpoints, System.ServiceModel.Channels.BindingParameterCollection bindingParameters)
        {
        }

        void IServiceBehavior.ApplyDispatchBehavior(ServiceDescription serviceDescription, ServiceHostBase serviceHostBase)
        {
            serviceHostBase.Opened += (sender, e) =>
                {
                    this.activationMethod.Invoke(null, null);
                };
        }

        void IServiceBehavior.Validate(ServiceDescription serviceDescription, ServiceHostBase serviceHostBase)
        {
            // Validation: can get method
            var method = ActivatorType.GetMethod(name: MethodToCall,
                             bindingAttr: System.Reflection.BindingFlags.Static | System.Reflection.BindingFlags.Public,
                             callConvention: System.Reflection.CallingConventions.Standard,
                             types: Type.EmptyTypes,
                             binder: null,
                             modifiers: null);
            if (method == null)
                throw new ServiceActivationException("The specified activation method does not exist or does not have a valid signature (must be public static).");

            this.activationMethod = method;
        }
        #endregion
    }
}

..可以这样使用:

public static class ServiceActivation
{
    public static void OnServiceActivated()
    {
        // Your startup code here
    }
}

[ServiceActivator(typeof(ServiceActivation), "OnServiceActivated")]
public class YourService : IYourServiceContract
{

}

这是我们已经使用了相当长一段时间和大量服务的确切方法。将WCF ServiceBehavior用于自定义启动代码(而不是依赖于IIS基础结构)的额外好处是它可以在任何托管环境(包括自托管)中运行,并且可以更容易地进行测试。

用户回答回答于

我知道这听起来很荒谬,但我面临同样的问题(在进行配置更改后,w3wp.exe未自动启动),这是因为我在编辑applicationHost.config文件时未在Admin模式下运行文本编辑器。对我来说愚蠢的错误。

我使用的是notepad++,它告诉我它实际上没有保存。

扫码关注云+社区