1

更新:我知道我不应该使用 aDbSet所以我按照 Erenga 的建议将实现更改为 ICollection

请考虑以下课程:

[Table("Tenant")]
public class Tenant : IEntity
{
    public int Id { get; set; }
    public string Name { get; set; }
    [Key]
    public string Guid { get; set; }
    public virtual ICollection<User> Users { get; set; }
}

[Table("User")]
public class User : IEntity
{
    public int Id { get; set; }

    public string Name { get; set; }
    public string EmailAddress { get; set; }
    public string Password { get; set; }
}

第一个测试创建一个新租户和一个新用户,并将它们存储在适当的表中。

    [Test]
    public void CreateNewUserForNewTenant()
    {
        var user = _applicationContext.Users.Create();
        user.Name = "barney";
        user.EmailAddress = "barney@flinstone.com";

        var tenant = _applicationContext.Tenants.Create();
        tenant.Name = "localhost";
        tenant.Guid = Guid.NewGuid().ToString();

        tenant.Users.Add(user); // NullReferenceException, I expected the EF would LazyLoad the reference to Users?!
        _tenantRepository.Add(tenant);
        _applicationContext.SaveChanges();
    }

NullReferenceException由于属性 Users 未初始化,因此该测试将失败。

我应该如何更改我可以依赖 EF 提供的 LazyLoading 的代码?

4

3 回答 3

2
    var tenant = new Tenant
        {
            Name = "localhost",
            Guid = Guid.NewGuid().ToString(),
            Users = new List<User> { user }
        };
于 2013-01-27T13:20:03.217 回答
2

我在这里看到了两个问题。

  1. 正如@SimonWhitehead 提到的,引用类型默认初始化为null。延迟加载仅适用于 EF 创建的实体。这些实际上是您的类的子类,其中包含延迟加载的附加逻辑。

  2. DbSet不是实体支持的集合类型。您需要将类型更改为ICollectionISetIList

这是一个工作示例

[Table("Tenant")]
public class Tenant : IEntity
{
    public int Id { get; set; }
    public string Name { get; set; }
    [Key]
    public string Guid { get; set; }
    public virtual ICollection<User> Users { get; set; }
}

[Table("User")]
public class User : IEntity
{
    public int Id { get; set; }

    public string Name { get; set; }
    public string EmailAddress { get; set; }
    public string Password { get; set; }
}

[Test]
public void CreateNewUserForNewTenant()
{
    var user = _applicationContext.Users.Create();
    user.Name = "barney"; 
    user.EmailAddress = "barney@flinstone.com";

    var tenant = _applicationContext.Tenents.Create();
    tenant.Name = "localhost";
    tenant.Guid = Guid.NewGuid().ToString();

    tenant.Users = new List<User> { user };
    _tenantRepository.Add(tenant);
    _applicationContext.SaveChanges();
}
于 2013-01-27T13:41:23.597 回答
0

我想你期待这样的事情(不是线程安全的):

[Table("Tenant")]
public class Tenant : IEntity
{
    private DbSet<User> _users;

    public int Id { get; set; }
    public string Name { get; set; }
    [Key]
    public string Guid { get; set; }

    public virtual ICollection<User> Users 
    {
        get 
        {
            if (_users == null)
                _users = new List<Users>();

            return _users;
        }
        set { _users = value; }
    }
}

我相信这个Lazy<T>类也可以以某种方式使用,但我不熟悉那个类。

于 2013-01-27T13:34:16.077 回答