我不确定协议是什么w.r.t.改变你的问题,但我想如果我改变它会更好,因为有很多信息,并且提出问题的原始意图已经丢失。
我在使用EF(代码优先方法/ POCO类)生成数据库时最初遇到了问题。 其中一个问题是我在实体类中使用了构造函数来初始化成员。在@RickStahl建议不需要之后,我改变了我的实现。
为了读者的利益,我不希望丢失这些信息,因为一些旧的意见打算解决这个问题。
但是,在最初创建此线程的过程中,情况发生了变化。我已经能够克服一些问题。
我在从底层数据库表中检索内容时遇到问题。
例外: 对象引用未设置为对象的实例。
foreach(PhoneNumber p in cd.Phones)
为了便于理解这个问题,我正在粘贴整个源代码。
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()
{
}
}
}
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; }
}
}
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;
}
}
}
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();
}
答案 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();
如果有任何疑问,请发表评论。