189

在 ASP.NET 网站上,每个 Web 请求的静态类是唯一的,还是在需要时实例化并在 GC 决定处理它们时进行 GC?

我问的原因是因为我之前在 C# 中编写了一些静态类,并且行为与我预期的不同。我本来希望静态类对每个请求都是唯一的,但情况似乎并非如此。

如果它们不是每个请求唯一的,有没有办法让它们成为唯一的?

更新:
driis 给我的答案正是我所需要的。我已经在使用单例类,但是它使用的是静态实例,因此即使用户不同,它也会在请求之间共享,这在这种情况下是一件坏事。使用HttpContext.Current.Items完美地解决了我的问题。对于将来偶然发现这个问题的任何人,这是我的实现,经过简化和缩短,以便于理解模式:

using System.Collections;
using System.Web;

public class GloballyAccessibleClass
{
    private GloballyAccessibleClass() { }

    public static GloballyAccessibleClass Instance
    {
        get
        {
            IDictionary items = HttpContext.Current.Items;
            if(!items.Contains("TheInstance"))
            {
                items["TheInstance"] = new GloballyAccessibleClass();
            }
            return items["TheInstance"] as GloballyAccessibleClass;
        }
    }
}
4

5 回答 5

151

您的静态类和静态实例字段在对应用程序的所有请求之间共享,并且与应用程序域具有相同的生命周期。因此,在使用静态实例时应该小心,因为您可能会遇到同步问题等。还要记住,在应用程序池被回收之前,不会对静态实例进行 GC,因此静态实例引用的所有内容都不会被 GC。这可能会导致内存使用问题。

如果您需要一个与请求具有相同生命周期的实例,我建议使用该HttpContext.Current.Items集合。这是设计用来存储您在请求中需要的东西的地方。为了更好的设计和可读性,您可以使用单例模式来帮助您管理这些项目。只需创建一个 Singleton 类,将其实例存储在HttpContext.Current.Items. (在我的 ASP.NET 通用库中,为此我有一个通用的 SingletonRequest 类)。

于 2008-10-12T09:51:34.513 回答
31

静态成员只有当前工作进程的范围,所以它与请求无关,因为不同的请求可能由同一个工作进程处理,也可能不处理。

  • 为了与特定用户和跨请求共享数据,请使用 HttpContext.Current.Session。
  • 为了在特定请求中共享数据,请使用 HttpContext.Current.Items。
  • 为了在整个应用程序之间共享数据,要么为此编写一种机制,要么将 IIS 配置为使用单个进程并编写一个单例/使用应用程序。

顺便说一句,工作进程的默认数量是 1,所以这就是为什么网络上到处都是认为静态成员具有整个应用程序范围的人的原因。

于 2013-08-20T12:09:08.743 回答
13

由于这些类型包含在应用程序域中,我希望只要应用程序域没有被回收,或者如果请求由不同的应用程序域提供服务,就会存在静态类。

我可以想到几种方法来使特定于特定请求的对象取决于您想要做什么,例如,您可以在 Application.BeginRequest 中实例化该对象,然后将其存储在 HttpRequest 对象中,以便它可以被所有对象访问请求处理管道。

于 2008-10-12T02:30:05.887 回答
4

如果它们不是每个请求唯一的,有没有办法让它们成为唯一的?

没有。静态成员归 ASP.NET 进程所有,并由Web 应用程序的所有用户共享。您需要求助于其他会话管理技术,例如会话变量。

于 2008-10-12T02:57:37.173 回答
2

通常静态方法、属性和类在该Application级别是通用的。只要应用程序存在,它们就会被共享。

您可以使用该ThreadStatic属性指定不同的行为。在这种情况下,它们将特定于当前线程,我认为这是特定于每个请求的。
不过,我不建议这样做,因为它似乎过于复杂。

您可以使用HttpContext.Current.Items为一个请求设置内容,或HttpContext.Current.Session为一个用户设置内容(跨请求)。

一般来说,除非你必须使用类似Server.Transfer的东西,否则最好的方法基本上是创建一次,然后通过方法调用显式传递它们。

于 2008-10-12T10:02:21.370 回答