在使用Runnable接口的时候,多线程要保证锁对象一致才能实现同步, synchronized代码块的锁对象是this,方法调用对象。synchronized方法的锁对象是方法打调用者。
- 在这个模拟多线程卖票的代码中,如果把while循环放在synchronized代码块之外,那么会出现卖负数票的情况。
public class MyRunnable implements Runnable {
private int tickets = 100;
public void run() {
//this指的就是run方法的调用者MyRunnable类对象,这几个线程是共享同一个对象
sellTickets();
}
//锁代码块,这才是正确的
public void sellTickets2() {
while (true) {
synchronized (this) {
if(tickets<1)
break;
System.out.println(Thread.currentThread().getName() + ":正在出售第" + tickets);
tickets--;
}
}
}
//错误示范,不会出现负数票,但是票都被同一个线程买走来。
public void sellTickets() {
synchronized (this) {
while (tickets>0) {
System.out.println(Thread.currentThread().getName() + ":正在出售第" + tickets);
tickets--;
}
}
}
//错误示范,会出现负数票,因为当tickets==0时,其他线程仍然是1,会进入循环。
public void sellTickets() {
while (tickets>0) {
synchronized (this) {
System.out.println(Thread.currentThread().getName() + ":正在出售第" + tickets);
tickets--;
}
}
}
}
总结:再多线程中,循环的时候,尽量使用while(ture){if()break;} 。
2. 再使用while(ture)循环的时候,如果没有break语句,循环下面写打代码都将出现编译错误。
3.String类中有个toCharArray()方法,可以得到字符串的字符数组。 CharAt(int i)方法返回第i索引上的字符。
4.map集合中的get(Object key)方法:如果键是不存在,那么返回的是null;如果该key所映射的value值为null时,也返回null。因此一般用containsKey(Object key)来判断是否包含某个键值对。