A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© 史卜坤 中级黑马   /  2012-7-11 21:34  /  1893 人查看  /  2 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

public class TestThreadcomtrue {
        public static void main(String [] srgs){
                Persion i=new Persion();
                new Thread(new Threadren(i)).start();
                new Thread(new Threadr(i)).start();
        }

}
class Threadren implements Runnable{
        Persion q=new Persion();
        Threadren(Persion q){this.q=q;}
        public void run(){
                q.get();
        }
}
class Threadr implements Runnable{
        Persion q=new Persion();
        Threadr(Persion q){this.q=q;}
        public void run(){
                q.put();
        }
}
class Persion{
        String name,sex;
        int i=1;
        public synchronized void get(){
                while(true){
                if(i%2==0){
                name="zs";
                try{Thread.sleep(1);}catch(Exception e){}
                sex="male";
                }
                else{
                name="lg";
                sex="famale";
                }
                i=(i+1)%2;
        }
}
        public synchronized void put(){
                while(true){
                System.out.println(name+" "+sex);
                }
        }
}

加上synchronized 也不出错就是没输出,去掉就没问题了。我看视频里加上就能实现同步,我加上就出问题了呢。

评分

参与人数 1技术分 +1 收起 理由
刘笑 + 1 赞一个!

查看全部评分

2 个回复

倒序浏览
你get方法里的while(true)一直在运行使得第一个线程一直占着这个对象的锁,这时你用另一个线程调用这个对象的另一个同步方法,当然获取不了这个对象的锁,所以你加上synchronized之后不出错也没输出,如果去掉get()方法 里的while(true),如下
public synchronized void get(){
                        if(i%2==0){
                                        name="zs";
                                        try{Thread.sleep(1);}catch(Exception e){}
                                        sex="male";
                        }else{
                                name="lg";
                                sex="famale";
                        }
                        i=(i+1)%2;
        }
这个时候就没问题了。
另外,当你去掉put()方法synchronized之后,那put()就是一个普通方法,这时第二个线程是可以调用这个对象的非同步方法的,所以就如你看到的结果一样了。
帮你总结一下
1、synchronized关键字的作用域有二种:
1)是某个对象实例内,synchronized aMethod(){}可以防止多个线程同时访问这个对象的synchronized方法(如果一个对象有多个synchronized方法,只要一个线程访问了其中的一个synchronized方法,其它线程不能同时访问这个对象中任何一个synchronized方法)。这时,不同的对象实例的 synchronized方法是不相干扰的。也就是说,其它线程照样可以同时访问相同类的另一个对象实例中的synchronized方法;
2)是某个类的范围,synchronized static aStaticMethod{}防止多个线程同时访问这个类中的synchronized static 方法。它可以对类的所有对象实例起作用。

2、除了方法前用synchronized关键字,synchronized关键字还可以用于方法中的某个区块中,表示只对这个区块的资源实行互斥访问。用法是: synchronized(this){/*区块*/},它的作用域是当前对象;

评分

参与人数 2技术分 +2 收起 理由
黑马张扬 + 1
刘笑 + 1 赞一个!

查看全部评分

回复 使用道具 举报
本帖最后由 鲍霄霄 于 2012-7-12 12:57 编辑

 synchronized 方法控制对类成员变量的访问:每个类实例对应一把锁,每个 synchronized 方法都必须获得调用该方法的类实例的锁方能执行,否则所属线程阻塞,方法一旦执行,就独占该锁,直到从该方法返回时才将锁释放,此后被阻塞的线程方能获得该锁,重新进入可执行状态。
    主要是因为上面 那个 Persion i=new Persion();
                new Thread(new Threadren(i)).start();
                new Thread(new Threadr(i)).start();

他们用的是同一个Persion 对象 i
如果不是同一个 就可以了
你在Persion i=new Persion();
下再加一行Persion r=new Persion();
然后让第二个线程里传r  而不是i 你就会看到东西了

http://baike.baidu.com/view/1207212.htm可以参考理解下
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马