/*
-
出现线程安全问题的根本原因:
- 存在两个或者两个以上 的线程对象,而且线程之间共享着一个资源。
- 有多个语句操作了共享数据。
同步代码块
同步代码块的格式: synchronized(锁对象){ 需要被同步的代码... }
同步代码块要注意事项:
1. 任意的一个对象都可以做为锁对象。
2. 在同步代码块中调用了sleep方法并不是释放锁对象的。
3. 只有真正存在线程安全问题的时候才使用同步代码块,否则会降低效率的。
4. 多线程操作的锁 对象必须 是唯一共享 的。否则无效。
-
/
/
*自定义线程 的实现方式:方式一 :
1. 自定义一个类继承Thread类。
2. 重写Thread类的run方法,把自定义线程的任务代码写在run方法上。
3. 创建Thread的子类对象,并且调用start方法启动一个线程。注意:千万不要直接调用run方法,调用start方法的时候线程就会开启,线程一旦开启就会执行run方法中代码,如果直接调用
run方法,那么就 相当于调用了一个普通的方法而已。 -
方式二:同步函数 : 同步函数就是使用synchronized修饰一个函数。
同步函数要注意的事项 :
1. 如果是一个非静态的同步函数的锁 对象是this对象,如果是静态的同步函数的锁 对象是当前函数所属的类的字节码文件(class对象)。
2. 同步函数的锁对象是固定的,不能由你来指定 的。推荐使用: 同步代码块。
原因:
1. 同步代码块的锁对象可以由我们随意指定,方便控制。同步函数的锁对象是固定 的,不能由我们来指定。
2. 同步代码块可以很方便控制需要被同步代码的范围,同步函数必须是整个函数 的所有代码都被同步了。 -
*/
package demo.thread;
class SaleTicket implements Runnable {
private static int num = 50; //票数
public void run() {
while (true) {
//同步代码块
synchronized (this) {//锁
if (num > 0) {
System.out.println(Thread.currentThread().getName() + "售出第" + num + "张票");
SaleTicket.num--;
try {
Thread.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
} else {
break;
}
}
}
}
}
class ThreadSafety {
public static void main(String[] args) {
SaleTicket s = new SaleTicket();
SaleTicket s2 = new SaleTicket();
Thread thread1 = new Thread(s, "A");
Thread thread2 = new Thread(s2, "B");
Thread thread3 = new Thread(s, "C");
thread1.start();
thread2.start();
thread3.start();
}
}