1. ホーム
  2. c#

[解決済み] ASP.NET MVC - 'MODELNAME' 型のエンティティのアタッチに失敗しました。

2022-06-01 11:03:22

質問

簡単に言うと、ラッパーモデルをPOSTして、あるエントリーの状態を'Modified'に変更する際に例外が発生します。状態を変更する前に、状態は'Detached'に設定されていますが、Attach()を呼び出すと、同じエラーがスローされます。私はEF6を使用しています。

以下、私のコードをご覧ください(読みやすくするため、モデル名を変更しました)。

モデル名

// Wrapper classes
        public class AViewModel
        {
            public A a { get; set; }
            public List<B> b { get; set; }
            public C c { get; set; }
        }   

コントローラ

        public ActionResult Edit(int? id)
        {
            if (id == null)
            {
                return new HttpStatusCodeResult(HttpStatusCode.BadRequest);
            }

            if (!canUserAccessA(id.Value))
                return new HttpStatusCodeResult(HttpStatusCode.Forbidden);

            var aViewModel = new AViewModel();
            aViewModel.A = db.As.Find(id);

            if (aViewModel.Receipt == null)
            {
                return HttpNotFound();
            }

            aViewModel.b = db.Bs.Where(x => x.aID == id.Value).ToList();
            aViewModel.Vendor = db.Cs.Where(x => x.cID == aViewModel.a.cID).FirstOrDefault();

            return View(aViewModel);
        }

[HttpPost]
        [ValidateAntiForgeryToken]
        public ActionResult Edit(AViewModel aViewModel)
        {
            if (!canUserAccessA(aViewModel.a.aID) || aViewModel.a.UserID != WebSecurity.GetUserId(User.Identity.Name))
                return new HttpStatusCodeResult(HttpStatusCode.Forbidden);

            if (ModelState.IsValid)
            {
                db.Entry(aViewModel.a).State = EntityState.Modified; //THIS IS WHERE THE ERROR IS BEING THROWN
                db.SaveChanges();
                return RedirectToAction("Index");
            }
            return View(aViewModel);
        }

上の行のように

db.Entry(aViewModel.a).State = EntityState.Modified;

は例外を投げます。

タイプ 'A' のエンティティのアタッチは、同じタイプの別のエンティティが既に同じ主キー値を持っているため、失敗しました。 同じ型の別のエンティティがすでに同じ主キー値を持っているためです。これは、以下の場合に発生します。 Attach' メソッドを使用した場合、またはエンティティの状態を グラフ内のエンティティが競合するキー値を持つ場合、'Attach' メソッドを使用するか、エンティティの状態を 'Unchanged' または 'Modified' に設定すると発生します。 グラフ内のエンティティが矛盾するキー値を持つ場合です。これは、いくつかのエンティティが新規で がまだデータベースで生成されたキー値を受け取っていないためと思われます。この場合は Add' メソッドまたは 'Added' エンティティの状態を使用してグラフを追跡し、新規でないエンティティの状態を設定します。 新規でないエンティティの状態を 'Unchanged' または 'Modified' に設定します。 に設定します。

どなたか、私のコードに何か間違いがあるとか、モデルの編集中にこのようなエラーを投げるのはどのような状況なのかを理解している方はいらっしゃいますか?

どのように解決するのですか?

問題は解決しました。

Attach メソッドは誰かの役に立つかもしれませんが、この状況では役に立ちません。なぜなら、Edit GET コントローラ関数で読み込まれたドキュメントはすでに追跡されていたからです。Attach はまったく同じエラーを投げるでしょう。

私がここで遭遇した問題は、関数 canUserAccessA() がオブジェクト a の状態を更新する前に A のエンティティをロードしていたためです。これは追跡されたエンティティを台無しにしてしまい、オブジェクト a の状態を Detached .

解決策は canUserAccessA() を修正して、読み込んでいるオブジェクトが追跡されないようにすることでした。機能 AsNoTracking() はコンテキストをクエリしている間に呼び出されなければなりません。

// User -> Receipt validation
private bool canUserAccessA(int aID)
{
    int userID = WebSecurity.GetUserId(User.Identity.Name);
    int aFound = db.Model.AsNoTracking().Where(x => x.aID == aID && x.UserID==userID).Count();

    return (aFound > 0); //if aFound > 0, then return true, else return false.
}

なぜか .Find(aID)AsNoTracking() を追加しましたが、クエリを変更することで同じことを達成できたので、それは本当に重要ではありません。

これが同じような問題を持つ誰かの助けになることを願っています。