案例:3个售票窗口
用synchronized控制同步代码块。其的参数是任意的对象都可以作为锁对象。
同步代码块注意事项:
1、任何一个对象都可以做为锁的对象。
2、在同步代码中调用了sleep方法不是释放了对象。
3、只有真正在线程安全问题的时候才使用同步代码块,否则会降低效率。
4、线程操作的锁对象必须是唯一共享的,否则无效。
出现线程安全问题的根本原因:
1、存在两个或者两个以上的线程对象,而且线程之间共享着一个资源。
2、有多个语气操作了共享资源。
class test extends Thread
{
int tick= 40;
test(String name)
{
super(name);
}
@Override
public void run() {
while(true)
{
if(0 < tick)
{
System.out.println(Thread.currentThread().getName()+"售了第"+tick+"票");
tick--;
}
else
{
System.out.println("票买完了");
break;
}
}
}
}
class wu
{
public static void main(String [] args)
{
test t1 = new test("窗口一");
test t2 = new test("窗口二");
test t3 = new test("窗口三");
t1.start();
t2.start();
t3.start();
}
}
这样的远行结果就是一共买了40*3=120张票,所以,在定义票时,应该把票设定为static,不然每次new test类时,就相当于把票的数字翻一倍。
再加了static后,运行了,然后就会发现,还有另一个问题,就是信息安全的问题,例如我们现实票已经买完了,但是还可以卖票。
用synchronized控制同步代码块。其的参数是任意的对象都可以作为锁对象。
同步代码块注意事项:
1、任何一个对象都可以做为锁的对象。
2、在同步代码中调用了sleep方法不是释放了对象。
3、只有真正在线程安全问题的时候才使用同步代码块,否则会降低效率。
4、线程操作的锁对象必须是唯一共享的,否则无效。
出现线程安全问题的根本原因:
1、存在两个或者两个以上的线程对象,而且线程之间共享着一个资源。
2、有多个语气操作了共享资源。
class test extends Thread
{
static int tick= 40;
static Object d = new Object();//这个对象资源为静态的,不然无效
test(String name)
{
super(name);
}
@Override
public void run() {
while(true)
{
synchronized(d)//这个对象,也可以是“文字”例如:synchronized("锁对象")
{
if(0 < tick)
{
System.out.println(Thread.currentThread().getName()+"售了第"+tick+"票");
tick--;
}
else
{
System.out.println("票买完了");
break;
}
}
}
}
}
class wu
{
public static void main(String [] args)
{
test t1 = new test("窗口一");
test t2 = new test("窗口二");
test t3 = new test("窗口三");
t1.start();
t2.start();
t3.start();
}
}