日期:2014-05-19  浏览次数:20817 次

Java的同步与锁的问题
请大神们帮帮小弟这个synchronized问题吧~都纠结好久了,下面这是一个运算类的代码:

public class Foo {

private int x = 100;

public int getX() {
return x;
}

public int fix(int y) {

synchronized (this) {
x = x - y;
}
return x;

}

这下边是主函数的:

public class MyRunable implements Runnable {

/**
 * @param args
 */
private Foo foo = new Foo();

public static void main(String[] args) throws InterruptedException {

MyRunable r = new MyRunable();
new Thread(r, "Thread-A").start();
new Thread(r, "Thread-B").start();
}

@Override
public void run() {
for (int i = 0; i < 3; i++) {

this.fix(30);

try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()
+ "  当前foo对象的值为:" + foo.getX());
}

}

public  int fix(int y) {
return foo.fix(y);
}


我的输出是这样的:
Thread-A  当前foo对象的值为:40
Thread-B  当前foo对象的值为:40
Thread-A  当前foo对象的值为:-20
Thread-B  当前foo对象的值为:-20
Thread-A  当前foo对象的值为:-80
Thread-B  当前foo对象的值为:-80



可是我锁了运算的方法了阿~为啥还是两个线程同时运行了该方法呢~?要是改怎么改正呢~我如果把synchronized加在run上~那个到时算数对了~可是不就跟jion的方法一样了吗
synchronized? java thread Runnble

------解决方案--------------------
同时运行该方法没有什么不对啊。
你想一个线程运行
this.fix(30);
执行完这句话就已经结束了,那么就释放资源,另外一个线程也可以获取,有什么错么?、
如果你想要看效果的话sleep时间应该加在方法内部。

------解决方案--------------------
呵呵,楼主只看到结果,而忽略了过程了。
其实楼主的锁是起作用的,两个线程也是交替执行foo.fix()方法,但你并不是foo.fix()执行完后马上调用foo.getX(),等你调用foo.getX()时,两个线程的单回合都已经结束了,这一瞬间x的值是一样的。你有没有发现你没有打印70,因为两个线程各减30,所以你打印的是40

你要真想看到效果,请把
System.out.println(Thread.currentThread().getName()
                    + "  当前foo对象的值为:" + foo.getX());
挪到
synchronized (this) {
            x = x - y;
            System.out.println(Thread.currentThread().getName()
                    + "  当前foo对象的值为:" + getX());
        }
------解决方案--------------------
代码是没问题的,你把打印放到减法里面

public int fix(int y) {
 
        synchronized (this) {
            x = x - y;
            Syst