gpt4 book ai didi

java - 同步静态方法和非静态方法的区别

转载 作者:IT老高 更新时间:2023-10-28 21:09:23 29 4
gpt4 key购买 nike

在java中同步静态方法和非静态方法有什么区别?谁能用一个例子解释一下。另外同步一个方法和同步一个代码块有什么区别吗?

最佳答案

我将尝试添加一个示例以使这一点更加清晰。

如前所述,Java 中的同步是 Monitor 的实现。概念。当您将代码块标记为同步时,您使用一个对象作为参数。当一个执行线程来到这样的代码块时,它必须首先等待,直到在同一对象的同步块(synchronized block)中没有其他执行线程。

Object a = new Object();
Object b = new Object();
...
synchronized(a){
doStuff();
}
...
synchronized(b){
doSomeStuff();
}
...
synchronized(a){
doOtherStuff();
}

在上面的例子中,一个线程正在运行 doOtherStuff()会阻止另一个线程进入保护 doStuff() 的代码块.但是,线程可以进入 doSomeStuff() 附近的 block 。没有问题,因为它在 Object b 上同步,而不是 Object a .

当您在实例方法(非静态方法)上使用同步修饰符时,它与使用“this”作为参数的同步块(synchronized block)非常相似。所以在下面的例子中,methodA()methodB()会以同样的方式行事:

public synchronized void methodA() {
doStuff();
}
...
public void methodB() {
synchronized(this) {
doStuff();
}
}

请注意,如果您有 methodC()在那个不同步且没有同步块(synchronized block)的类中,没有什么可以阻止线程进入该方法,粗心的编程可能会让该线程访问对象中的非安全代码。

如果你有一个带 synchronized 修饰符的静态方法,它实际上与有一个带 ClassName.class 的同步块(synchronized block)相同。作为参数(如果您有该类的对象 ClassName cn = new ClassName(); ,您可以使用 Class c = cn.getClass(); 访问该对象)

class ClassName {
public void static synchronized staticMethodA() {
doStaticStuff();
}
public static void staticMethodB() {
synchronized(ClassName.class) {
doStaticStuff();
}
}
public void nonStaticMethodC() {
synchronized(this.getClass()) {
doStuff();
}
}
public static void unSafeStaticMethodD() {
doStaticStuff();
}
}

所以在上面的例子中,staticMethodA()staticMethodB()以同样的方式行事。正在执行的线程也将被阻止访问 nonStaticMethodC() 中的代码块因为它在同一个对象上同步。

但是,重要的是要知道没有什么可以阻止正在执行的线程访问 unSafeStaticMethodD() .即使我们说静态方法“在 Class 对象上同步”,也并不意味着它会同步对该类中方法的所有访问。它只是意味着它使用 Class 对象进行同步。非安全访问仍然是可能的。

关于java - 同步静态方法和非静态方法的区别,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6367885/

29 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com