[解決済み] ASP.NET MVC - 'MODELNAME' 型のエンティティのアタッチに失敗しました。
質問
簡単に言うと、ラッパーモデルを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()
を追加しましたが、クエリを変更することで同じことを達成できたので、それは本当に重要ではありません。
これが同じような問題を持つ誰かの助けになることを願っています。
関連
-
[解決済み] [Solved] 1つ以上のエンティティで検証に失敗しました。詳細は'EntityValidationErrors'プロパティを参照してください [重複]。
-
[解決済み】C#におけるtypedefの等価性
-
[解決済み】文字列が有効な DateTime " format dd/MM/yyyy " として認識されなかった。
-
[解決済み] 'SubSonic.Schema .DatabaseColumn' 型のオブジェクトをシリアライズする際に、循環参照が検出されました。
-
[解決済み】Unity3DでOnTriggerEnterが動作しない件
-
[解決済み】Swashbuckle/Swagger + ASP.Net Core: "Failed to load API definition" (API定義の読み込みに失敗しました
-
[解決済み】"指定されたパスのフォーマットはサポートされていません。"
-
[解決済み】「...は'型'であり、与えられたコンテキストでは有効ではありません」を解決するにはどうすればよいですか?(C#)
-
VSでscanfエラーを恒久的に解決するには、ソースファイルを作成し、自動的に#define _CRT_SECURE_NO_WARNINGS 1を追加してください。
-
[解決済み] Entity Framework 5 レコードを更新する
最新
-
nginxです。[emerg] 0.0.0.0:80 への bind() に失敗しました (98: アドレスは既に使用中です)
-
htmlページでギリシャ文字を使うには
-
ピュアhtml+cssでの要素読み込み効果
-
純粋なhtml + cssで五輪を実現するサンプルコード
-
ナビゲーションバー・ドロップダウンメニューのHTML+CSSサンプルコード
-
タイピング効果を実現するピュアhtml+css
-
htmlの選択ボックスのプレースホルダー作成に関する質問
-
html css3 伸縮しない 画像表示効果
-
トップナビゲーションバーメニュー作成用HTML+CSS
-
html+css 実装 サイバーパンク風ボタン
おすすめ
-
[解決済み】C#におけるtypedefの等価性
-
[解決済み】ASP.NET Core Dependency Injectionのエラーです。アクティブ化しようとしているときに、タイプのサービスを解決できません。
-
[解決済み】トランスポート接続からデータを読み取れない:既存の接続は、リモートホストによって強制的に閉じられました。
-
[解決済み】C# ASP.NET使用時に「WebClientのリクエスト中に例外が発生しました。
-
[解決済み】バックスラッシュを含むパス文字列のエスケープシーケンスが認識されない件
-
[解決済み】リソースの読み込みに失敗した:ステータス500(内部サーバーエラー)のサーバーの応答)
-
[解決済み】HRESULTからの例外:0x800A03ECエラー
-
[解決済み】2つ(またはそれ以上)のリストを1つに統合する(C# .NETで
-
[解決済み] 2つのリストを結合する
-
[解決済み】URLから画像をダウンロードする方法