49 |
|
* read-only mode, and then lock. When in read-only mode, they |
50 |
|
* validate only at the end of an array scan unless the element is |
51 |
|
* actually used (for example, as an argument of method equals). |
52 |
+ |
* |
53 |
+ |
* We rely on some invariants that are always true, even for field |
54 |
+ |
* reads in read-only mode that have not yet been validated: |
55 |
+ |
* - array != null |
56 |
+ |
* - count >= 0 |
57 |
|
*/ |
58 |
|
|
59 |
|
/** |
653 |
|
long seq = lock.awaitAvailability(); |
654 |
|
int n = count; |
655 |
|
Object[] items = array; |
651 |
– |
if (n > items.length) |
652 |
– |
continue; |
653 |
– |
boolean outOfBounds = (index < 0 || index >= n); |
656 |
|
@SuppressWarnings("unchecked") |
657 |
< |
E e = outOfBounds ? null : (E) items[index]; |
657 |
> |
E e = (index < items.length) ? (E) items[index] : null; |
658 |
|
if (lock.getSequence() == seq) { |
659 |
< |
if (outOfBounds) |
659 |
> |
if (index >= n) |
660 |
|
throw new ArrayIndexOutOfBoundsException(index); |
661 |
< |
else |
660 |
< |
return e; |
661 |
> |
return e; |
662 |
|
} |
663 |
|
} |
664 |
|
} |
691 |
|
if (lock.getSequence() != seq) { |
692 |
|
lock.lock(); |
693 |
|
try { |
694 |
< |
return rawLastIndexOf(o, 0, count); |
694 |
> |
return rawLastIndexOf(o, count, 0); |
695 |
|
} finally { |
696 |
|
lock.unlock(); |
697 |
|
} |
875 |
|
for (;;) { |
876 |
|
long seq = lock.awaitAvailability(); |
877 |
|
Object[] items = array; |
877 |
– |
int len = items.length; |
878 |
|
int n = count; |
879 |
– |
if (n > len) |
880 |
– |
continue; |
881 |
– |
boolean empty = (n == 0); |
879 |
|
@SuppressWarnings("unchecked") |
880 |
< |
E e = empty ? null : (E) items[0]; |
880 |
> |
E e = (items.length > 0) ? (E) items[0] : null; |
881 |
|
if (lock.getSequence() == seq) { |
882 |
< |
if (empty) |
882 |
> |
if (n <= 0) |
883 |
|
throw new NoSuchElementException(); |
884 |
< |
else |
888 |
< |
return e; |
884 |
> |
return e; |
885 |
|
} |
886 |
|
} |
887 |
|
} |
892 |
|
for (;;) { |
893 |
|
long seq = lock.awaitAvailability(); |
894 |
|
Object[] items = array; |
899 |
– |
int len = items.length; |
895 |
|
int n = count; |
901 |
– |
if (n > len) |
902 |
– |
continue; |
903 |
– |
boolean empty = (n == 0); |
896 |
|
@SuppressWarnings("unchecked") |
897 |
< |
E e = empty ? null : (E) items[n - 1]; |
897 |
> |
E e = (n > 0 && items.length >= n) ? (E) items[n - 1] : null; |
898 |
|
if (lock.getSequence() == seq) { |
899 |
< |
if (empty) |
899 |
> |
if (n <= 0) |
900 |
|
throw new NoSuchElementException(); |
901 |
< |
else |
910 |
< |
return e; |
901 |
> |
return e; |
902 |
|
} |
903 |
|
} |
904 |
|
} |