다음을 통해 공유


자습서: ASP.NET MVC 5 앱에서 EF와의 동시성 처리

이전 자습서에서는 데이터를 업데이트하는 방법을 알아보았습니다. 이 자습서에서는 낙관적 동시성을 사용하여 여러 사용자가 동일한 엔터티를 동시에 업데이트할 때 충돌을 처리하는 방법을 보여 줍니다. 엔터티와 함께 Department 작동하는 웹 페이지를 변경하여 동시성 오류를 처리합니다. 다음 그림은 동시성 충돌이 발생하는 경우 표시되는 일부 메시지를 포함하여 편집 및 삭제 페이지를 보여 줍니다.

스크린샷은 현재 값이 강조 표시된 부서 이름, 예산, 시작 날짜 및 관리자 값이 있는 편집 페이지를 보여줍니다.

스크린샷은 삭제 작업에 대한 메시지와 삭제 단추가 있는 레코드의 삭제 페이지를 보여줍니다.

이 자습서에서는 다음을 수행합니다.

  • 동시성 충돌에 대해 알아보기
  • 낙관적 동시성 추가
  • 부서 컨트롤러 수정
  • 동시성 처리 테스트
  • Delete 페이지 업데이트

필수 조건

동시성 충돌

한 명의 사용자가 편집하기 위해 엔터티의 데이터를 표시한 다음, 다른 사용자가 첫 번째 사용자의 변경 내용이 데이터베이스에 기록되기 전에 동일한 엔터티의 데이터를 업데이트하는 경우 동시성 충돌이 발생합니다. 이러한 충돌의 감지를 활성화하지 않는 경우 누구든지 데이터베이스를 마지막으로 업데이트하면 다른 사용자의 변경 내용을 덮어씁니다. 많은 애플리케이션에서 이 위험은 허용 가능합니다. 적은 수의 사용자 또는 적은 업데이트가 있거나 일부 변경 내용이 덮어쓰여지는지 여부가 실제로 중요하지 않은 경우 동시성에 대한 프로그래밍의 비용은 이점보다 클 수 있습니다. 이 경우 동시성 충돌을 처리하도록 애플리케이션을 구성할 필요가 없습니다.

비관적 동시성(잠금)

애플리케이션에서 동시성 시나리오에서 실수로 인한 데이터 손실을 방지할 필요가 있는 경우 해당 작업을 수행하는 한 가지 방법은 데이터베이스 잠금을 사용하는 것입니다. 이를 비관적 동시성이라고 합니다. 예를 들어 데이터베이스에서 행을 읽기 전에 읽기 전용 또는 업데이트 액세스에 대한 잠금을 요청합니다. 업데이트 액세스에 대한 행을 잠그는 경우 변경 중인 데이터의 복사본을 가져오기 때문에 다른 사용자는 읽기 전용 또는 업데이트 액세스에 대한 행을 잠그도록 허용되지 않습니다. 읽기 전용 액세스에 대한 행을 잠그는 경우 다른 사용자도 읽기 전용에 대해 잠글 수 있지만 업데이트에 대해서는 잠글 수 없습니다.

잠금 관리에는 단점이 있습니다. 프로그램을 설정하는 데 복잡할 수 있습니다. 상당한 데이터베이스 관리 리소스가 필요하며 애플리케이션의 사용자 수가 증가할 수록 성능 문제가 발생할 수 있습니다. 이러한 이유로 모든 데이터베이스 관리 시스템은 비관적 동시성을 지원하지 않습니다. Entity Framework는 기본 제공 지원을 제공하지 않으며, 이 자습서에서는 구현 방법을 보여 주지 않습니다.

낙관적 동시성

비관적 동시성에 대한 대안은 낙관적 동시성입니다. 낙관적 동시성은 동시성 충돌 발생을 허용한 다음, 그럴 경우 적절하게 반응하는 것을 의미합니다. 예를 들어 John은 부서 편집 페이지를 실행하여 영어 부서의 예산 금액을 $350,000.00에서 $0.00로 변경합니다.

John이 저장클릭하기 전에 Jane은 동일한 페이지를 실행하고 시작 날짜 필드를 2007/9/1에서 2013년 8월 8일로 변경합니다.

John은 저장먼저 클릭하고 브라우저가 인덱스 페이지로 돌아오면 변경 내용을 확인하고 Jane은 [저장]을 클릭합니다. 다음 작업은 동시성 충돌을 처리하는 방법에 따라 결정됩니다. 몇 가지 옵션에는 다음이 포함됩니다.

  • 사용자가 수정한 속성의 추적을 유지하고 데이터베이스에서 해당하는 열만 업데이트할 수 있습니다. 예제 시나리오에서 서로 다른 속성이 두 사용자에 의해 업데이트되었기 때문에 데이터가 손실되지 않습니다. 다음에 누군가가 영어 부서를 찾아볼 때 John과 Jane의 변경 내용(2013년 8월 8일 시작 날짜 및 0달러 예산)을 모두 볼 수 있습니다.

    이 업데이트의 메서드는 데이터 손실이 발생할 수 있는 충돌 수를 줄일 수 있지만 경쟁하는 변경 내용이 동일한 엔터티의 속성에 만들어지는 경우 데이터 손실을 방지할 수 없습니다. Entity Framework가 이 방식으로 작동하는지 여부는 업데이트 코드를 구현하는 방법에 따라 달라집니다. 엔터티에 대한 모든 기존 속성 값 뿐만 아니라 새 값의 추적을 유지하기 위해 많은 양의 상태를 유지 관리해야 하므로 웹 애플리케이션에서는 종종 실용적이지 않습니다. 서버 리소스가 필요하거나 웹 페이지 자체(예: 숨겨진 필드에) 또는 쿠키에 포함되어야 하기 때문에 많은 양의 상태를 유지 관리하는 것은 애플리케이션 성능에 영향을 줄 수 있습니다.

  • 제인의 변화가 존의 변화를 덮어쓰게 할 수 있습니다. 다음에 누군가가 영어 부서를 찾아볼 때 2013년 8월 8일과 복원된 $350,000.00 값이 표시됩니다. 이를 클라이언트 우선 또는 최종 우선 시나리오라고 합니다. (클라이언트의 모든 값이 데이터 저장소에 있는 값보다 우선합니다.) 이 섹션 소개에서 설명한 것처럼 동시성 처리를 위해 코딩을 수행하지 않으면 자동으로 발생합니다.

  • Jane의 변경 내용이 데이터베이스에서 업데이트되지 않도록 방지할 수 있습니다. 일반적으로 오류 메시지를 표시하고, 데이터의 현재 상태를 표시하고, 변경 내용을 계속 적용하려는 경우 변경 내용을 다시 적용할 수 있습니다. 이를 저장소 우선 시나리오라고 합니다. (데이터 저장소 값은 클라이언트가 제출한 값보다 우선합니다.) 이 자습서에서는 Store Wins 시나리오를 구현합니다. 이 메서드는 상황에 대한 경고를 받는 사용자 없이 변경 내용을 덮어쓰지 않도록 합니다.

동시성 충돌 검색

Entity Framework에서 throw하는 OptimisticConcurrencyException 예외를 처리하여 충돌을 해결할 수 있습니다. 이러한 예외를 throw하는 시기를 확인하기 위해 Entity Framework에서 충돌을 검색할 수 있어야 합니다. 따라서 데이터베이스와 데이터 모델을 적절하게 구성해야 합니다. 충돌 검색을 활성화하기 위한 몇 가지 옵션은 다음과 같습니다.

  • 데이터베이스 테이블에서 행이 변경된 시기를 확인하는 데 사용될 수 있는 추적 열을 포함합니다. 그런 다음 SQL Update 또는 Delete 명령 절에 Where 해당 열을 포함하도록 Entity Framework를 구성할 수 있습니다.

    추적 열의 데이터 형식은 일반적으로 rowversion입니다. 행 변환 값은 행이 업데이트될 때마다 증가하는 순차적인 숫자입니다. Update 또는 Delete 명령에서 절에는 Where 추적 열의 원래 값(원래 행 버전)이 포함됩니다. 업데이트할 행이 다른 사용자가 변경한 경우 열의 rowversion 값이 원래 값과 다르므로 Update 절로 인해 Where 업데이트할 행을 Delete 찾을 수 없습니다. Entity Framework에서 명령(Delete즉, 영향을 받는 행 수가 0인 경우)에 의해 Update 업데이트된 행이 없음을 발견하면 이를 동시성 충돌로 해석합니다.

  • 테이블의 절 및 Delete 명령에 있는 모든 열 Where Update 의 원래 값을 포함하도록 Entity Framework를 구성합니다.

    첫 번째 옵션에서와 같이 행을 처음 읽 Where 은 이후 행의 내용이 변경된 경우 이 절은 엔터티 프레임워크가 동시성 충돌로 해석하는 업데이트할 행을 반환하지 않습니다. 열이 많은 데이터베이스 테이블의 경우 이 방법은 매우 큰 Where 절을 생성할 수 있으며 많은 양의 상태를 유지 관리해야 할 수 있습니다. 앞에서 설명한 것처럼 많은 양의 상태를 유지 관리하는 것은 애플리케이션 성능에 영향을 미칠 수 있습니다. 따라서 이 방법은 일반적으로 권장되지 않으며 이 자습서에서 사용되는 방법이 아닙니다.

    동시성에 대한 이 방법을 구현하려면 ConcurrencyCheck 특성을 추가하여 동시성을 추적하려는 엔터티의 모든 기본 키가 아닌 속성을 표시해야 합니다. 이러한 변경을 통해 Entity Framework는 문의 SQL WHEREUPDATE 에 모든 열을 포함할 수 있습니다.

이 자습서의 나머지 부분에서는 엔터티에 Department rowversion 추적 속성을 추가하고, 컨트롤러와 뷰를 만들고, 모든 것이 올바르게 작동하는지 테스트합니다.

낙관적 동시성 추가

Models\Department.cs 다음과 같은 RowVersion추적 속성을 추가합니다.

public class Department
{
    public int DepartmentID { get; set; }

    [StringLength(50, MinimumLength = 3)]
    public string Name { get; set; }

    [DataType(DataType.Currency)]
    [Column(TypeName = "money")]
    public decimal Budget { get; set; }

    [DataType(DataType.Date)]
    [DisplayFormat(DataFormatString = "{0:yyyy-MM-dd}", ApplyFormatInEditMode = true)]
    [Display(Name = "Start Date")]
    public DateTime StartDate { get; set; }

    [Display(Name = "Administrator")]
    public int? InstructorID { get; set; }

    [Timestamp]
    public byte[] RowVersion { get; set; }

    public virtual Instructor Administrator { get; set; }
    public virtual ICollection<Course> Courses { get; set; }
}

타임스탬프 특성은 이 열이 데이터베이스에 Where 전송되는 명령 및 DeleteUpdate 에 포함되도록 지정합니다. 이전 버전의 SQL Server는 SQL rowversion이 이를 바꾸기 전에 SQL 타임스탬프 데이터 형식을 사용했기 때문에 이 특성을 타임스탬프라고 합니다. rowversion.Net 형식은 바이트 배열입니다.

흐름 API를 사용하려는 경우 다음 예제와 같이 IsConcurrencyToken 메서드를 사용하여 추적 속성을 지정할 수 있습니다.

modelBuilder.Entity<Department>()
    .Property(p => p.RowVersion).IsConcurrencyToken();

속성을 추가하여 데이터베이스 모델을 변경했으므로 다른 마이그레이션을 수행해야 합니다. PMC(패키지 관리자 콘솔)에서 다음 명령을 입력합니다.

Add-Migration RowVersion
Update-Database

부서 컨트롤러 수정

Controllers\DepartmentController.cs 문을 추가 using 합니다.

using System.Data.Entity.Infrastructure;

DepartmentController.cs 파일에서 "LastName"의 4개 항목을 모두 "FullName"으로 변경하여 부서 관리자 드롭다운 목록에 성만이 아닌 강사의 전체 이름이 포함되도록 합니다.

ViewBag.InstructorID = new SelectList(db.Instructors, "ID", "FullName");

메서드의 기존 코드를 HttpPost Edit 다음 코드로 바꿉다.

[HttpPost]
[ValidateAntiForgeryToken]
public async Task<ActionResult> Edit(int? id, byte[] rowVersion)
{
    string[] fieldsToBind = new string[] { "Name", "Budget", "StartDate", "InstructorID", "RowVersion" };

    if (id == null)
    {
        return new HttpStatusCodeResult(HttpStatusCode.BadRequest);
    }

    var departmentToUpdate = await db.Departments.FindAsync(id);
    if (departmentToUpdate == null)
    {
        Department deletedDepartment = new Department();
        TryUpdateModel(deletedDepartment, fieldsToBind);
        ModelState.AddModelError(string.Empty,
            "Unable to save changes. The department was deleted by another user.");
        ViewBag.InstructorID = new SelectList(db.Instructors, "ID", "FullName", deletedDepartment.InstructorID);
        return View(deletedDepartment);
    }

    if (TryUpdateModel(departmentToUpdate, fieldsToBind))
    {
        try
        {
            db.Entry(departmentToUpdate).OriginalValues["RowVersion"] = rowVersion;
            await db.SaveChangesAsync();

            return RedirectToAction("Index");
        }
        catch (DbUpdateConcurrencyException ex)
        {
            var entry = ex.Entries.Single();
            var clientValues = (Department)entry.Entity;
            var databaseEntry = entry.GetDatabaseValues();
            if (databaseEntry == null)
            {
                ModelState.AddModelError(string.Empty,
                    "Unable to save changes. The department was deleted by another user.");
            }
            else
            {
                var databaseValues = (Department)databaseEntry.ToObject();

                if (databaseValues.Name != clientValues.Name)
                    ModelState.AddModelError("Name", "Current value: "
                        + databaseValues.Name);
                if (databaseValues.Budget != clientValues.Budget)
                    ModelState.AddModelError("Budget", "Current value: "
                        + String.Format("{0:c}", databaseValues.Budget));
                if (databaseValues.StartDate != clientValues.StartDate)
                    ModelState.AddModelError("StartDate", "Current value: "
                        + String.Format("{0:d}", databaseValues.StartDate));
                if (databaseValues.InstructorID != clientValues.InstructorID)
                    ModelState.AddModelError("InstructorID", "Current value: "
                        + db.Instructors.Find(databaseValues.InstructorID).FullName);
                ModelState.AddModelError(string.Empty, "The record you attempted to edit "
                    + "was modified by another user after you got the original value. The "
                    + "edit operation was canceled and the current values in the database "
                    + "have been displayed. If you still want to edit this record, click "
                    + "the Save button again. Otherwise click the Back to List hyperlink.");
                departmentToUpdate.RowVersion = databaseValues.RowVersion;
            }
        }
        catch (RetryLimitExceededException /* dex */)
        {
            //Log the error (uncomment dex variable name and add a line here to write a log.)
            ModelState.AddModelError("", "Unable to save changes. Try again, and if the problem persists, see your system administrator.");
        }
    }
    ViewBag.InstructorID = new SelectList(db.Instructors, "ID", "FullName", departmentToUpdate.InstructorID);
    return View(departmentToUpdate);
}

FindAsync 메서드가 Null을 반환하는 경우 부서가 다른 사용자에 의해 삭제되었습니다. 표시된 코드는 게시된 양식 값을 사용하여 부서 엔터티를 만들어 편집 페이지를 오류 메시지로 다시 표시할 수 있도록 합니다. 대신 부서 필드를 다시 표시하지 않고 오류 메시지만을 표시하는 경우 부서 엔터티를 다시 만들 필요가 없습니다.

뷰는 숨겨진 필드에 원래 RowVersion 값을 저장하고 메서드는 매개 변수에서 rowVersion 이 값을 받습니다. SaveChanges를 호출하기 전에 엔터티에 대한 OriginalValues 컬렉션에 해당 원래 RowVersion 속성 값을 넣어야 합니다. 그런 다음 Entity Framework에서 SQL UPDATE 명령을 만들 때 해당 명령에는 원래 RowVersion 값이 있는 행을 찾는 절이 포함 WHERE 됩니다.

명령의 영향을 받는 UPDATE 행이 없는 경우(원래 RowVersion 값이 있는 행이 없는 경우) Entity Framework는 예외를 DbUpdateConcurrencyException throw하고 블록의 catch 코드는 예외 개체에서 영향을 받는 Department 엔터티를 가져옵니다.

var entry = ex.Entries.Single();

이 개체에는 해당 속성에 Entity 사용자가 입력한 새 값이 있으며, 메서드를 호출 GetDatabaseValues 하여 데이터베이스에서 읽은 값을 가져올 수 있습니다.

var clientValues = (Department)entry.Entity;
var databaseEntry = entry.GetDatabaseValues();

다른 사용자가 데이터베이스에서 행을 삭제한 경우 메서드는 GetDatabaseValues null을 반환합니다. 그렇지 않으면 반환된 개체를 Department 클래스로 캐스팅하여 속성에 액세스 Department 해야 합니다. (이미 삭제 databaseEntry 를 확인했으므로 실행 후 FindAsync 실행하기 전에 SaveChanges 부서가 삭제된 경우에만 null이 됩니다.)

if (databaseEntry == null)
{
    ModelState.AddModelError(string.Empty,
        "Unable to save changes. The department was deleted by another user.");
}
else
{
    var databaseValues = (Department)databaseEntry.ToObject();

다음으로, 이 코드는 사용자가 편집 페이지에서 입력한 것과 다른 데이터베이스 값이 있는 각 열에 대한 사용자 지정 오류 메시지를 추가합니다.

if (databaseValues.Name != currentValues.Name)
    ModelState.AddModelError("Name", "Current value: " + databaseValues.Name);
    // ...

더 긴 오류 메시지는 발생한 작업과 이에 대해 수행할 작업을 설명합니다.

ModelState.AddModelError(string.Empty, "The record you attempted to edit "
    + "was modified by another user after you got the original value. The"
    + "edit operation was canceled and the current values in the database "
    + "have been displayed. If you still want to edit this record, click "
    + "the Save button again. Otherwise click the Back to List hyperlink.");

마지막으로, 코드는 개체의 Department 값을 데이터베이스에서 검색된 새 값으로 설정합니다RowVersion. 이 새로운 RowVersion 값은 편집 페이지가 다시 표시되고, 다음 번에 사용자가 저장을 클릭할 때 숨겨진 필드에 저장되고, 편집 페이지의 다시 표시로 인해 발생하는 동시성 오류만 catch됩니다.

Views\Department\Edit.cshtml에서 속성의 숨겨진 필드 바로 다음에 속성 값을 저장할 RowVersion 숨겨진 필드를 DepartmentID 추가합니다.

@model ContosoUniversity.Models.Department

@{
    ViewBag.Title = "Edit";
}

<h2>Edit</h2>

@using (Html.BeginForm())
{
    @Html.AntiForgeryToken()
    
    <div class="form-horizontal">
        <h4>Department</h4>
        <hr />
        @Html.ValidationSummary(true)
        @Html.HiddenFor(model => model.DepartmentID)
        @Html.HiddenFor(model => model.RowVersion)

동시성 처리 테스트

사이트를 실행하고 부서를 클릭합니다.

영어 부서의 하이퍼링크 편집 을 마우스 오른쪽 단추로 클릭하고 새 탭에서 열기를 선택한 다음 영어 부서의 하이퍼링크 편집 을 클릭합니다. 두 탭에 동일한 정보가 표시됩니다.

첫 번째 브라우저 탭의 필드를 변경하고 저장을 클릭합니다.

브라우저에 변경된 값과 인덱스 페이지가 표시됩니다.

두 번째 브라우저 탭에서 필드를 변경하고 저장을 클릭합니다. 오류 메시지가 표시됩니다.

스크린샷은 다른 사용자가 값을 변경했기 때문에 작업이 취소되었음을 설명하는 메시지와 함께 편집 페이지를 보여줍니다.

다시 저장을 클릭합니다. 두 번째 브라우저 탭에 입력한 값은 첫 번째 브라우저에서 변경한 데이터의 원래 값과 함께 저장됩니다. 인덱스 페이지가 나타날 때 저장된 값이 표시됩니다.

Delete 페이지 업데이트

삭제 페이지의 경우 Entity Framework는 비슷한 방식으로 부서를 편집하는 사용자에 의해 발생한 동시성 충돌을 감지합니다. 메서드에 HttpGet Delete 확인 보기가 표시되면 뷰에는 숨겨진 필드에 원래 RowVersion 값이 포함됩니다. 그런 다음 사용자가 삭제를 HttpPost Delete 확인할 때 호출되는 메서드에서 해당 값을 사용할 수 있습니다. Entity Framework에서 SQL DELETE 명령을 만들 때 원래 RowVersion 값이 있는 WHERE 절이 포함됩니다. 명령에서 영향을 받는 행이 0개인 경우(즉, 삭제 확인 페이지가 표시된 후 행이 변경됨) 동시성 예외가 throw되고 HttpGet Delete 오류 메시지와 함께 확인 페이지를 다시 표시하기 true 위해 오류 플래그가 설정된 상태에서 메서드가 호출됩니다. 다른 사용자가 행을 삭제했기 때문에 행 0개가 영향을 받을 수도 있으므로 이 경우 다른 오류 메시지가 표시됩니다.

DepartmentController.cs 메서드를 HttpGet Delete 다음 코드로 바꿉다.

public async Task<ActionResult> Delete(int? id, bool? concurrencyError)
{
    if (id == null)
    {
        return new HttpStatusCodeResult(HttpStatusCode.BadRequest);
    }
    Department department = await db.Departments.FindAsync(id);
    if (department == null)
    {
        if (concurrencyError.GetValueOrDefault())
        {
            return RedirectToAction("Index");
        }
        return HttpNotFound();
    }

    if (concurrencyError.GetValueOrDefault())
    {
        ViewBag.ConcurrencyErrorMessage = "The record you attempted to delete "
            + "was modified by another user after you got the original values. "
            + "The delete operation was canceled and the current values in the "
            + "database have been displayed. If you still want to delete this "
            + "record, click the Delete button again. Otherwise "
            + "click the Back to List hyperlink.";
    }

    return View(department);
}

메서드는 동시성 오류가 발생한 후 페이지가 다시 표시되고 있는지 여부를 나타내는 선택적 매개 변수를 허용합니다. 이 플래그인 true경우 속성을 사용하여 오류 메시지가 보기로 ViewBag 전송됩니다.

메서드의 코드 HttpPost Delete (명명됨 DeleteConfirmed)를 다음 코드로 바꿉니다.

[HttpPost]
[ValidateAntiForgeryToken]
public async Task<ActionResult> Delete(Department department)
{
    try
    {
        db.Entry(department).State = EntityState.Deleted;
        await db.SaveChangesAsync();
        return RedirectToAction("Index");
    }
    catch (DbUpdateConcurrencyException)
    {
        return RedirectToAction("Delete", new { concurrencyError = true, id=department.DepartmentID });
    }
    catch (DataException /* dex */)
    {
        //Log the error (uncomment dex variable name after DataException and add a line here to write a log.
        ModelState.AddModelError(string.Empty, "Unable to delete. Try again, and if the problem persists contact your system administrator.");
        return View(department);
    }
}

방금 바꾼 스캐폴드된 코드에서 이 메서드는 레코드 ID만 허용했습니다.

public async Task<ActionResult> DeleteConfirmed(int id)

이 매개 변수를 모델 바인더로 만든 Department 엔터티 인스턴스로 변경했습니다. 이렇게 하면 레코드 키 외에 RowVersion 속성 값에 액세스할 수 있습니다.

public async Task<ActionResult> Delete(Department department)

또한 DeleteConfirmed에서 Delete로 작업 메서드 이름을 변경했습니다. 메서드에 고유한 서명을 제공하기 HttpPost 위해 메서드 DeleteConfirmed 라는 HttpPost Delete 스캐폴드된 코드입니다. (CLR에는 오버로드된 메서드에 다른 메서드 매개 변수가 있어야 합니다.) 이제 서명이 고유하므로 MVC 규칙을 고수하고 메서드 및 HttpGet 삭제 메서드에 대해 HttpPost 동일한 이름을 사용할 수 있습니다.

동시성 오류가 catch되는 경우 코드는 삭제 확인 페이지를 다시 표시하고 동시성 오류 메시지를 표시해야 함을 나타내는 플래그를 제공합니다.

Views\Department\Delete.cshtml에서 스캐폴드된 코드를 다음 코드로 바꿔서 DepartmentID 및 RowVersion 속성에 대한 오류 메시지 필드와 숨겨진 필드를 추가합니다. 변경 내용은 강조 표시되어 있습니다.

@model ContosoUniversity.Models.Department

@{
    ViewBag.Title = "Delete";
}

<h2>Delete</h2>

<p class="error">@ViewBag.ConcurrencyErrorMessage</p>

<h3>Are you sure you want to delete this?</h3>
<div>
    <h4>Department</h4>
    <hr />
    <dl class="dl-horizontal">
        <dt>
            Administrator
        </dt>

        <dd>
            @Html.DisplayFor(model => model.Administrator.FullName)
        </dd>

        <dt>
            @Html.DisplayNameFor(model => model.Name)
        </dt>

        <dd>
            @Html.DisplayFor(model => model.Name)
        </dd>

        <dt>
            @Html.DisplayNameFor(model => model.Budget)
        </dt>

        <dd>
            @Html.DisplayFor(model => model.Budget)
        </dd>

        <dt>
            @Html.DisplayNameFor(model => model.StartDate)
        </dt>

        <dd>
            @Html.DisplayFor(model => model.StartDate)
        </dd>

    </dl>

    @using (Html.BeginForm()) {
        @Html.AntiForgeryToken()
        @Html.HiddenFor(model => model.DepartmentID)
        @Html.HiddenFor(model => model.RowVersion)

        <div class="form-actions no-color">
            <input type="submit" value="Delete" class="btn btn-default" /> |
            @Html.ActionLink("Back to List", "Index")
        </div>
    }
</div>

이 코드는 제목과 h3 제목 사이에 h2 오류 메시지를 추가합니다.

<p class="error">@ViewBag.ConcurrencyErrorMessage</p>

필드로 FullName Administrator 바뀝니다LastName.

<dt>
  Administrator
</dt>
<dd>
  @Html.DisplayFor(model => model.Administrator.FullName)
</dd>

마지막으로 문 뒤 Html.BeginFormRowVersion 속성에 DepartmentID 대한 숨겨진 필드를 추가합니다.

@Html.HiddenFor(model => model.DepartmentID)
@Html.HiddenFor(model => model.RowVersion)

부서 인덱스 페이지를 실행합니다. 영어 부서의 하이퍼링크 삭제를 마우스 오른쪽 단추로 클릭하고 새 탭에서 열기를 선택한 다음 첫 번째 탭에서 영어 부서의 하이퍼링크 편집을 클릭합니다.

첫 번째 창에서 값 중 하나를 변경하고 저장을 클릭합니다.

인덱스 페이지에서 변경 사항을 확인합니다.

두 번째 탭에서 삭제를 클릭합니다.

동시성 오류 메시지가 표시되며 부서 값은 데이터베이스의 현재 값으로 새로 고쳐집니다.

Department_Delete_confirmation_page_with_concurrency_error

삭제를 다시 클릭하면 부서가 삭제되었음을 보여 주는 인덱스 페이지로 리디렉션됩니다.

코드 가져오기

완료된 프로젝트 다운로드

추가 리소스

다른 Entity Framework 리소스에 대한 링크는 ASP.NET 데이터 액세스 - 권장 리소스에서 찾을 수 있습니다.

다양한 동시성 시나리오를 처리하는 다른 방법에 대한 자세한 내용은 낙관적 동시성 패턴MSDN의 속성 값 작업을 참조하세요. 다음 자습서에서는 계층별 테이블 상속과 Student 엔터티를 구현하는 Instructor 방법을 보여 줍니다.

다음 단계

이 자습서에서는 다음을 수행합니다.

  • 동시성 충돌에 대해 알아보기
  • 낙관적 동시성 추가됨
  • 수정된 부서 컨트롤러
  • 테스트된 동시성 처리
  • 삭제 페이지 업데이트

데이터 모델에서 상속을 구현하는 방법을 알아보려면 다음 문서로 진행합니다.