• 关闭FCNs(文件修改监控)


    也许你知道,修改站点的某些特定文件和目录会导致整个站点重启或者重新编译。也许你不注意,你不会知道删除或重命名站点下的任意目录,会导致整个站 点重启(添加目录并不会)。这个问题其实是很多Session丢失的根源,比如《关于 ASP.NET 2.0的目录结构变化导致Session丢失的问题》,同样你还可以找到很多这样的例子。

    这个其实是 ASP.NET 2.0的一个“精心”设计,因为在很多情况下。ASP.NET会缓存很多资源,而如果没有监控目录的变动的话,那么有些已删除的资源就有可能仍然被使用 着,造成其它不必要的资源泄露。通过这篇文章我们可以找到可能导致站点重启的一些条件,其中的第一条就是删除或重命名目录会导致站点重启。

    这 个问题早在ASP.NET 2.0刚发布的时候就被发现,并且已经有了专门的讨论《Deleting ASP.NET 2.0 Application Sub-Directories Shuts Down the AppDomain》,在文中也提了一种解决方案,但是做法太过复杂。最理想的方案是,有相关的补丁来解决这个问题,幸运的 是,我们可以通过这里找到这个补丁,并且介绍如何配置来解决这个问题。

    解决方案是:监控目录删除或重命名的机制 叫:File Change Notifications (FCNs),这个补丁就是用来设置FCNS禁用或启用的开关,我们只要把FNCS禁用就可以避免这个问题。

    做法:现在的ASP.NET 2.0已经到了SP2,我们不需要单独去安装这个补丁。我们只需要在注册表在做一步操作即可解决这个问题。在HKLM\Software \Microsoft\ASP.NET 添加一个名为:FCNMode 的DWORD值,默认它应该是不存在的,它的值含义为:

    Does not exist
    This is the default behavior. For each subdirectory, the application will create an object that will monitor the subdirectory.

    0 or greater than 2
    This is the default behavior. For each subdirectory, the application will create an object that will monitor the subdirectory.

    1
    The application will disable File Change Notifications (FCNs).

    2
    The application will create one object to monitor the main directory. The application will use this object to monitor each subdirectory.

    我们将它的值设为1即可禁用FCNS。更为详细的可以参 考:http://support.microsoft.com/kb/911272

    退出注册表,重启IIS,问题即可解决。

    说明一下:Session丢失 只是AppDomain重启的一个副作用,并不是本文的重点,只是为了方便别人查找相关问题时的一个标题优化。另外导致站点重启的原因有很多种,可以参考 上文给出的链接,这里不再遨述。

    再次说明一下:如果你禁用 了FCNs,那么不管你是修改了Web.config,还是BIN目录,都不会引起整个站点的重启。我想这个也会相应的给我们带来一些新的问题,比如缓存 过期的问题。至于是否禁用,大家自己选择吧。:(

    补充,通过代码可 以去除目录的改动监控,同时不影响bin目录和web.config改动时,程序需要的重启。

    public class StopFileMonitorModule : IHttpModule
    {
    #region IHttpModule Members

    public void Dispose()
    {

    }

    public void Init(HttpApplication context)
    {
    PropertyInfo p = typeof(System.Web.HttpRuntime).GetProperty("FileChangesMonitor", BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.Static);
    object o = p.GetValue(null, null);
    FieldInfo f = o.GetType().GetField("_dirMonSubdirs", BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.IgnoreCase);
    object monitor = f.GetValue(o);
    MethodInfo m = monitor.GetType().GetMethod("StopMonitoring", BindingFlags.Instance | BindingFlags.NonPublic);
    m.Invoke(monitor, new object[] { });

    }

    #endregion
    }
    作者:KKcat
        
    个人博客:http://jinzhao.me/
        
    本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
  • 相关阅读:
    pdflatex, xelatex, texstudio中文编码问题
    secure erase 时必须umount
    浪潮不能进bios解决过程
    ycsb-命令及参数-与生成的负载类型相关
    zt-Simple source policy routing
    oracle 分页查找数据
    前台调用后台的过程
    初识 java script
    java中的BigDecimal和String的相互转换
    一级缓存、二级缓存、延迟加载、hibernate session 域 pojo三种状态
  • 原文地址:https://www.cnblogs.com/jinzhao/p/1684402.html
Copyright © 2020-2023  润新知