繁体   English   中英

实体框架模型生成奇怪的错误消息

[英]Entity framework model generating weird error message

我有一个PostgreSql 9.04数据库,其中包含ASPNet Membership Services数据库的postgres实现。 此实现与我自己的表位于同一数据库中,但位于称为“安全”的不同架构中。

由于业务原因,我在实体模型中包含了aspnet_Users,aspnet_Membership和aspnet_Profiles表。 下面是这些表的DDL。

CREATE TABLE security.aspnet_users (
    userid UUID NOT NULL,
    applicationname VARCHAR(255) NOT NULL,
    username VARCHAR(255),
    lastactivitydate TIMESTAMP,
    isanonymous INT,
    CONSTRAINT aspnet_users_userid_pk PRIMARY KEY (userid),
    CONSTRAINT aspnet_users_username_pk UNIQUE (username, applicationname)
);

CREATE TABLE security.aspnet_membership (
    userid UUID NOT NULL,
    password VARCHAR(255),
    passwordsalt VARCHAR(255),
    passwordformat INT,
    email VARCHAR(255),
    passwordquestion VARCHAR(255),
    passwordanswer VARCHAR(255),
    comments VARCHAR(255),
    isapproved INT,
    islockedout INT,
    creationdate TIMESTAMP,
    lastlogindate TIMESTAMP,
    lastpasswordchangeddate TIMESTAMP,
    lastlockoutdate TIMESTAMP,
    failedpasswordattemptcount INT,
    failedpasswordattemptstart TIMESTAMP,
    failedpasswordanswercount INT,
    failedpasswordanswerstart TIMESTAMP,

    CONSTRAINT aspnet_membership_userid_pk PRIMARY KEY (userid),
    CONSTRAINT aspnet_membership_userid_ref FOREIGN KEY (userid) REFERENCES security.aspnet_users (userid)
);

CREATE TABLE security.aspnet_profiles (
    userid                      UUID,
    propertynames               BYTEA NOT NULL,
    propertyvaluesstring        BYTEA NOT NULL,
    propertyvaluesbinary        BYTEA NULL,
    lastupdateddate             TIMESTAMP NOT NULL,

    CONSTRAINT aspnet_profiles_userid_ref FOREIGN KEY (userid) REFERENCES security.aspnet_users (userid),
    CONSTRAINT aspnet_profiles_userid_key UNIQUE (userid)
);

同样,这些是该模式中模型中仅有的表。

现在,当我在这些表中插入一行并调用SaveChanges时,出现以下错误消息:

无法确定“ Membership_Users”关系的主要结尾。 多个添加的实体可能具有相同的主键。

以下是我用于插入新行或更新现有行的代码示例。 这代表了我正在执行的有关更新数据库的所有操作。

public static void SaveUserProfile( CarSystemEntities context, UserProfileData data ) {
    if ( context == null )
        throw new ArgumentNullException( "context", "You must pass a non-null CarSystemEntities instance." );

    if ( data == null )
        throw new ArgumentNullException( "data", "You must pass a non-null UserProfileData instance." );

    try {
        Profile profile = null;

        if ( !context.Profiles.Any( p => p.Userid == data.ID ) ) {
            profile = new CarSystem.Profile{
                LastUpdatedDate      = data.LastUpdatedDate.LocalDateTime,
                PropertyNames        = data.PropertyNames, 
                PropertyValuesBinary = data.PropertyValuesBinary, 
                PropertyValuesString = data.PropertyValuesString,
                Userid               = data.ID
            };

            context.Profiles.AddObject( profile );
        } else {
            profile = QueryUerProfiles( context ).Where( p => p.Userid == data.ID ).Single();

            if ( profile.LastUpdatedDate      != data.LastUpdatedDate )      profile.LastUpdatedDate      = data.LastUpdatedDate.LocalDateTime;
            if ( profile.PropertyNames        != data.PropertyNames )        profile.PropertyNames        = data.PropertyNames;
            if ( profile.PropertyValuesBinary != data.PropertyValuesBinary ) profile.PropertyValuesBinary = data.PropertyValuesBinary;
            if ( profile.PropertyValuesString != data.PropertyValuesString ) profile.PropertyValuesString = data.PropertyValuesString;
        }

    } catch ( Exception ex ) {
        throw new DataAccessException( DataAccessOperation.SaveProfile, FailureReason.DatabaseError, ex );
    }
}

此错误的原因是什么? 我该如何解决?

谢谢

托尼

聚苯乙烯

经过进一步研究,问题不在于插入行,而是在于尝试在与插入行的事务相同的事务中更新行。

本质上,存在要写入数据库的对象队列。 依次从队列中删除每个对象,并调用类似于上述对象的方法将其保存到正确的表中。

第一次看到特定ID时,该方法将其插入。 也就是说,Any检查返回false。 该代码创建新的实体对象,并为与该表相对应的实体集调用AddObject方法。 到现在为止还挺好。

此后,假定Any检查将返回true,表示已经存在具有给定ID的行。 然后应该更新该行。 但是,即使我在与第一次调用的表相对应的实体集上调用了AddObject,似乎Any检查也第二次返回false。

知道我在做什么错吗?

PPS

我在打开Sql监视器的情况下进行了更多测试。 我们正在使用Devart dotConnect for PostgreSql库,他们有一个可以下载的名为DbMonitor的工具。 这使您可以跟踪由实体框架发出和执行的SQL。

事实证明,对.Any的调用会作为对数据库的查询立即执行(发生时是可预测的),而一旦您调用SaveChanges,所有插入都将排队并应用。 Any调用似乎并没有考虑任何等待插入的行,只是数据库调用返回的行。 由于尚未插入要插入的行,因此该查询将始终返回false,直到调用SaveChanges。

结果,我目前的逻辑将无法正常工作。 当等待插入的行被更新时(也就是说,它再次出现在队列中),插入逻辑将再次运行。 有时,我会收到一个异常,指示违反了唯一键或主键约束,而其他时候,我会得到最初发布的消息。

我需要在单个事务中完成所有插入和更新,并且需要在调用AddObject时在数据库中进行插入。 但是,如果在插入或更新单行时出错,或者C#代码中发生其他错误,我仍然需要所有插入和更新来回滚。

有人对我如何使用实体框架做到这一点有任何想法吗?

似乎您要添加到数据库的Profile对象上没有设置主键。 Guid的默认值为Guid.Empty,因此第一个被保存,但随后的每个对象均失败。

下面的代码设置主键

if ( !context.Profiles.Any( p => p.Userid == data.ID ) ) {
            profile = new CarSystem.Profile{
                LastUpdatedDate      = data.LastUpdatedDate.LocalDateTime,
                PropertyNames        = data.PropertyNames, 
                PropertyValuesBinary = data.PropertyValuesBinary, 
                PropertyValuesString = data.PropertyValuesString,
                Userid               = data.ID

                //add the next line to ensure there is a pk field
                ID                   = Guid.NewGuid()
            };

我设法解决了这个问题,希望它能起作用。

起初我所有问题的原因是实体框架一直等到您调用SaveChanges将任何更改写入数据库。 但是,当您执行.Any()调用时,将立即生成并执行SQL,而忽略任何等待写入数据库的更改。 所以我的代码始终认为,即使其中一个是真正的插入而另一个应该是更新,也必须插入新行。

接下来,我尝试使用TransactionScope对象,在插入或更新每个对象后调用SaveChanges()。 然后,我将调用TransactionScope实例的Completed()方法来“提交”更改。 好吧,低估了,这最终的工作原理与等待调用SaveChanges时完全相同! 除非调用scope.Completed(),否则什么都不会写入数据库。

我发现可行的解决方案如下代码所示:

using ( CarSystemEntities context = new CarSystemEntities() ) { 
    if ( context.Connection.State != ConnectionState.Open ) { 
        context.Connection.Open(); 
    } 

    DbTransaction transaction = context.Connection.BeginTransaction(); 

    try { 
        <Data processing code here> 

        transaction.Commit(); 

    } catch ( Exception ex ) { 
        transaction.Rollback(); 

        <More exception code here as needed> 
    }             
}

结合在每个context..AddObject或属性更新之后调用SaveChanges,一切都按我需要的方式工作。 调用SaveChanges时,将生成并执行INSERT&Update语句。 对.Any的调用还考虑了上一次迭代插入表中的任何行。 一切都在一个实际的数据库事务中,该事务可以作为整体提交或回滚。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM