使用Entity框架检索数据时出现异常(代码优先)

时间:2015-05-19 01:56:40

标签: asp.net-mvc-4 c#-4.0 ef-code-first entity-framework-6 poco

我不确定协议是什么w.r.t.改变你的问题,但我想如果我改变它会更好,因为有很多信息,并且提出问题的原始意图已经丢失。

原始问题:

我在使用EF(代码优先方法/ POCO类)生成数据库时最初遇到了问题。 其中一个问题是我在实体类中使用了构造函数来初始化成员。在@RickStahl建议不需要之后,我改变了我的实现。

为了读者的利益,我不希望丢失这些信息,因为一些旧的意见打算解决这个问题。

但是,在最初创建此线程的过程中,情况发生了变化。我已经能够克服一些问题。

当前问题:

我在从底层数据库表中检索内容时遇到问题。

例外: 对象引用未设置为对象的实例。

enter image description here

异常发生在Program.cs的以下语句中(第21行):

foreach(PhoneNumber p in cd.Phones)

为了便于理解这个问题,我正在粘贴整个源代码。

这是我的Entity类的源代码。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;

namespace ConsoleApplication1
{

public class ContactData
{

    [Key]
    public int ContactId { get; set; }

    public string FirstName { get; set; }
    public string LastName { get; set; }

    public IList<PhoneNumber> Phones { get; set; }
    public IList<EmailAddress> Emails { get; set; }

}

public class PhoneNumber
{
    [Key]
    public int PhoneId { get; set; }
    public int ContactId { get; set; }
    public string Phone { get; set; }

    public ContactData ContactData { get; set; }

    public PhoneNumber()
    {

    }
}

public class EmailAddress
{
    [Key]
    public int EmailId { get; set; }
    public int ContactId { get; set; }
    public string Email { get; set; }

    public ContactData ContactData { get; set; }

    public EmailAddress()
    {

    }
}
}

ContactContext.cs

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Data.Entity;

namespace ConsoleApplication1
{
public class ContactContext : DbContext
{
    public ContactContext()
        : base("ContactDBContext")
    {
        Database.SetInitializer<ContactContext>(new DropCreateDatabaseIfModelChanges<ContactContext>());
    }

    public DbSet<ContactData> Contacts { get; set; }

}
}

ContactManager.cs

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace ConsoleApplication1
{
class ContactManager
{
    ContactContext cDbContext = new ContactContext();

    public IList<ContactData> GetContactList()
    {

        IQueryable<ContactData> cContactList = cDbContext.Contacts;

        IList<ContactData> cListData = new List<ContactData>();

        cListData = cContactList.ToList();

        return cListData;
    }
}
}

Program.cs(入口点)

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace ConsoleApplication1
{
class Program
{
    static void Main()
    {
        ContactManager cMgr = new ContactManager();

        foreach (ContactData cd in cMgr.GetContactList())
        {
            Console.WriteLine(cd.ContactId);
            Console.WriteLine(cd.FirstName);
            Console.WriteLine(cd.LastName);

            foreach(PhoneNumber p in cd.Phones)
            {
                Console.WriteLine(p.Phone);
            }

            foreach (EmailAddress e in cd.Emails)
            {
                Console.WriteLine(e.Email);
            }
        }
        Console.ReadKey();

    }

1 个答案:

答案 0 :(得分:0)

我终于设法找到了解决我自己问题的方法。但我想向@SteveGreene和@RickStahl致谢,感谢他们的宝贵意见。

@RickStahl给出了一个很好的输入或建议,即不在我的Entity类中使用参数化构造函数。 Per Rick,参数化构造函数不适用于EF。

@SteveGreene - 我终于意识到你指的是我正确的方向。由于我对EF缺乏了解,我当时无法得到它。然而,在详细阅读了EF之后,特别是有关Eager加载和延迟加载的信息,最终帮助了我。

在我发现我必须在ContactData类构造函数中实例化“电话”和“电子邮件”的集合属性后,与“对象引用未设置为对象实例”相关的错误很容易解决。 / p>

请参阅下面的代码更改。

public class ContactData
{

    [Key]
    public int ContactId { get; set; }

    public string FirstName { get; set; }
    public string LastName { get; set; }
    public string BusinessName { get; set; }

    public IList<PhoneNumber> Phones { get; set; }
    public IList<EmailAddress> Emails { get; set; }

    public ContactData()
    {
        //By instantiating Phones and Emails member collections below, resolved 'Object reference not  set to an instance of an object' exception

        Phones = new List<PhoneNumber>(); 
        Emails = new List<EmailAddress>();
    }
}

下一步是将数据填充到相关实体中,因为到目前为止我只能填充父实体联系人。

为此,我必须从我的研究中发现并了解不同的方法,例如Eager loading,Lazy loading等,以便将数据加载到相关实体中。

通过在ContactManager.GetContactList()方法中使用以下语句来完成。据我所知,我使用了急切的加载方法。对于延迟加载,在这种情况下,您必须使用 子类 的虚拟成员,例如电话和电子邮件。

 var phones = cDbContext.Contacts.Include("Phones").ToList();

 var emails = cDbContext.Contacts.Include("Emails").ToList();

这是一个完整的ContactManager类代码。

class ContactManager
{
    ContactContext cDbContext = new ContactContext();

    public IList<ContactData> GetContactList()
    {
        ContactData cd = new ContactData();

        IQueryable<ContactData> cContactList = cDbContext.Contacts;

        IList<ContactData> cListData = new List<ContactData>();

        var phones = cDbContext.Contacts.Include("Phones").ToList();

        var emails = cDbContext.Contacts.Include("Emails").ToList();

        cListData = cContactList.ToList();

        return cListData;
    }
}

在了解了这一点之后,我明白@SteveGreene指出了我正确的方向。只有我遇到的问题才是我使用包含方法的lambda表达式。但是我得到了一个编译时异常'Include方法需要一个字符串'。我尝试使用include方法使用以下lambda表达式。     var phones = cDbContext.Contacts.Include(b =&gt; b.Phones).ToList(); //这导致了编译时错误'Include方法需要一个字符串'。

在传递(如下所示)参数的名称作为字符串,其名称为“电话”和“电子邮件”的ContactData类集合成员后,它运行正常。

var emails = cDbContext.Contacts.Include("Emails").ToList();

如果有任何疑问,请发表评论。