<< Previous exercise (3.37) | Index | Next exercise (3.39) >>
a)
If no interleaving is possible the resulting values can be:
45: Peter +10; Paul -20; Mary /2
35: Peter +10; Mary /2; Paul -20
45: Paul -20; Peter +10; Mary /2
50: Paul -20; Mary /2; Peter +10
40: Mary /2; Peter +10; Paul -20
40: Mary /2; Paul -20; Peter +10
b) There are 9!/(3!3!3!)= 1680 possible timing diagrams.
The following procedure yields all possible values.
(define cnt 1) (define (execute-list lst) (display cnt) (display ":") (set! cnt (inc cnt)) (define (iter lst) (if (null? lst) (newline) (begin ((car lst)) (iter (cdr lst))))) (iter lst)) (define (factorial n) (if (= n 0) 1 (* n (factorial (dec n))))) (define balance 100) (define (make-person) (define mybalance 100) (define (access) (set! mybalance balance)) (define (deposit x) (set! mybalance (+ mybalance x))) (define (withdraw x) (set! mybalance (- mybalance x))) (define (withdraw-half) (set! mybalance (/ mybalance 2))) (define (sync) (set! balance mybalance)) (define (check) (display mybalance) (newline)) (define (dispatch m) (cond ((eq? m 'access) access) ((eq? m 'deposit) deposit) ((eq? m 'withdraw) withdraw) ((eq? m 'withdraw-half) withdraw-half) ((eq? m 'sync) sync) ((eq? m 'check) check))) dispatch) (define petter (make-person)) (define paul (make-person)) (define mary (make-person)) (define petter-seq (list (petter 'access) (lambda () ((petter 'deposit) 10)) (lambda () ((petter 'sync))))) (define paul-seq (list (paul 'access) (lambda () ((paul 'withdraw) 20)) (lambda () ((paul 'sync))))) (define mary-seq (list (mary 'access) (lambda () ((mary 'withdraw-half))) (lambda () ((mary 'sync))))) (define result '()) (define (interleave petter paul mary temp) (if (and (null? petter) (null? paul) (null? mary)) (set! result (cons (reverse temp) result))) (if (not (null? petter)) (interleave (cdr petter) paul mary (cons (car petter) temp))) (if (not (null? paul)) (interleave petter (cdr paul) mary (cons (car paul) temp))) (if (not (null? mary)) (interleave petter paul (cdr mary) (cons (car mary) temp)))) (interleave petter-seq paul-seq mary-seq '()) (define (in) (display balance) (set! balance 100)) (define op (map (lambda (x) (append x (list in))) result)) (for-each execute-list op)
It worths noting that 65 can appear as a result. Mary's operation is (balance - (balance / 2)), NOT (balance / 2).
; 1. Peter changes balance to 110. ; 2. Mary reads balance for the argument of subtraction, getting 110. ; 3. Paul changes balance to 90. ; 4. Mary reads balance for the argument of division, getting 90. ; 5. Mary sets balance to (110 - (90 / 2)) = 65.
The process described above may have a little problem.. Mary read balance as 110, so (balance / 2) = 55 and read balance again as 90, finally ans = 90 - 55 = 45
A solution without using complicated code.
;the list is: ;40, 35, 45, 50. (enumerate all possibilities.) ;some other values may occur if processes are interleafed are: ;referencing the timing diagram in the text: ;balance can be first set to 110, 80, 50, (overrides the set-balance!) ;then a third process can still occur. ;from here onwards, 110-20=90, or 110/2=55. ;(1 set-balance! overidden, then we update balance accordingly to the third process) ;80+10=90. 80/2=40. ;50+10=60, 50-20=30. ;if 2 set-balance! are overidden then 110, 80, 50 is reached. ;thus, the other values are: 110, 80, 90, 55, 60, 30.
The balance can also be 25 or 70: To get 25: ========== Peter Paul Mary Read 100 Read 100 Write 110 Read 110 Write 45 Read 45 Write 25 To get 70: ========== Peter Paul Mary Read 100 Read 100 Write 80 Read 80 Write 60 Read 60 Write 70
I think it's 90 possibilities.