61

我有一个asp.net 网站在我们的Intranet 上运行了几个星期。我刚刚从我的 application_error emailer 方法收到一封电子邮件,其中包含未处理的异常。

在这里(我已经清理了一些路径以使其更好地显示)

异常:对象引用未设置为对象的实例。堆栈跟踪:在 System.Collections.Generic.Dictionary`2.Insert(TKey key, TValue value, Boolean add) at System.Collections.Generic.Dictionary`2.Add(TKey key, TValue value) 在 TimesheetDomain.DataMappers.StaffMemberData TimesheetDomain\DataMappers\StaffMemberData.cs:line 362 中的 .ReadStaff(SqlDataReader 阅读器)

在 TimesheetDomain.DataMappers.StaffMemberData.GetStaffMember(String name) 在 TimesheetDomain\DataMappers\StaffMemberData.cs:line 401

在 TimesheetDomain\ServiceLayer\TimesheetManager.cs:line 199 中的 TimesheetDomain.ServiceLayer.TimesheetManager.GetUserFromName(String name)

在 \App_Code\UserVerification.cs: 第 29 行的 UserVerification.GetCurrentUser() 在 \WebTimesheets\WebTimesheets.master.cs: 第 159 行的 WebTimesheets.OnInit(EventArgs e)

在 System.Web.UI.Control.InitRecursive(控制命名容器) 在 System.Web.UI.Control.InitRecursive(控制命名容器) 在 System.Web.UI.Page.ProcessRequestMain(布尔 includeStagesBeforeAsyncPoint,布尔 includeStagesAfterAsyncPoint)

基本上,我的 ReadStaff 方法似乎出错了,该方法读取数据读取器以构建员工对象。这是一段代码:

while (reader != null && reader.Read())
{
    StaffMember newMember = null;
    string firstName = reader["FirstName"].ToString();
    string lastName = reader["LastName"].ToString();
    int staffID = (int)reader["StaffID"];
    int employSection = (int)reader["EmploySection"];
    StaffType employType = (StaffType)employSection;
    string emailAddress = reader["EmailInt"].ToString();
    int employCode = (int)reader["ibbwid"];

    //check if they are an admin staff member 
    if (IsAdminStaff(employType))
    {
        newMember = new AdminOfficer(firstName, lastName, employType, staffID, emailAddress, employCode);
    }
    else
    {
        //check if they are a supervisor
        if (IsASupervisor(staffID))
            newMember = new Supervisor(firstName, lastName, employType, staffID, emailAddress, employCode);
        else
            newMember = new StaffMember(firstName, lastName, employType, staffID, emailAddress, employCode);
    }

    //add to identity map
    if (!_staffMembers.ContainsKey(staffID))
        _staffMembers.Add(staffID, newMember); //****THIS IS LINE 362*****
    else
        _staffMembers[staffID] = newMember;
}

(第 362 行是倒数第 3 行)我正在使用身份图(只是阅读 fowlers 关于模式的书,并认为这是一个好主意 - 可能做错了,很高兴评论)但这并不太相关,因为我稍后使用其他地方的newMember对象,所以如果我删除那个块,NullReferenceException就会发生。

我正在努力看看newMember那里的最后第三行(这是错误的行)到底是如何为空的。

Resharper/VS 没有给我一个可能的警告null——因为我可以选择 3 个构造函数。

谁能建议我在哪里可以尝试修复此错误?它只发生过一次,自网站上线以来,该方法已被调用了数千次。

谢谢

[编辑] 根据要求,这是工作人员的 IComparer

/// <summary>
/// Comparer for staff members - compares on name
/// </summary>
public class StaffMemberComparer : IComparer
{
    public int Compare(object x, object y)
    {
        //check they are staff members
        if (x is StaffMember && y is StaffMember)
        {
            //do a simple string comparison on names
            StaffMember staffX = x as StaffMember;
            StaffMember staffY = y as StaffMember;

            return String.Compare(staffX.FirstName, staffY.FirstName);
        }

        throw new Exception("This is for comparing Staff Members");
    }
}

它用于 IComparable 实现

/// <summary>
/// IComparable implementaiton
/// </summary>
/// <param name="obj">object to compare to</param>
/// <returns></returns>
public int CompareTo(object obj)
{
    StaffMemberComparer comparer = new StaffMemberComparer();
    return comparer.Compare(this, obj);
}
4

6 回答 6

139

这几乎可以肯定是一个线程问题 - 请参阅这个问题及其接受的答案

Dictionary<>.Insert()NullReferenceException如果在插入操作期间从另一个线程修改了字典实例,则会在内部抛出一个。

于 2010-02-03T20:41:40.110 回答
22

从 .NET 4.0 开始,您可以使用 ConcurrentDictionary 并避免与同时从多个线程操作同一字典相关的线程问题。

于 2013-01-17T16:44:48.020 回答
1

它只发生过一次,自网站上线以来,该方法已被调用了数千次。

读完这篇文章后,我可以得出结论,.NET 可能已经耗尽了它的内存并且它无法创建更多的 Dictionary 键,这可能不是你的错。但是,是的,当我们尝试在会话/应用程序变量中存储太多信息时确实遇到了这类错误,从而增加了 Web 应用程序的内存占用。但是当我们的数字非常高时,我们会遇到这样的错误,比如在 Dictionary 或 List 中存储 10,000 个项目等。

模式很好,但你也必须意识到,我们使用数据库来存储关系格式的信息,如果我们开始使用内存来存储类似的东西,那么我们就忽略了强大的数据库。数据库也可以为您缓存值。

这可能听起来很傻,但我们让我们的 Windows 服务器每 24 小时重新启动一次,在没有流量的午夜。这确实帮助我们摆脱了这些错误。我们会定期按照修复计划重新启动服务器,以清除所有缓存/日志。

于 2009-08-24T09:46:21.007 回答
1

我看不到任何明显的东西。我会运行一些 SQL 来检查数据库中是否存在任何错误数据。问题可能是相关输入表单中的异常错误。如果到目前为止代码已经运行了数千次而没有发生任何事故,我会在有问题的代码块周围包装一些额外的异常处理/报告,这样你至少可以在下次发生时获得一个 staffId。

你可能会在这样的事情上浪费很多时间。最方便的方法可能只是让它在上述/受控条件下再次失败......假设它造成的破坏程度是可接受的/可管理的/轻微的。

我很欣赏这不能满足即时的了解需求,但它可能是解决问题的最佳方法,尤其是在故障率如此低的情况下。

于 2009-08-24T05:26:38.050 回答
0

正如其他人所说,比较可能会导致问题。
是否有任何比较标准包含空值?具体来说,字符串属性?

即,如果 firstname 或 lastname 或 emailId 为 null 并且如果在比较中使用它,则在字典中用于比较时可能会失败。

编辑:Supervisor 和 StaffMember 和 AdminStaff 类有什么关系?
在代码中,您将两个实例都转换为 StaffMember。我的猜测是,如果 Supervisor 和 StaffMember 类相关,这可能是个问题。

EDIT2:字典实例的范围是什么?它是否在应用程序级别/会话级别共享?是否有可能多个线程可以尝试从中读取/写入?

于 2009-08-24T03:45:49.160 回答
0

我看到这个与线程无关的异常的另一种方式是在序列化字典时。在这种情况下,它是空的,但我仍然在反序列化实例的 Insert() 方法中得到 NullReferenceException。

在我的案例中,简单的更改只是在反序列化后创建一个新实例。我不确定序列化是否只是破坏了字典,或者它是否是为字典定义的类型。

在我的情况下,如果没有序列化代理(我已经提供了这些),这些类型是不可序列化的,但也许字典中的某些东西在这里有问题。尽管如此,字典还是空的,这仍然发生。

于 2012-08-17T19:09:17.473 回答