黑马程序员技术交流社区
标题:
线程
[打印本页]
作者:
山水游客
时间:
2012-6-27 11:23
标题:
线程
本帖最后由 孙胜录 于 2012-6-27 14:35 编辑
当一个线程进入一个对象的一个synchronized方法后,其它线程是否可进入此对象的其它方法? 为什么?
作者:
周兴中
时间:
2012-6-27 11:44
本帖最后由 周兴中 于 2012-6-27 11:45 编辑
当一个线程进入一个对象的一个synchronized方法后,其他线程也无法进入其对象的其他方法(根本进入不了任何方法),因为在同一个时间内,只有一个线程有CPU执行权,而同步synchronized之后的方法,是为了让CPU执行完被同步的方法之后,释放执行权,然后再和其他线程一起竞争CPU执行权,谁拿到执行权,谁才再有执行的权利.
如果不加同步修饰,那么当执行到某个方法的一部分语句之后,就有可能被其他CPU抢到执行权,那么容易出现错误(在操作共享数据时).
可以参考生产者和消费者代码:
class ProducerConsumerDemo
{
public static void main(String[] args)
{
Resource r = new Resource();
Producer pro = new Producer(r);
Consumer con = new Consumer(r);
Thread t1 = new Thread(pro);
Thread t2 = new Thread(pro);
Thread t3 = new Thread(con);
Thread t4 = new Thread(con);
t1.start();
t2.start();
t3.start();
t4.start();
}
}
/*
对于多个生产者和消费者。
为什么要定义while判断标记。
原因:让被唤醒的线程再一次判断标记。
为什么定义notifyAll,
因为需要唤醒对方线程。
因为只用notify,容易出现只唤醒本方线程的情况。导致程序中的所有线程都等待。
*/
class Resource
{
private String name;
private int count = 1;
private boolean flag = false;
// t1 t2
public synchronized void set(String name)
{
while(flag)
try{this.wait();}catch(Exception e){}//t1(放弃资格) t2(获取资格)
this.name = name+"--"+count++;
System.out.println(Thread.currentThread().getName()+"...生产者.."+this.name);
flag = true;
this.notifyAll();
}
// t3 t4
public synchronized void out()
{
while(!flag)
try{wait();}catch(Exception e){}//t3(放弃资格) t4(放弃资格)
System.out.println(Thread.currentThread().getName()+"...消费者........."+this.name);
flag = false;
this.notifyAll();
}
}
class Producer implements Runnable
{
private Resource res;
Producer(Resource res)
{
this.res = res;
}
public void run()
{
while(true)
{
res.set("+商品+");
}
}
}
class Consumer implements Runnable
{
private Resource res;
Consumer(Resource res)
{
this.res = res;
}
public void run()
{
while(true)
{
res.out();
}
}
}
复制代码
作者:
孙浩迪
时间:
2012-6-27 14:14
一个线程进入一个对象的一个synchronized方法后,其他线程也进入不了其对象的其他方法,因为在同一个时间内,只有一个线程有CPU执行权,而同步synchronized之后的方法,是为了让CPU执行完被同步的方法之后,释放执行权,然后再和其他线程一起竞争CPU执行权,谁拿到执行权,谁才再有执行的权利. synchronized的可以加在方法上,也可以用synchronized代码块。。。
作者:
张文强
时间:
2012-6-27 14:57
/*对象的synchronized方法不能进入了,其他线程还是可以进入此对象的非同步方法的
*/
public class ThreadTest {
int i = 0;
int j = 0;
public static void main(String[] args) {
final ThreadTest threadTest = new ThreadTest();
new Thread() {
public void run() {
threadTest.printI();
}
}.start();
new Thread() {
public void run() {
threadTest.printJ();
}
}.start();
}
public synchronized void printI() {
while (i < 100)
System.out.println("i==" + i++);
}
public void printJ() {
while (j < 50)
System.out.println("j==" + j++);
}
}
复制代码
作者:
刘笑
时间:
2012-6-27 18:00
不可以 ,但是在synchronized方法引用了一个全局的变量,synchronized方法
还没有运行完成的期间,在另外的一个方法中就可以修改synchronized方法的这
个变量
例如:A和B 一起去吃饭,在进一个大门,A站在门口不进去,B就不能进去,
但是现在有个专门的老师通道,C老师和B玩得好,C就可以把B带从老师通道进去
欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/)
黑马程序员IT技术论坛 X3.2