Luận văn -Phương pháp kiểm chứng tính đúng đắn của một chương trình Java đa luồng thông qua sử dụng logic Hoare

Với sự phát triển không ngừng của nhu cầu tính toán, xử lý số liệu, khai phá thông tin thì lập trình đa luồng ngày càng quan trọng và là một phần không thể thiếu trong các hệ thống tính toán, cũng như trong các hệ thống ứng dụng. Sự quan trọng và cần thiết của việc kiểm chứng tính đúng của chương trình luôn được quan tâm đúng mức do vai trò to lớn của nó.

pdf64 trang | Chia sẻ: lylyngoc | Lượt xem: 2447 | Lượt tải: 0download
Bạn đang xem trước 20 trang tài liệu Luận văn -Phương pháp kiểm chứng tính đúng đắn của một chương trình Java đa luồng thông qua sử dụng logic Hoare, để xem tài liệu hoàn chỉnh bạn click vào nút DOWNLOAD ở trên
ớp của các đối tác đang giao tiếp, và các điều kiện trước của các lệnh đang giao tiếp là đúng trước giao tiếp. Đối với sự thi hành phương thức, điều kiện trước của phương thức được gọi là bất biến lớp của nó. Hai lệnh đại diện các đối tác đang giao tiếp được nắm giữ bởi khẳng định comm, phụ thuộc vào kiểu của giao tiếp: Với sự thi hành phương thức  me0 , khẳng định   '0 zzE  phát biểu rằng giá trị của z’ nhận dạng đối tượng được gọi. Chú ý rằng sự thi hành phương thức điều khiển thông qua sự trả về địa chỉ như là một tham biến phụ trợ, và các giá trị của các tham biến hình thức còn lại không được thay đổi. Hơn nữa, các tham biến không thể chứa các biến thể hiện, nghĩa là, thể hiện của chúng không thay đổi trong suốt quá trình thi hành phương thức. Bởi vậy, các tham biến hình thức và các tham biến thực sự có thể được sử dụng trong sự trả về từ một phương thức để nhận dạng các đối tác đang trong mối quan hệ caller- - 38 - callee, bằng cách sử dụng các biến phụ trợ. Do đó với trường hợp trả về, comm thêm vào phát biểu  zEu  , trong đó u và e là các biến hình thức và biến thực sự. Sự trả về từ phương thức run kết thúc luồng đang thi hành; phương thức run không có các kết quả với giao tiếp. Các thay đổi trạng thái được biểu thị bởi các phép gán. Ví dụ của sự thi hành một phương thức, giao tiếp được phát biểu bởi các phép gán  zEu : , trong đó sự khởi tạo của các biến cục bộ v được bao hàm bởi  vInitv :' . Các phép gán  zEyz 11 :.  và  '':''. 22 zEyz  đại diện cho các quan sát gọi và quan sát được gọi 11 : ey và 22 : ey , theo thứ tự được thi hành trong z và z’. Định nghĩa 2.4.14. (Kiểm tra sự hợp tác: Sự giao tiếp) Một phác thảo chứng minh thỏa mãn kiểm tra sự hợp tác với giao tiếp, nếu G|         nullznullzcommzIzQzIzPGI cc  '''' '11 commf     ''22 zQzP  và (2.6) G|         nullznullzcommzIzQzIzPGI cc  '''' '11 21;; obsobscomm fff (2.7)     ''33 zQzPGI  là đúng với các biến logic mới khác nhau cLVarz và '' cLVarz  , trong các trường hợp sau: 1.CALL: Với tất cả các lệnh       waitcallcallret peypemeup 3 ! 11 ! 201 :.:  (hoặc là các lệnh này mà không nhận một giá trị) trong lớp c với 0e của kiểu c’, trong đó phương thức m của c’ có thân     ret callcall ereturnstmqeyq ;: 3 ? 22 ? 2  , các tham biến hình thức u , và các biến cục bộ v trừ các tham biến hình thức. Bất biến của lớp được gọi là '1 cIq  . Khẳng định comm được cho bởi   '0 zzE  . Hơn nữa, commf là    vInitzEvu ,:','  , 1obsf là  zEyz 11 :.  , và 2obsf là  '':''. 22 zEyz  . - 39 - 2. RETURN: Với tất cả        3 ? 44 ? 21 ! 0 :.: peyppstmemeu retretwaitcall ret  hoặc không nhận một giá trị đang xuất hiện trong c với 0e của kiểu c’, sao cho phương thức m của c’ có lệnh trả về      3 ! 33 ! 21 : qeyqereturnq retret ret  , và danh sách biến hình thức u , các phương trình trên phải là đúng với comm được cho bởi    zEuzzE  ''0 , và trong đó commf là  '': zEu retret  , 1obsf là  '':''. 33 zEyz  , và 2obsf là  zEyz 44 :.  . 3. runRETURN : Với      3 ! 33 ! 21 : qeyqreturnq retret  đang xuất hiện trong phương thức run của lớp chính, trueppp  321 , truecomm  , và hơn nữa commf và 2obsf là các lệnh rỗng, và 1obsf là  '':''. 33 zEyz  . Bên cạnh các lời gọi phương thức và các trả về phương thức, kiểm tra sự hợp tác cần điều khiển Ví dụ 2.4.15. Ví dụ này minh họa cách chứng minh các thuộc tính của việc truyền tham biến. Cho    evmep ,.0 , với p được cho bằng v > 0, là một lệnh được chú thích trong lớp c với 0e của kiểu c’, và cho phương thức  ,um của c’ có thân của dạng   returnstmq ; trong đó q là u > 0. Tính quy nạp của phác thảo chứng minh yêu cầu rằng nếu p là đúng trước khi gọi (bên cạnh sự đúng đắn của các bất biến toàn cục và bất biến lớp), thì q được thỏa mãn sau sự thi hành. Điều kiện 2.7 của kiểm tra sự hợp tác yêu cầu chứng minh      '':'| zQvuzPG  , khai triển thành    0':'0|  uvuvG . Ví dụ 2.4.16. Ví dụ sau chứng minh cách biểu thị sự phụ thuộc giữa các trạng thái thể hiện trong cấu hình toàn cục và sử dụng thông tin này trong kiểm tra giao tiếp. Cho    emep .0 , với p được cho bằng oex  00 , là một lệnh được chú thích trong lớp c với 0e của kiểu c’, x là một biến thể hiện kiểu nguyên, và o là một biến thể hiện của kiểu c’, và cho phương thức  um của c’ có chú thích thân phương thức   returnstmq ; trong đó q là y > 0 và y là một biến thể hiện kiểu nguyên. Hơn nữa cho cLVarz và cho bất biến toàn cục được cho bằng   0..0...  yozxznulloznullzz . Tính quy nạp yêu cầu rằng nếu p và bất biến toàn cục là đúng trước khi gọi, thì q được thỏa mãn sau sự thi hành. Điều kiện 2.7 của kiểm tra sự hợp tác     0..0...| yozxznulloznullzzG - 40 -      nullznullzzzEozzExz  ''.0. 00  zEu :'  0'. yz tạo đối tượng lượng hóa bằng z, mệnh đề kéo theo ozzyoz .'0..  , có nghĩa là 0'. yz . Bên cạnh các lời gọi phương thức và các sự trả về, kiểm tra sự hợp tác yêu cầu quản lý tạo đối tượng, duy trì các bất biến toàn cục, điều kiện sau của lệnh new và quan sát của nó, và các bất biến lớp của của đối tượng mới. Ta có thể giả thiết rằng điều kiện trước của lệnh tạo đối tượng, bất biến lớp của đối tượng tạo, và bất biến toàn cục là đúng trong cấu hình trước khi tạo đối tượng. Sự mở rộng của trạng thái toàn cục với đối tượng được tạo mới được đưa ra trong điều kiện sau mạnh nhất, có nghĩa là, nó được yêu cầu là đúng ngay sau khi tạo đối tượng. Ta sử dụng lượng hóa tồn tại để tham chiếu tới giá trị cũ: z’ của kiểu listObjectLVar đại diện các đối tượng đang tồn tại trước khi mở rộng. Hơn nữa, định danh đối tượng được tạo ra được lưu trữ trong u là mới và thể hiện mới có thể được khởi tạo được biểu thị bằng khẳng định toàn cục  uzFresh ,' được định nghĩa là     uvzvObjectvzuuInitState  '.:' . Để biểu thị rằng một khẳng định tham chiếu tới tập các đối tượng đang tồn tại trước sự mở rộng của trạng thái toàn cục, ta chỉ cần giới hạn lượng hóa phổ biến trong khẳng định tới dãy thông qua các đối tượng từ z’. Cho P là một khẳng định toàn cục và listObjectLVarz ' là một biến logic không xuất hiện trong P. Thì 'zP  là khẳng định toàn cục với tất cả các lượng hóa '. Pz được thay thế bằng   ''. Pzzobjz  , trong đó obj(v) ký hiệu tập các đối tượng xuất hiện trong giá trị v. Bổ đề 2.4.18. Giả thiết rằng một trạng thái toàn cục  , một sử mở rộng  initcinst ,'   đối với   ValVal c  , , và một môi trường logic  chỉ tham chiếu tới các giá trị đang tồn tại trong  . Cho v là dãy bao gồm tất cả các phần tử của  nullcc Val . Thì đối với tất cả các khẳng định P và các biến logic listObjectLVarz ' không xuất hiện trong P, PG|, khi và chỉ khi   '|',' zPvz G   . Vị từ   '. zPu  được ước lượng ngay sau khi tạo đối tượng cnewu : , biểu thị rằng P đúng trước khi tạo đối tượng mới. - 41 - Định nghĩa 2.4.19. (Kiểm tra sự hợp tác: Tạo đối tượng) Một phác thảo chứng minh thỏa mãn kiểm tra sự hợp tác đối với việc tạo đối tượng, nếu với tất cả các lớp c’ và các lệnh      321 :: peypnewup newnewc  trong c’: G|         '.,''. '1 zzIzPuGIuzFreshzuznullz c     uIzP c 2 và G|          '.,''. '1 zzIzPuGIuzFreshzuznullz c   zEyz :.   zPGI 3 là đúng với 'cLVarz và listObjectLVarz ' mới. - 42 - CHƯƠNG 3. NGÔN NGỮ TƯƠNG TRANH 3.1. Cú pháp  startmethmethmethcclassclass runmeth...:: classclass main :: mainclassclassclassprog ...:: Ta tập trung vào khía cạnh tương tranh, tất cả các lớp là các lớp Thread theo nghĩa của Java: Mỗi lớp chứa một phương thức start được định nghĩa trước có thể được thi hành chỉ một lần đối với mỗi đối tượng, kết quả là tạo ra một luồng mới của sự thi hành. Luồng mới bắt đầu thi hành phương thức run được định nghĩa bởi người dùng của đối tượng được cho trong khi luồng ban đầu tiếp tục sự thi hành của nó. Phương thức run không thể được thi hành trực tiếp. Phương thức start không có tham biến không có giá trị trả về thì không được cài đặt một cách đúng cú pháp. Cú pháp không cho phép các tham chiếu hạn chế tới các biến thể hiện. Như là một hệ quả, các biến chia sẻ tương tranh được tạo ra bởi sự thi hành tương tranh chỉ trên một đối tượng duy nhất, không phải thông qua tập các đối tượng. 3.2. Ngữ nghĩa Các ngữ nghĩa toán tử của concJava mở rộng các ngữ nghĩa của seqJava bởi tạo các luồng động. Các quy tắc thêm:                 start c CALL stmstarteT stmstarteTstartedVale      ,;.,, ,;.,,,        ,,,,,, ,, cruncruninit bodystmT                     skip startCALLstmTstmstarteT stmstarteTstartedVale    ,,,,;.,, ,;.,,,     Sự thi hành phương thức start tạo ra một luồng mới bắt đầu thi hành phương thức run của đối tượng được gọi. Chỉ có lần gọi đầu tiên thi hành phương thức start có hiệu quả (quy tắc skipstartCALL ). Vị từ  ,Tstarted là đúng khi và chỉ khi tồn tại một ngăn - 43 - xếp có dạng  000 ,, stm  111 ,, stm …  nnn stm,,  T sao cho 0  . Một luồng kết thúc vòng đời của nó bằng sự trả về từ một phương thức run. 3.3. Hệ chứng minh 3.3.1. Phác thảo chứng minh Để đạt được một hệ chứng minh đầy đủ, đối với ngôn ngữ tương tranh ta phải có khả năng nhận dạng các luồng. Ta nhận dạng một luồng qua đối tượng mà trong đó nó bắt đầu sự thi hành. Do đó ta sử dụng kiểu Thread như là sự viết tắt cho kiểu Object. Một định danh là duy nhất, bởi vì một luồng của đối tượng chỉ có thể được bắt đầu một lần. Trong suốt một lời gọi phương thức, luồng phương thức được gọi nhận định danh của nó như là tham biến phụ trợ hình thức được tích hợp thread. Ta mở rộng tham biến hình thức phụ trợ caller bằng định danh luồng gọi, nghĩa là, cho caller là kiểu ThreadIntObject  , lưu trữ các định danh của đối tượng của luồng gọi, lời gọi cấu hình cục bộ, và luồng gọi. Chú ý các định danh luồng của đối tượng gọi và đối tượng được gọi là như nhau trong tất cả các trường hợp ngoại trừ đối với sự thi hành của phương thức start. Phương thức run của đối tượng ban đầu được thi hành với các tham biến  callerthread , có các giá trị   nullnull ,0,,0 , trong đó 0 biểu thị đối tượng ban đầu. Giá trị của biến thể hiện phụ trợ được tích hợp vào started nhớ bất kỳ khi nào phương thức start của đối tượng đã được thi hành. Theo cú pháp, mỗi danh sách tham biến hình thức u trong chương trình gốc được mở rộng thành  callerthreadu ,, . Tương ứng với phương thức gọi, mỗi danh sách tham biến thực sự e trong các câu lệnh thi hành một phương thức khác với phương thức start được mở rộng thành   threadconfthisthreade ,,,, . Sự thi hành của phương thức start không có tham biến của đối tượng 0e lấy danh sách tham biến thực sự   threadconfthise ,,,0 . Sự quan sát của phương thức được gọi ở điểm bắt đầu của phương thức run thi hành started := true. Các biến conf và counter cũng được cập nhật. Bởi vì một luồng gọi phương thức start không đợi sự trả về của phương thức start nhưng tiếp tục thi hành, sự bổ sung và chú thích của sự thi hành phương thức start có dạng là        3!!201 . pstmpestartep callcall . 3.3.2. Kiểm chứng các điều kiện Tính đúng đắn ban đầu - 44 - Tính đúng đắn cục bộ chỉ thay đổi ở chỗ các tham biến hình thức thread và caller được gán các giá trị ban đầu  và  nullnull ,0, , trong đó  là đối tượng ban đầu. Định nghĩa 3.3.1. (Tính đúng đắn ban đầu) Cho thân của phương thức run của lớp chính c là     returnstmpeyp callcall ;: 3 ? 22 ? 2  với biến cục bộ v không có các tham biến hình thức, cLVarz , và ObjectLVarz ' . Một phác thảo chứng minh là đúng đắn ban đầu, nếu G|   ''.' zznullzzzInitState     nullnullzvInitcallerthreadv ,0,,,:,,    zP2 , và G|   ''.' zznullzzzInitState       zEyznullnullzvInitcallerthreadv 22 :.;,0,,,:,,      zIzPGI c 3 . Khẳng định   '''. zznullzzzInitState  phát biểu rằng trạng thái toàn cục ban đầu định nghĩa một đối tượng z đang tồn tại trong trạng thái thể hiện ban đầu của nó, và quan sát 22 : ey  ở phần bắt đầu của phương thức run của đối tượng ban đầu z được thể hiện bởi phép gán  zEyz 22 :.  . Sự khác nhau là ở trong sự khởi tạo của cấu hình cục bộ, cấu hình cục bộ bây giờ được biểu diễn bởi phép gán    0,0,,,:,, nullthreadvInitcallerthreadv  . Kiểm tra tính không có can thiệp Sự can thiệp của một luồng duy nhất dưới sự thi hành của nó giống như trong ngôn ngữ tuần tự. Tuy nhiên, ta phải giải quyết với sự bất biến của các thuộc tính của một luồng dưới sự thi hành của một luồng khác. Một khẳng định q tại một điểm điều khiển phải là bất biến dưới một phép gán ey : trong cùng một lớp nếu cấu hình cục bộ được mô tả bởi khẳng định không hoạt động trong bước tính toán đang thi hành phép gán. Để phân biệt các biến cục bộ của các cấu hình cục bộ khác nhau, ta đặt lại tên của các biến cục bộ trong khẳng định mà phải là bất biến, có kết quả là các biến, các biểu thức, và các khẳng định được đánh - 45 - dấu phẩy. Ví dụ, trong các điều kiện ta sử dụng thread để nhận dạng luồng đang thi hành phép gán, và thread’ để nhận dạng luồng được mô tả bởi q. Nếu q và ey : thuộc về cùng một luồng, có nghĩa là , thead’ = thread, thì ta có cùng mệnh đề như đối với ngôn ngữ tuần tự. Nếu khẳng định và phép gán thuộc về các luồng khác nhau, tính không có can thiệp phải được biểu diễn trong tất cả các trường hợp trừ sự thi hành chính nó của phương thức start: Quan sát được gọi của một sự thi hành chính nó của phương thức start không thể can thiệp với điều kiện trước của sự thi hành. Để mô tả sự thiết lập này, ta định nghĩa  eyqstartself :,_ bằng  ',', threadconfthiscaller  khi và chỉ khi q là điều kiện trước của sự thi hành phương thức  estarte .0 và phép gán là quan sát được gọi tại phần bắt đầu của phương thức run, và bằng false nếu ngược lại. Định nghĩa 3.3.2. (Tính không có can thiệp) Một phác thảo chứng minh là không có can thiệp, nếu các điều kiện của định nghĩa 2.4.1 đúng với  eyqretforwaits :,__ được thay thế bằng   defe :yq,interferes thread = thread’   eyqretforwaits :,__  eyqstartselfthreadthread  :,_' Ví dụ 3.3.3. Giả thiết một lệnh gán được chú thích  stmp trong một phương thức, và một khẳng định q tại một điểm điều khiển không đợi sự trả về trong cùng một phương thức, sao cho p và q kéo theo thread = this. Điều này có nghĩa là, phương thức được thì hành chỉ bởi luồng của đối tượng mà phương thức đó thuộc về. Rõ ràng, p và q không thể đồng thời đạt tới được bằng cùng một luồng. Đối với tính bất biến của q dưới phép gán stm, mệnh đề của điều kiện không có can thiệp kéo theo  stmqqp ,interferes' . Từ 'qp  ta kết luận thread = thread’, và do đó bằng định nghĩa của  e:yq,interferes khẳng định q là một điểm điều khiển chờ trả về, nhưng trong trường hợp này thì không đúng, và do đó mệnh đề của điều kiện ước lượng là sai. Kiểm tra sự hợp tác Kiểm tra sự hợp tác đối với tạo đối tượng không bị ảnh hưởng bởi việc thêm vào tính tương tranh. Sự thi hành của các phương thức khác với phương thức start, được thi hành bởi một luồng duy nhất, không bị ảnh hưởng bởi sự có mặt của tính tương tranh. z và z’ là các biến logic mới đại diện cho đối tượng gọi và đối tượng được gọi. Bên - 46 - cạnh điều kiện trước của lời gọi, các bất biến toàn cục, bất biến lớp, ta giả sử rằng sự thi hành được cho phép, có nghĩa là, luồng của đối tượng được gọi chưa được bắt đầu, được định nghĩa bằng startedz'. . Định nghĩa 3.3.5. (Kiểm tra sự hợp tác: Giao tiếp) Một phác thảo chứng minh thỏa mãn kiểm tra sự hợp tác đối với sự giao tiếp, nếu các điều kiện của định nghĩa 2.4.14 đúng đối với các lệnh được liệt kê ở đó với startm  , và thêm vào các trường hợp sau: 1. startCALL : Đối với tất cả các lệnh       3 ! 11 ! 201 :. peypestartep callcall  trong lớp c với 0e của kiểu c’, comm được cho bằng   startedzzzE '.'0  , trong đó     returnstmqeyq callcall ;: 3 ? 22 ? 2  là thân của của phương thức run của c’ có các tham biến hình thức u , và các biến cục bộ v trừ các tham biến hình thức. Bất biến của lớp được gọi là '1 cIq  . Hơn nữa, commf là    vInitzEvu ,:','  , 1obsf là  zEyz 11 :.  , và 2obsf là  '':''. 22 zEyz  . 2. skipstartCALL : Đối với các lệnh phía trên, các phương trình phải đúng với khẳng định comm được cho bằng   trueqqstartedzzzE  320 ,'.' , 1q và 1obsf như phần trên, và commf và 2obsf là lệnh rỗng. - 47 - CHƯƠNG 4. BỘ ĐIỀU PHỐI LẶP LẠI 4.1. Cú pháp Các phương thức được bao bọc bởi một từ bổ nghĩa modif phân biệt giữa các phương thức đồng bộ và không đồng bộ. Trong phần tiếp theo ta tham chiếu tới các lệnh trong thân của một phương thức đã đồng bộ như là đang được đồng bộ. Do vậy, ta xem các phương thức thêm vào được định nghĩa trước là wait, notify, và notifyAll modif ::= nsyn | sync meth ::= modif m (u, …, u) { stm; return retexp } runmeth ::= nsync run () { stm; return} waitmeth ::= nsync wait () { ?signal; getlockreturn } notifymeth ::= nsync notify () {!signal; return } notifyAllmeth ::= nsync notifyAll () { !signal_all; return } predefmeth ::= notifyAllnotifywaitstart methmethmethmeth class ::= class c { meth … meth predefrun methmeth } mainclass ::= class prog ::= class … class mainclass Các định nghĩa của cú pháp sử dụng các câu lệnh bổ trợ !sinal, !signal_all, ?signal, và getlockreturn mô tả ở mức cao của kỹ thuật trừu tượng hóa signal-and- continue dưới các phương thức wait, notify, và notifyAll. 4.2. Ngữ nghĩa Các ngữ nghĩa toán tử mở rộng các ngữ nghĩa của concJava bằng các quy tắc sau, trong đó quy tắc CALL được thay thế. Đối với các lời gọi phương thức được đồng bộ hóa, khóa của đối tượng được gọi phải là tự do hoặc được sở hữu bởi luồng đang thi hành, do vậy được biểu thị bởi vị từ owns, được định nghĩa bên dưới. - 48 - Các quy tắc còn lại quản lý các ngữ nghĩa của các phương thức điều phối wait, notify, và notifyAll. Trong cả ba trường hợp luồng gọi phải sở hữu khóa của đối tượng được gọi (quy tắc monitorCALL ). Một luồng có thể khóa chính nó trong một đối tượng mà nó sở hữu khóa bằng cách thi hành phương thức wait của đối tượng, do đó nó giải phóng khóa và đưa chính nó vào trong tập đợi của đối tượng. Tập đợi  ,Twait của một đối tượng được cho như là một tập của tất cả các ngăn xếp trong T với một phần tử ở đỉnh ngăn xếp có dạng  stmsignal;?,, . Sau khi đóng băng chính nó, luồng đợi thông báo từ luồng thi hành phương thức notify của đối tượng. Câu lệnh !signal trong phương thức notify do đó kích hoạt lại một luồng duy nhất được chọn đang chờ thông báo trong đối tượng được cho (quy tắc SIGNAL). Tương tự như tập chờ, tập được thông báo  ,Tnotified của đối tượng  biểu thị tập tất cả các ngăn xếp trong T với phần tử ở đỉnh ngăn xếp có dạng  getlockreturn,, , nghĩa là, các luồng được thông báo và cố gắng giữ khóa lại. Theo quy tắc waitRETURN , luồng được thông báo chỉ có thể tiếp tục thi hành getlockreturn sau thông báo nếu khóa là tự do. Chú ý rằng luồng thông báo không được bàn giao lại khóa cho một luồng đang được thông báo nhưng tiếp tục sở hữu nó. Hành vi này được biết đến như là quy tắc điều phối signal-and-continue. Các quy tắc:      cMethbodyumnotifyAllnotifywaitrunstartm  modif,,,,               ,,,0 ' euVale cminitc          CALLstmemeuT Towns     ,;.:,, ,syncmodif 0      ,,',;,, bodystmureceiveT   notifyAllnotifywaitm ,,                monitor c CALL stmmeT stmmeownsVale      ,;.,, ,;.,,,        ,,,;,, ,, cmcminit bodystmreceiveT  - 49 -        waitgetlock RETURN returnstmreceiveT Towns     ,,',;,, ,     ,,, stmT        SIGNAL stmsignalstmsignalT  ,';?,',';!,,        ,',',',, stmstmT          skip SIGNAL stmTstmsignalT Twait   ,,,,;!,, ,            SIGNALALL stmTstmallsignalT TsignalT   ,,,',;_!,, ,'    Nếu không có luồng nào đang đợi trong đối tượng, lệnh !signal của luồng thông báo không có kết quả (quy tắc skipSIGNAL ). Phương thức notifyAll tổng quát phương thức notify ở chỗ thông báo tất cả các luồng đang đợi được thông báo qua sự quảng bá !signal_all (quy tắc SIGNALALL). Kết quả của câu lệnh này được cho bởi định nghĩa  ,Tsignal là           ,;?,,|,,,\ TwaitstmsignalstmTwaitT  . Bằng cách sử dụng các tập đợi và tập được thông báo, ta có thể hình thức hóa vị từ owns: Một luồng  sở hữu khóa của đối tượng  khi và chỉ khi  thi hành phương thức đồng bộ của đối tượng  , nhưng không phải phương thức wait của nó. Vị từ owns (T,  ) là đúng khi và chỉ khi tồn tại một luồng T và một cấu hình    stm,, với stm được đồng bộ hóa và     ,, TnotifiedTwait  . 4.3. Hệ chứng minh 4.3.1. Phác thảo chứng minh Để nắm bắt sự loại trừ lẫn nhau và quy tắc điều phối, biến thể hiện được tích hợp thêm lock có kiểu IntThread  lưu trữ định danh của luồng sở hữu khóa, nếu có, cùng với số của các lời gọi đồng bộ trong chuỗi lời gọi của luồng sở hữu khóa. Giá trị ban đầu - 50 - của khóa 0) (null, free  biểu thị rằng khóa tự do. Các biến thể hiện wait và notified của kiểu  IntThread list là tương đương với các tập wait và notified của các ngữ nghĩa và lưu trữ các luồng đang đợi ở một bộ điều phối, theo thứ tự các luồng được thông báo. Bên cạnh định danh luồng, số của các lời gọi đồng bộ cũng được lưu trữ. Nói cách khác, các biến này nhớ giá trị của khóa cũ trước khi tạm thời ngừng, giá trị này được trả lại khi luồng hoạt động trở lại. Tất cả các biến bổ trợ được khởi tạo một cách bình thường. Đối với các giá trị thread của kiểu Thread và wait của kiểu list ( IntThread  ), ta sẽ viết waitthread  thay vì   waitnthread , với các giá trị n. Theo cú pháp, bên cạnh bổ sung được tích hợp của chương trước, quan sát được gọi ở điểm đầu và điểm cuối của mỗi thân phương thức được đồng bộ hóa thi hành theo thứ tự  lockinclock : và  lockdeclock : . Các ngữ nghĩa của sự tăng khóa      ,instlockinc là   1, nthread đối với    nvlockinst , . Chú ý rằng định danh của luồng sở hữu khóa không những trong trường hợp khóa tự do mà còn trong trường hợp nếu có một luồng đang sở hữu khóa. Tuy nhiên, bởi vì những cập nhật này được thi hành trong các phương thức được đồng bộ, nên các ngữ nghĩa đảm bảo cho trường hợp có một luồng đang sở hữu khóa, luồng sở hữu khóa là luồng đang thực thi, nghĩa là, nếu khóa không tự do thì thành phần luồng của khóa không được sửa đổi. Điều này có nghĩa là, tăng giá trị khóa  n, tạo ra  1, n , trong khi đó tăng một khóa tự do  0,null bởi một luồng  có kết quả là  1, . Giảm khóa  lockdec được thực hiện ngược lại:      ,instlockdec với    nlockinst ,  là  1, n nếu n > 1, và tự do nếu ngược lại. Thay vì các câu lệnh phụ trợ của các ngữ nghĩa, thông báo được thể hiện trong hệ chứng minh dựa trên trạng thái bởi các phép gán phụ trợ trong các biến wait và notified: Các câu lệnh phụ trợ !signal và !signal_all được thay thế bởi các phép gán phụ trợ. Các câu lệnh phụ trợ ?signal không được thể hiện. Điều này có nghĩa là, thông báo được thể hiện bởi duy nhất một phép gán phụ trợ được thi hành bởi luồng đưa ra thông báo. Đối với các luồng đang được thông báo, các điểm điều khiển trước và sau thông báo được mô tả bởi duy nhất một khẳng định trong phương thức wait. Các điểm điều khiển khác có thể được phân biệt bởi các giá trị của các biến phụ trợ tích hợp wait và notified. Vào phương thức wait nhận được quan sát   freelockwaitlockwait ,:,  trả về từ phương thức wait quan sát - 51 -     threadnotifiedgetnotifiedthreadnotifiedgetnotifiedlock ,\,,:,  . Đối với một luồng ThreadVal và một danh sách  IntthreadlistValnotified  ,  ,notifiedget lấy giá trị  n, từ danh sách. Các ngữ nghĩa đảm bảo tính duy nhất của sự kết hợp. Câu lệnh signal! của phương thức notify được thay thế bằng lệnh gán bội phụ trợ  notifiedwaitnotifynotifiedwait ,:,  , trong đó giá trị  notifiedwaitnotify , là một cặp của các tập được cho với một phần tử được chọn, được chuyển từ tập tập wait sang tập notified; nếu tập wait là rỗng, nó là định danh hàm. Cuối cùng, câu lệnh allsignal _! của phương thức notifyAll được thay thế bởi phép gán phụ trợ  ,:, waitnotifiedwaitnotified  . 4.3.2. Kiểm chứng các điều kiện Tính đúng đắn cục bộ và tính đúng đắn ban đầu thì như là đối với concJava . Đối với tính đúng đắn cục bộ, chú ý rằng các điều kiện bao hàm thêm vào tính bất biến đối với các luồng đang thi hành thông báo. Tuy nhiên, ta không cần thêm vào các điều kiện cho trường hợp này, bởi vì kết quả của thông báo được giữ lại bởi một phép gán phụ trợ. Đối với các luồng đang được thông báo, các điểm điều khiển trước và sau thông báo được mô tả bởi một khẳng định. Kiểm tra tính không có can thiệp Các phương thức được đồng bộ của một đối tượng duy nhất chỉ có thể được thi hành tương tranh nếu một trong các cấu hình cục bộ tương ứng đang chờ sự trả về: Nếu các luồng đang thi hành là khác nhau, thì một trong các luồng thi hành trong phương thức không được đồng bộ hóa wait của đối tượng; mặt khác, cả hai cấu hình cục bộ đang thi hành là trong cùng một dãy lời gọi. Định nghĩa 4.3.1. (Tính không có can thiệp) Một phác thảo chứng minh là không có can thiệp, nếu các điều kiện của định nghĩa 3.3.2 đúng đối với tất cả các lớp c, tất cả các phép gán bội ey : với điều kiện p trong c, và tất cả các khẳng định q tại các điểm điều khiển trong c, sao cho hoặc là cả hai p và q không xuất hiện trong một phương thức đồng bộ, hoặc là q là một điểm điều khiển đang chờ sự trả về. Chú ý rằng đối với thông báo, ta cũng yêu cầu tính bất biến của các khẳng định của các luồng đang đợi thông báo. Thông báo được mô tả bằng một phép gán phụ trợ được thi hành bởi luồng thông báo. Điều đó có nghĩa là, cả hai tình trạng đang đợi và tình trạng được thông báo của một luồng bị tạm thời ngừng được thay thế bằng một điểm điều khiển duy nhất trong phương thức wait. Hai tình trạng có thể được phân biệt - 52 - bằng các giá trị của các biến wait và biến notified. Tính bất biến của điều kiện trước của lệnh trả về trong phương thức wait dưới phép gán trong phương thức notify biểu thị quá trình thông báo, trong khi đó tính bất biến của khẳng định thông qua các phép gán thay đổi khóa biểu thị kỹ thuật đồng bộ hóa. Thông tin về giá trị khóa được khai báo từ kiểm tra sự hợp tác bởi vì thông tin này phụ thuộc vào hành vi toàn cục. Ví dụ 4.3.2. Ví dụ này biểu diễn cách nhiều nhất một luồng có thể sở hữu khóa của một đối tượng có thể được sử dụng để biểu diễn sự loại trừ lẫn nhau. Ta sử dụng khẳng định  lockthreadowns , đối với   threadlockthreadnullthread  , trong đó  lockthread là thành phần đầu tiên của giá trị khóa. Cho  lockthreadforfree ,_ là   freelocklockthreadownsnullthread  , . Cho q, được cho bởi  lockthreadowns , , là một khẳng định tại một điểm điều khiển và cho   callcall stmp ?? với  lockthreadforfreep def ,_ là quan sát được gọi tại phần bắt đầu của một phương thức được đồng bộ hóa trong cùng lớp. Chú ý rằng quan sát stm thay đổi giá trị khóa. Điều kiện không có can thiệp     ',interferes'| qstmstmqqpL  đảm bảo tính bất biến của q dưới quan sát stm. Các khẳng định p và q’ kéo theo thread = thread’. Các điểm tại p và q chỉ có thể đạt tới được đồng thời bởi cùng một luồng nếu q mô tả một điểm đang chờ trả về. Nhưng điều đó là sai bởi định nghĩa của vị từ interferes: Nếu q không ở một điểm điều khiển đang chờ trả về, thì mệnh đề của điều kiện ước lượng là sai. Mặt khác, sau sự thi hành của lệnh phụ trợ tích hợp  lockinclock : trong stm ta có  lockthreadowns , , có nghĩa là  lockthreadowns ,' . Kiểm tra sự hợp tác Ta mở rộng kiểm tra sự hợp tác cho concJava với kỹ thuật đồng bộ hóa và với sự thi hành của các phương thức điều phối. Trong các ngôn ngữ trước, khẳng định comm phát biểu rằng các lệnh được cho đại diện các đối tác giao tiếp. Trong ngôn ngữ hiện tại với sự đồng bộ hóa điều phối, giao tiếp luôn luôn được cho phép. Do vậy khẳng định comm phải giữ thêm tính được cho phép của giao tiếp: Trong trường hợp thi hành một phương thức được đồng bộ hóa, khóa của đối tượng phải là tự do hoặc thuộc sở hữu của luồng gọi. Điều này được biểu diễn bởi   threadlockzthreadfreelockz  '.'. , trong đó thread là luồng gọi, z’ là đối tượng được gọi, và trong đó  lockzthread '. là thành phần đầu tiên của giá trị khóa, có nghĩa là, luồng đang sở hữu khóa của z’. Đối với sự thi hành của các phương thức điều phối ta yêu cầu rằng luồng đang thi hành - 53 - đang giữ khóa. Sự trả về từ phương thức wait giả thiết rằng luồng được thông báo và khóa của đối tượng được gọi là tự do. Định nghĩa 4.3.3. (Kiểm tra sự hợp tác: Giao tiếp) Một phác thảo chứng minh thỏa mãn kiểm tra sự hợp tác đối với giao tiếp, nếu các điều kiện của định nghĩa 3.3.5 đúng đối với các lệnh được liệt kê ở đó với ngoại lệ của trường hợp quy tắc CALL, và thêm vào các trường hợp sau đây: 1.CALL: Sự thi hành của các phương thức không được đồng bộ hóa m với  notifyAllnotifywaitstartm ,,, được xử lý như trước đó. Đối với tất cả các lệnh       waitcallcallret peypemeup 3 ! 11 ! 201 :.:  (hoặc là các lệnh này mà không nhận một giá trị) trong lớp c với 0e của kiểu c’, trong đó phương thức  notifyAllnotifywaitstartm ,,, của c’ được đồng bộ hóa với thân     ret callcall ereturnstmqeyq ;: 3 ? 22 ? 2  , các biến hình thức u , và các biến cục bộ v ngoại trừ các biến hình thức, các điều kiện 2.6 và 2.7 phải đúng với các định nghía sau: Bất biến của lớp được gọi là '1 : cIq  . Khẳng định comm được cho bằng     threadlockzthreadfreelockzzzE  '.'.'0 . Hơn nữa, commf là    vInitzEvu ,:','  , 1obsf được cho bằng  zEyz 11 :.  , và 2obsf là  '':''. 22 zEyz  . 2. monitorCALL : Đối với  notifyAllnotifywaitm ,, , comm được cho bằng     threadlockzthreadzzE  '.'0 . 3. waitRETURN : Đối với      3 ! 33 ! 21 : qeyqreturnq retret getlock  trong phương thức wait, comm là     notifiedzthreadfreelockzzEuzzE '.''.''0  . Ví dụ 4.3.4. Giả thiết rằng sự thi hành của một phương thức được đồng bộ m của một lớp c, trong đó m của c có thân    returnstmthreadlockthreadstm call ;'?  . Chú ý rằng sự mở rộng được tích hợp trong stm thiết lập luồng sở hữu khóa bằng phép gán  lockinclock : . Kiểm tra sự hợp tác yêu cầu rằng       ''.'.:'.| threadlockzthreadlockzinclockztrueG  là đúng với định nghĩa của inc. - 54 - CHƯƠNG 5. PHÉP TOÁN ĐIỀU KIỆN TRƯỚC YẾU NHẤT Để tăng tính dễ đọc, kiểm chứng các điều kiện của các chương trước được đưa ra như là các bộ ba logic Hoare chuẩn. Mục đích của chúng ta là sử dụng một bộ chứng minh định lý để chứng minh các điều kiện này. Thay vì thi hành các ngữ nghĩa của các bộ ba logic Hoare trong bộ chứng minh định lý, ta đưa ra ngữ nghĩa trong các phép kéo theo logic thông qua một phép toán điều kiện trước yếu nhất. Theo cách này ta chỉ phải thi hành các ngữ nghĩa của các khẳng định trong bộ chứng minh định lý. 5.1. Các phép toán thay thế Sự thay thế cục bộ  yep / thay thế trong khẳng định cục bộ p tất cả các sự xuất hiện của các biến khác biệt y bằng các biểu thức cục bộ e . Ta cũng áp dụng sự thay thế cho các biểu thức cục bộ. Bổ đề sau biểu diễn thuộc tính chuẩn của sự thay thế ở trên, liên kết nó với cập nhật trạng thái. Quan hệ giữa sự thay thế và cập nhật được đưa ra trong bổ đề khẳng định rằng  yep / là điều kiện trước yếu nhất của p đối với phép gán ey : . Bổ đề được đưa ra cho các khẳng định, nhưng cùng thuộc tính đúng cho các biểu thức. Bổ đề 5.1.1. (Sự thay thế cục bộ) Đối với các môi trường logic  bất kỳ và các trạng thái thể hiện cục bộ   ,inst ta có  yepLinst /|,,  khi và chỉ khi         peyey LLLinst instinst |,, ,,,,    Sự kết quả của các phép gán được biểu diễn trong mức toàn cục bởi sự thay thế toàn cục  ezEP ./ , thay thế trong khẳng định toàn cục P các biến thể hiện x của đối tượng được tham chiếu tới bởi z bằng các biểu thức toàn cục E . Để có thể phù hợp với kết quả của các phép gán, ta phải không những thay thế theo cú pháp các xuất hiện của ixz. của các biến thể hiện, mà còn cho tất cả các bí danh của nó ixE '. , khi z và kết quả của sự thay thế được áp dụng cho E’ tham chiếu tới cùng đối tượng. Bởi vì điều kiện định danh không thể được kiểm tra theo cú pháp, ta định nghĩa trường hợp chính của sự thay thế bằng một biểu thức điều kiện      zxzEEifxzExE i  ./'./'. then iE else   ixxzEE ../' fi - 55 - Sự thay thế này được mở rộng cho các khẳng định toàn cục. Ta sử dụng sự thay thế  xzEP ./ cho dãy các biến bất kỳ y có thể chứa các biến logic, mà các ngữ nghĩa của chúng được định nghĩa bởi sự thay thế đồng thời  xzEx ./ và  uEu / , trong đó x và u là các dãy của các biến thể hiện và logic của y , các dãy con tương ứng xE và uE của E và  uEu / là sự thay thế như trong thay thế cục bộ; nếu chỉ các biến logic được thay thế, ta viết đơn giản  uEP / . Bổ đề 5.1.2. (Thay thế toàn cục) Đối với các trạng thái toàn cục  bất kỳ và các môi trường logic  chỉ tham chiếu tới các giá trị đang tồn tại trong  ta có  yzEPG ./|,  khi và chỉ khi PG|','  , trong đó     ,' GEy và        ,, .' GG Eyz  5.2. Kiểm chứng các điều kiện Trong kiểm chứng các điều kiện cục bộ, kết quả của một phép gán ey : được biểu diễn bởi sự thay thế e cho y trong các khẳng định. Trong các điều kiện toàn cục của kiểm tra sự hợp tác, kết quả của giao tiếp, chỉ thay đổi các trạng thái cục bộ, được biểu diễn bởi thay thế đồng thời các biến sẽ lưu trữ kết quả bằng các giá trị được giao tiếp. Điều đó có nghĩa là, đối với trường hợp lời gọi phương thức, các tham biến hình thức được thay thế bởi các tham biến thực được biểu diễn trong ngôn ngữ toàn cục. Kết quả của quan sát gọi call ey ! : trên một khẳng định toàn cục P được biểu diễn bằng sự thay thế   yzzEP ./ , trong đó z biểu diễn đối tượng gọi. Hiệu quả của quan sát được gọi được quản lý tương tự. Chú ý rằng thứ tự: đầu tiên giao tiếp diễn ra, được tiếp theo bởi quan sát gửi và sau đó là quan sát nhận. Để mô tả hiệu quả tách rời, đầu tiên ta phải thay thế đối với quan sát nhận, sau đó đối với quan sát nhận, và cuối cùng là đối với giao tiếp. Đối với một lời gọi phương thức, ta phải thêm vào thay thế đối với sự khởi tạo của các biến cục bộ. Để cho dễ đọc, trong các định nghĩa sau ta sử dụng ký hiệu fp với  yef / đối với sự thay thế  yep / ; ta sử dụng một ký hiệu tương tự cho các thay thế toàn cục. Chú ý rằng toán tử thay thế kết nối mạnh hơn các toán tử logic. - 56 - Định nghĩa 5.2.1. (Tính đúng đắn ban đầu) Một phác thảo chứng minh là đúng đắn ban đầu, nếu      ''.'| zznullzzzInitStateG        initobscinit ffzIzPGIfzP   32 . , trong đó c là lớp chính,     returnstmpeyp callcall ;: 3 ? 22 ? 2  là thân và các biến cục bộ v của phương thức run của c, cLVarz , và ObjectLVarz ' . Khẳng định toàn cục InitState được định nghĩa trước. Hơn nữa,      vvInitcallerthreadnullnullzf init /,/,0,, , và   22 ./ yzzEf obs  Định nghĩa 5.2.2. (Tính đúng đắn cục bộ: Phép gán) Một phác thảo chứng minh là đúng đắn cục bộ, nếu với tất cả các phép gán bội    21 : peyp  trong lớp c, không phải là quan sát của giao tiếp hoặc tạo đối tượng, asscL fpIp 21|  , với  yefass / . Định nghĩa 5.2.3. (Không có can thiệp) Một phác thảo chứng minh là không có can thiệp, nếu với tất cả các lớp c, và với tất cả các phép gán bội ey : với điều kiện trước p trong c, assccL fIIp | , với  yef ass / . Hơn nữa, đối với tất cả các khẳng định q tại các điểm điều khiển trong c, sao cho hoặc cả p và q không xuất hiện trong một phương thức được đồng bộ, hoặc q ở một điểm điều khiển đang chờ trả về,   asscL fqeIqp ':yq,interferes'|  . với khẳng định interferces như được định nghĩa ở phần trên. Định nghĩa 5.2.4. (Kiểm tra sự hợp tác: Giao tiếp) Một phác thảo chứng minh thỏa mãn kiểm tra sự hợp tác đối với giao tiếp, nếu - 57 -          nullznullzcommzIzQzIzPGI ccG ''''| '11       commfzQzP ''22      commobsobs fffzQzPGI  1233 '' (5.5) là đúng đối với các biến logic mới khác biệt cLVarz và '' cLVarz  , trong các trường hợp sau: 1. (a) CALL: Đối với tất cả các lời gọi        waitcallcallret peypemeup 3 ! 11 ! 201 :.:  (hoặc các lời gọi không nhận một giá trị) trong lớp c với 0e thuộc kiểu 'c , trong đó phương thức  notifyAllnotifywaitstartm ,,, của c’ được đồng bộ hóa với thân     ret callcall ereturnstmqeyq ;: 3 ? 22 ? 2  , các tham biến hình thức u , và các biến cục bộ v trừ các tham biến hình thức. Bất biến của lớp được gọi là '1 cIq  . Khẳng định comm được cho bằng     threadlockzthreadfreelockzzzE  '.'.'0 . Hơn nữa,     ','/, vuvInitzEfcomm ,   111 ./ yzzEfobs  ,   222 './'' yzzEfobs  . Nếu m không được đồng bộ hóa,   threadlockzthreadfreelockz  '.'. trong comm được loại bỏ. (b) monitorCALL : Với  notifyAllnotifywaitm ,, , comm được cho bằng     threadlockzthreadzzE  '.'0 . (c) startCALL : Với m = start, comm là   startedzzzE '.'0  , trong đó     returnstmqeyq callcall ;: 3 ? 22 ? 2  là thân của phương thức run của c’. (d) skipstartCALL : Với m = start, thêm vào, (5.5) phải là đúng với comm được cho bằng   startedzzzE '.'0  , trueqq  32 , và commf và 2obsf là định danh các hàm. 2. (a) RETURN : Đối với tất các lệnh gọi phương thức        3 ? 44 ? 21 ! 110 ::.: peyppeyemeu retretwaitcall ret  (hoặc các lệnh này mà không nhận một giá trị) xuất hiện trong c với 0e của kiểu c’, sao cho - 58 - phương thức  um của c’ có lệnh trả về      3 ! 33 ! 21 : qeyqereturnq retret ret , phương trình (5.5) phải là đúng với comm được cho bằng    zEuzzE  ''0 , và trong đó   retretcomm uzEf /'' ,   331 ''./'' yzzEfobs  , và   442 ./ yzzEfobs  . (b) waitRETURN : Với      3 ! 33 ! 21 : qeyqreturnq retret getlock  trong phương thức wait, comm là     notifiedzthreadfreelockzzEuzzE '.''.''0  . (c) runRETURN : Với      3 ! 33 ! 21 : qeyqreturnq retret  xuất hiện trong phương thức run, trueppp  321 , comm = true, và hơn nữa commf và 2obsf là định danh các hàm. Định nghĩa 5.2.5. (Kiểm tra sự hợp tác: Tạo đối tượng) Một phác thảo chứng minh thỏa mãn kiểm tra sự hợp tác đối với tạo đối tượng, nếu với tất cả các lớp c’ và các lệnh      321 :: peypnewup newnewc  trong c’:          '.,''.| '1 zzIzPuGIuzFreshzuznullz cG        obsfzPGIuzP 3c2 I  , với 'cLVarz và listObjectLVarz ' là đối tượng mới,   yzzEfobs ./ , và Fresh và  được định nghĩa ở trong 2.4.2. - 59 - CHƯƠNG 6. TÍNH ĐÚNG ĐẮN Một chương trình được cho cùng với chú thích của nó, hệ chứng minh quy định một số kiểm chứng các điều kiện đối với các kiểu khác nhau của các khẳng định và các cấu trúc chương trình. Tính đúng đắn của hệ chứng minh có nghĩa là đối với một phác thảo chứng minh thỏa mãn kiểm chứng các điều kiện, tất cả các cấu hình có thể đạt được trong các ngữ nghĩa của toán tử thỏa mãn các khẳng định được cho. Để thuận tiện, ta giới thiệu các chú thích sau: Một chương trình được cho prog, ta viết prog hoặc  cho chú thích của nó, và viết |prog , nếu prog thỏa mãn tất cả các yêu cầu được phát biểu trong các khẳng định, và '|' prog , nếu prog’ với chú thích ' thỏa mãn kiểm chứng các điều kiện của hệ chứng minh: Định nghĩa 6.0.1. Một chương trình được cho prog với chú thích  , thì |prog khi và chỉ khi với tất cả các cấu hình có thể đạt được ,T của prog, với tất cả   Tstm ,,  , và với tất cả các môi trường logic  chỉ tham chiếu tới các giá trị đang tồn tại trong  : 1.    stmpreL|,,  , và 2. GIG|, . Hơn nữa, với tất cả các lớp c, các đối tượng   cVal , và các trạng thái cục bộ ' : 3.   cL I|',,  . Đối với các phác thảo chứng minh, ta viết '|' prog khi và chỉ khi prog’ với chú thích ' thỏa mãn kiểm chứng các điều kiện của hệ chứng minh. 6.1. Tính đúng đắn Tính đúng đắn có nghĩa là tất cả các cấu hình có thể đạt được thỏa mãn các khẳng định của chúng đối với một chương trình được chú thích – được kiểm chứng bằng sử dụng chứng minh các điều kiện. Tính đúng đắn của một phương thức được chứng minh bằng một quy nạp đơn giản trong số của các bước tính toán. - 60 - Trước khi bắt tay vào trình bày tính đúng đắn và chứng minh của nó, ta cần hiểu rõ mối quan hệ giữa chương trình gốc và phác thảo chứng minh, có nghĩa là, chương trình gốc được mở rộng bằng các biến phụ trợ, và được bao bọc với các khẳng định. Để tạo sự rõ ràng trong mối quan hệ giữa chương trình gốc và phác thảo chứng minh, ta định nghĩa một toán tử chiếu prog , xóa tất cả các phần thêm vào của sự biến đổi. Cho prog’ là một phác thảo chứng minh với prog, và ',' T là một cấu hình toàn cục của prog’. Thì prog' được định nghĩa bởi bỏ đi tất cả các biến thể hiện phụ trợ từ các miền trạng thái thể hiện. Với tập các cấu hình luồng, progT ' được cho bởi hạn chế các miền của các trạng thái cục bộ để không có các biến phụ trợ và bỏ đi tất cả các bổ sung. Thêm vào đó, với các cấu hình cục bộ   ',, ! Tstmreturn retgetlock  , nếu luồng đang thi hành ở trong tập đợi, nghĩa là, nếu      waitnthread  ',  với một số n nào đó, thì lệnh getlockreturn được thay thế bằng getlockreturnsignal;? . Hơn nữa, với các cấu hình cục bộ   '';,, ! Tstmreturnstm ret  với stm một phép gán phụ trợ trong phương thức notify hoặc trong phương thức notifyAll, phép gán phụ trợ stm được thay thế theo thứ tự bằng !signal và !signal_all. Bổ đề sau biểu thị rằng nếu sự biến đổi không thay đổi hành vi của các chương trình: Bổ đề 6.1.1. Cho prog’ là một phác thảo chứng minh với một chương trình prog. Thì ,T là một cấu hình có thể đạt được của prog khi và chỉ khi tồn tại một cấu hình có thể đạt được ',' T của prog’ với  ,',' TprogprogT  . Bổ đề 6.1.2. (Định danh) Cho ,T là một cấu hình có thể đạt được của một phác thảo chứng minh. Thì: 1. với tất cả các ngăn xếp  và ' trong T và với tất cả các cấu hình cục bộ    stm,, và   '',','  stm ta có    threadthread '  khi và chỉ khi '  , và 2. với mỗi ngăn xếp    nnn stmstm ,,...,, 000  trong T và các chỉ số nji  ,0 , (a)   0 threadi ; (b) i < j và ji   kéo theo       counterconfconf iji   , (c) 0 < j kéo theo       threadconfcaller jjjj 111 ,,   , và (d)     threadcallerproj 00 3,   , - 61 - trong đó  ivproj , là thành phần thứ i của bộ v. Bổ đề sau phát biểu rằng luồng sở hữu khóa, và các tập đợi và tập được thông báo của các ngữ nghĩa được trình bày bởi các biến lock, wait, và notified. Hơn nữa, bổ đề đảm bảo rằng tính phân đoạn của các dãy được lưu trữ trong các biến wait và notified; nếu thứ tự của các phần tử không quan trọng, ta sử dụng tập ký hiệu cho các giá trị của chúng. Bổ đề 6.1.3. (Lock, Wait, Notify) Cho ,T là một cấu hình có thể đạt được của một phác thảo chứng minh với chương trình gốc prog,   Val là một định danh đối tượng, và cho   Tstm  ',, 000   . Hơn nữa cho n là số của các thi hành phương được đồng bộ của  trong  , nghĩa là, n =   |.|,,| synchrstmstm   . Thì: 1. (a)  ,progTowns  khi và chỉ khi    freelock  (b)   ,progowns  khi và chỉ khi     nlock ,0  2. (a)   ,progTwait  khi và chỉ khi     waitn  ,0 (b)   ,progTnotified  khi và chỉ khi     notifiedn  ,0 (c)     1,iwaitproj  =     1,jwaitproj  kéo theo i = j (d)     1,inotifiedproj  =     1,jnoitifiedproj  kéo theo i = j (e) nếu     waitm  ,0 hoặc     notifiedm  ,0 thì m = n (f)      notifiedwait    trong đó  is là thành phần thứ i của dãy s. Cuối cùng, biến thể hiện phụ trợ started của một đối tượng lưu trữ nếu luồng của đối tượng đã được khởi động hoặc không: Bổ đề 6.1.4. (Started) Đối với tất cả các cấu hình có thể đạt được ,T của một phác thảo chứng minh cho một chương trình prog, và tất cả các đối tượng   Val , ta có  ,progTstarted  khi và chỉ khi   started . Cho prog là một chương trình với chú thích  , và prog’ một phác thảo chứng minh tương ứng với chú thích ' . Cho GI’ là bất biến toàn cục của ' , 'cI biểu thị các - 62 - bất biến lớp của nó, và với một khẳng định p của  cho p’ biểu thị khẳng định của ' được kết hợp với cùng điểm điều khiển. Ta viết '|  khi và chỉ khi GIGIG  '| , ccL II  '| đối với tất cả các lớp c, và '| ppL  , đối với tất cả các khẳng định p của  được kết hợp với các điểm điều khiển nào đó. Để cho các biến phụ trợ ngữ nghĩa, các kéo theo ở trên được ước lượng trong ngữ cảnh của các trạng thái của chương trình được bổ sung. Định lý sau phát biểu tính đúng đắn của phương pháp chứng minh. Định lý 6.1.5. (Tính đúng đắn) Cho prog’ là một phác thảo chứng minh với chú thích 'prog . Nếu '|' progprog  thì '|' progprog  Chứng minh tính đúng đắn bao gồm một chứng minh quy nạp trong chiều dài của tính toán, đồng thời trong tất cả ba phần từ định nghĩa 6.0.1. Đối với bước quy nạp, ta giả thiết rằng kiểm chứng các điều kiện được thỏa mãn và giả thiết một cấu hình có thể đạt được thỏa mãn chú thích. Ta tạo trường hợp khác biệt trong cú pháp của bước tính toán kế tiếp: Nếu bước tính toán thi hành một phép gán, thì ta sử dụng các điều kiện tính đúng đắn cục bộ để chứng minh tính quy nạp của sự thi hành các thuộc tính của cấu hình cục bộ, và kiểm tra tính không có can thiệp đối với tất cả các cấu hình cục bộ khác và các bất biến lớp khác. Đối với giao tiếp, tính bất biến đối với thi hành các đối tác và bất biến toàn cục được chứng tỏ bằng sử dụng kiểm tra sự hợp tác đối với giao tiếp. Giao tiếp với chính nó không ảnh hưởng trạng thái toàn cục; tính bất biến của các thuộc tính còn lại dưới các quan sát tương ứng được chứng tỏ với sự giúp đỡ của của kiểm tra tính không có can thiệp. Cuối cùng, đối với tạo đối tượng, tính bất biến đối với bất biến toàn cục, tạo cấu hình cục bộ, bất biến lớp của đối tượng được tạo được đảm bảo các điều kiện của kiểm tra hợp tác đối với tạo đối tượng; tất cả các thuộc tính khác được chứng tỏ là bất biến thông qua sử dụng kiểm tra tính không có can thiệp. Định lý 6.1.5 được đưa ra đối với tính có thể đạt được của các chương trình được bổ sung. Với sự giúp đỡ của bổ đề 6.1.1, ta có ngay lập tức : Hệ quả 6.1.6. Nếu '|' progprog  và progprog   '| , thì progprog | . - 63 - KẾT LUẬN Với sự phát triển không ngừng của nhu cầu tính toán, xử lý số liệu, khai phá thông tin thì lập trình đa luồng ngày càng quan trọng và là một phần không thể thiếu trong các hệ thống tính toán, cũng như trong các hệ thống ứng dụng. Sự quan trọng và cần thiết của việc kiểm chứng tính đúng của chương trình luôn được quan tâm đúng mức do vai trò to lớn của nó. Kiểm chứng tính đúng đắn của chương trình Java đa luồng không chỉ mang ý nghĩa về mặt lý thuyết của kiểm chứng mà nó thực sự có vai trò không thể thiếu trong thực tế, qua đó giúp ta tránh được những sai lầm, thiếu sót dù là rất nhỏ nhưng có thể dẫn tới những hậu quả thật khó lường. Trong các hệ thống đòi hỏi tính an toàn cao như các hệ thời gian thực, các hệ mô phỏng, hoặc các hệ thống giám sát vệ tinh, máy bay thì kiểm chứng tính đúng đắn của các chương trình mang một ý nghĩa vô cùng to lớn. Nó cho phép xác định được tính đúng đắn của chương trình từ các giả thiết và các khẳng định ban đầu mà không cần phải chứng thực điều đó khi chạy chương trình, mà nếu trong thực tế chương trình chạy sai gây ra hậu quả vô cùng nghiêm trọng. Phương pháp kiểm chứng trên còn được sử dụng để chứng minh tính không có deadlock của một chương trình Java. Phác thảo chứng minh của chương trình nếu thỏa mãn các điều kiện của chứng minh tính không có deadlock thì chương trình sẽ không có deadlock. Để có thể kiểm chứng tính đúng đắn của một chương trình Java đa luồng, ta có thể sử dụng phương pháp đã trình bày ở trên. Nó là cơ sở, tiền đề để xây dựng các công cụ hỗ trợ việc kiểm chứng tự động. Do thời gian thực hiện đề tài có hạn và phạm vi rộng lớn của đề tài, khóa luận chưa đề cập hết được các phương pháp cũng như kỹ thuật hiện đại kiểm chứng tính đúng đắn của chương trình, cũng như các công cụ hỗ trợ kiểm chứng hiện nay. - 64 - TÀI LIỆU THAM KHẢO [1] Erika Abraham - An Assertional Proof System for Multitheaded Java – Theory and Tool Support WEBSITE: [2] [3]

Các file đính kèm theo tài liệu này:

  • pdfLuận văn-Phương pháp kiểm chứng tính đúng đắn của một chương trình Java đa luồng thông qua sử dụng logic Hoare.pdf
Luận văn liên quan