您的位置:首页 > 其它

13.7 线程安全

2016-05-23 22:26 225 查看
案例:3个售票窗口
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();

}
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: