System.out.println 会同步哪些资源
时间: 2024-05-19 22:15:52 浏览: 9
System.out.println() 方法会同步标准输出流(System.out)的资源,以确保在多线程环境下,输出内容不会混乱或交错。具体来说,当一个线程调用 System.out.println() 方法时,它会先获取标准输出流的锁(synchronized),然后输出内容,并释放锁,以便其他线程可以访问标准输出流。这个同步过程保证了输出内容的正确性和可读性。
相关问题
@Test void test3() throws InterruptedException { Dog dog = new Dog(); System.out.println(getMark(dog,4)); synchronized (dog){ System.out.println(getMark(dog,4)); dog.hashCode(); System.out.println(getMark(dog,4)); } System.out.println(getMark(dog,4)); }
这段代码中,首先创建了一个Dog对象,并输出了它的标记。然后,在一个Synchronized代码块中,先输出了Dog对象的标记,然后调用了Dog对象的hashCode方法,最后再输出了Dog对象的标记。最后,又输出了Dog对象的标记。
在这段代码中,Synchronized代码块的锁对象是Dog对象,而在Synchronized代码块中调用了Dog对象的hashCode方法。由于hashCode方法并不会获取对象的锁,因此不会影响Synchronized代码块的同步和锁状态。
该代码输出结果类似于下面的内容:
```
Dog@hash1
Dog@hash1
Dog@hash1
Dog@hash1
```
其中,hash1是Dog对象的哈希值。可以看到,在Synchronized代码块中和在Synchronized代码块外部,输出的Dog对象的标记都是一样的,说明hashCode方法并没有对锁对象造成影响。
请分析一下代码有什么错误:synchronized(key) { if(!station) { food++; System.out.println(name+"生产了一份食物,现在有"+food+"份食物"); if(food>=500) station=true; } else try { System.out.println("食物生产得差不多了"+name+"进入等待态"); while(station) { notifyAll(); wait(); } System.out.println("食物消费完了"+name+"进入就绪态"); }catch(InterruptedException e) { e.printStackTrace(); } }
这段代码实现了一个生产者-消费者的场景,其中 `key` 是共享的锁对象,`station` 表示仓库是否已经满了,`food` 表示仓库中的食物数量。这段代码的问题主要是在于对锁的使用不够准确:
1. `notifyAll()` 和 `wait()` 方法应该在同步块中使用,但是只有在当前线程持有锁对象的时候才能调用它们,否则会抛出 `IllegalMonitorStateException` 异常。
2. 在 `while(station)` 循环中,应该使用 `wait()` 方法来等待其他线程的通知,而不是使用 `while` 循环一直占用 CPU 资源。
3. 在 `wait()` 方法的调用前,应该先释放锁对象,否则其他线程无法获得锁对象来执行代码,从而导致死锁。
下面是修改后的代码:
```java
synchronized(key) {
if(!station) {
food++;
System.out.println(name+"生产了一份食物,现在有"+food+"份食物");
if(food>=500) station=true;
} else {
try {
System.out.println("食物生产得差不多了"+name+"进入等待态");
while(station) {
key.wait();
}
System.out.println("食物消费完了"+name+"进入就绪态");
food--;
key.notifyAll();
} catch(InterruptedException e) {
e.printStackTrace();
}
}
}
```
修改后的代码中,`wait()` 和 `notifyAll()` 方法都在同步块中调用,并且使用 `key.wait()` 来等待其他线程的通知。在消费者线程中,先减少仓库中的食物数量,然后再调用 `key.notifyAll()` 方法来通知其他线程可以继续执行了。这样可以正确地实现生产者-消费者的场景。
相关推荐
![pdf](https://img-home.csdnimg.cn/images/20210720083512.png)
![pdf](https://img-home.csdnimg.cn/images/20210720083512.png)
![pdf](https://img-home.csdnimg.cn/images/20210720083512.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)