28 tháng 5, 2017

PostgreSQL Concurrency và MVCC

Một trong những điểm hấp dẫn nhất của hệ quản trị cơ sở dữ liệu Postgres đó là cách mà nó thực hiện điều khiển tương tranh giữa các transaction, nghĩa là read sẽ không bao giờ chặn write và ngược lại. Nói một cách dễ hình dung hơn, nếu hai transaction thực thi cùng một lúc thì nguyên tắc thực thi là thực thi độc lập, Postgres thực hiện được điều này là nhờ một cơ chế gọi là Multi Version Concurrency Control (MVCC). Kĩ thuật này không phải chỉ riêng Postgres mới có, mà đã có nhiều hệ quản trị cơ sở dữ liệu khác cũng đang thực hiện các cách tương tự với MVCC để điều khiển tương tranh giữa các transaction bao gồm: Oracle, Berkeley DB, CouchDB và các hệ quản trị cơ sở dữ liệu khác nữa. Việc hiểu cách MVCC được triển khai như thế nào trong PostgreSQL rất quan trọng, vì điều này giúp các nhà phát triển phần mềm có thể thiết kế các ứng dụng có tính đồng thời cao mà có sử dụng đến PostgreSQL, hoặc phần nào giúp giải quyết được rất nhiều vấn đề khó có thể gặp phải trong tương lai.

MVCC hoạt động như thế nào?

Mỗi một transaction trong PostgreSQL đều có một transaction id - id này là một số nguyên 32-bit, gọi là XID. Các transaction này bao gồm các câu lệnh đơn như INSERT, UPDATE hoặc DELETE, và một nhóm các câu lệnh tạo thành một khối lệnh được đặt bên trong hai từ khóa BEGIN - COMMIT. Khi bắt đầu một transaction, Postgres tạo ra một XID và gán nó cho transaction hiện tại. Ta có thể nhìn thấy được XID của transaction hiện tại bằng cách gọi hàm txid_current() có sẵn trong PostgreSQL.

SELECT CAST(txid_current() AS TEXT);
Postgres lưu tất cả thông tin của một transaction vào bên trong table data của hệ thống, những thông tin này sẽ được Postgres dùng để xác định một bản ghi sẽ có trạng thái ẩn hay hiện đối với một transaction.

Một điều thú vị nữa trong cơ chế MVCC này của Postgres đó là ngoài các column đã được khai báo trong quá trình tạo bảng thì mỗi bản ghi của một bảng sẽ có thêm hai cột bổ sung:

  • xmin - xác định XID của transaction đã insert bản ghi này
  • xmax - xác định XID của transaction đã delete bản ghi này

Nếu ta không truy vấn tới hai cột này, mặc định chúng sẽ bị ẩn đi. Ta có thể truy vấn giá trị của hai cột này như các cột bình thường khác có trong một bảng.

Ví dụ ta có câu truy vấn DDL sau:

CREATE TABLE numbers (value int);

Để lấy được hai giá trị này ta chỉ cần truy vấn:

SELECT *, xmin, xmax FROM numbers;

Quá trình thực hiện chèn một bản ghi

Để hiểu INSERT hoạt động như thế nào trong MVCC, cùng xem xét sơ đồ sau:

  1. Alice và Bob khởi tạo transaction mới, ta có thể thấy XID của hai transaction này bằng cách gọi hàm txid_current() trong PostgreSQL.
  2. Khi Alice chèn một bản ghi mới vào bảng post, giá trị cột xmin sẽ được gán bằng XID của transaction Alice.
  3. Mặc định PostgreSQL sử dụng mức độ cô lập READ COMMITTED, vì vậy Bob sẽ không thể nhìn thấy được bản ghi mà Alice đã chèn thêm vào bảng post cho đến khi Alice thực hiện commit transaction.
  4. Sau khi Alice đã thực hiện commit, bây giờ Bob mới có thế nhìn thấy được bản ghi mới mà Alice đã thêm.
  5. Bất kì transaction nào có XID > xmin của bản ghi đã được chèn vào, thì đều có thể thấy được bản ghi đó.

    Nếu transaction bất kì có XID < xmin của một bản ghi đã được thêm, thì Postgres sẽ dựa vào mức độ cô lập transaction để quyết định một bản ghi sẽ bị ẩn hay hiện đối với transaction đó. Với mức độ READ COMMITED (mặc định), transaction hiện tại sẽ không thể thấy được sự thay đổi của một bản ghi nếu transaction chèn bản ghi chưa được commit. Với mức độ REPEATABLE READ hoặc SERIALIZABLE, thì transaction hiện tại có thể thấy được sự thay đổi của bản ghi mặc dù transaction chèn dữ liệu chưa được commit.

Quá trình thực hiện xóa một bản ghi

Để hiểu cách DELETE hoạt động trong MVCC, cùng xem xét sơ đồ sau:

  1. Alice và Bob khởi tạo transaction mới, ta có thể thấy XID của hai transaction này bằng cách gọi hàm txid_current() trong PostgreSQL.
  2. Khi Bob thực hiện xóa một bản ghi của bảng post, giá trị của cột xmax được gán bằng với giá trị XID của transaction Bob.
  3. Mặc định PostgreSQL sử dụng mức độ cô lập READ COMMITTED, vì vậy cho đến khi Bob vẫn chưa thực hiện commit transaction, thì Alice vẫn sẽ nhìn thấy bản ghi sẽ bị Bob xóa khỏi bảng post.
  4. Sau khi Bob thực hiện commit transaction thì Alice không còn nhìn thấy bản ghi đã bị Bob xóa khỏi bảng post nữa.

Thao tác DELETE dữ liệu không trực tiếp xóa ngay lập tức bản ghi khỏi bộ nhớ vật lý, nó chỉ đánh dấu một bản ghi đã sẵn sàng để xóa. Tiến trình VACUUM sẽ thực hiện thu hồi bộ nhớ đã cấp phát cho bản ghi này khi nào nó không còn được sử dụng bởi bất kì transaction nào đang chạy.

    Nếu một transaction có XID > xmax của một bản ghi đã được commit, thì transaction này không thể đọc bản ghi này nữa.

    Nếu một transaction có XID < xmax, thì PostgreSQL sẽ dựa vào mức độ cô lập để quyết định xem bản ghi bị ẩn hay có thể đọc đối với transaction này. Với mức độ READ COMMITED (mặc định), transaction hiện tại vẫn có thể thấy được bản ghi, nếu thao tác DELETE chưa được commit. Với mức độ REPEATABLE READ hoặc SERIALIZABLE, thì transaction hiện tại không thể thấy được bản ghi mặc dù transaction xóa dữ liệu chưa được commit.

Quá trình thực hiện cập nhật một bản ghi

Để hiểu cách UPDATE hoạt động trong MVCC, cùng xem xét sơ đồ sau:

  1. Alice và Bob khởi tạo transaction mới, ta có thể thấy XID của hai transaction này bằng cách gọi hàm txid_current() trong PostgreSQL.
  2. Khi Bob thực hiện thao tác UPDATE một bản ghi của bảng post, ta có thể thấy có hai thao tác xảy ra: một DELETE và một INSERT.

    Bản ghi trước đó bị đánh dấu sẵn sàng xóa bằng việc gán giá trị xmax = XID của transaction Bob, và một bản ghi mới được tạo ra được gán giá trị xmin = XID của transaction Bob.

  3. Mặc định PostgreSQL sử dụng mức độ cô lập transaction là READ COMMITTED, vì vậy cho đến khi Bob vẫn chưa thực hiện commit transaction, thì Alice vẫn sẽ nhìn thấy bản ghi trước đó mà chưa bị Bob cập nhật.
  4. Sau khi Bob thực hiện commit transaction, thì Alice đã có thể thấy được bản ghi mới mà Bob cập nhật.

Không có nhận xét nào:

Đăng nhận xét