多线程 i++ 问题

1,在多线程中我们知道一个问题就是i++操作时,我们得到的结果和我们想象的结果不一样,因为i++操作会被分成三步。分别是取值,++操作,赋值(读,改,写)。当线程执行到取值或者++操作时,线程突然切换,所以最终得到的结果可能有些奇怪!看看下面程序

创建是个线程对i进行++操作

public class AtomicTest {
	public static void main(String[] args) {
		AtomicDemo ad = new AtomicDemo();
		for(int i=0; i<10; i++) {
			new Thread(ad).start();
		}
	}
}
class AtomicDemo implements Runnable {
	private volatile int i = 0; //volatile保证可见性
	@Override
	public void run() {
		try {
			Thread.sleep(200);
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
		System.out.println(Thread.currentThread().getName()+":"+getI());
	}
	public int getI() {
		return i++;
	}
}

运行结果
在这里插入图片描述

解决方法

jdk1.5后java.util.concurrent.atomic包提供了常用的原子操作
1,volatile 保证内存可见性
2,CAS:保证数据原子性 unsafe
CAS是硬件对于并发操作共享数据的支持,包含了三个操作数
内存值 V 预期值 A 更新值 B
当且仅当V==A时 才更新值 否则不做任何操作

使用Atomic操作

public class AtomicTest {
	public static void main(String[] args) {
		AtomicDemo ad = new AtomicDemo();
		for(int i=0; i<10; i++) {
			new Thread(ad).start();
		}
	}
}
class AtomicDemo implements Runnable {

	AtomicInteger a = new AtomicInteger(0);
	@Override
	public void run() {
		try {
			Thread.sleep(200);
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
		System.out.println(Thread.currentThread().getName()+":"+getI());
	}
	public int getI() {
		return a.getAndIncrement(); /类似/a++
	}

这样就可以保证i++问题了

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
下面是多线程 QProgressBar 实现进度条的示例代码: ```python from PyQt5.QtCore import QThread, pyqtSignal from PyQt5.QtWidgets import QApplication, QWidget, QProgressBar, QPushButton, QVBoxLayout import time class Worker(QThread): progress_updated = pyqtSignal(int) def run(self): for i in range(101): time.sleep(0.1) self.progress_updated.emit(i) class MainWindow(QWidget): def __init__(self): super().__init__() self.progress_bar = QProgressBar() self.start_button = QPushButton("Start") self.start_button.clicked.connect(self.start_worker) layout = QVBoxLayout() layout.addWidget(self.progress_bar) layout.addWidget(self.start_button) self.setLayout(layout) self.worker = Worker() self.worker.progress_updated.connect(self.update_progress) def start_worker(self): self.worker.start() def update_progress(self, value): self.progress_bar.setValue(value) if __name__ == "__main__": app = QApplication([]) main_window = MainWindow() main_window.show() app.exec_() ``` 在这个示例中,我们创建了一个 `Worker` 类,它继承自 `QThread` 类。在 `Worker` 类中,我们定义了一个 `progress_updated` 信号,用于在进度更新时发射信号。在 `run` 方法中,我们使用一个循环模拟了一个长时间的任务,并且在每次循环中都让线程休眠了一小段时间。在循环中,我们发射了 `progress_updated` 信号,并且将当前进度作为参数传递。 在 `MainWindow` 类中,我们创建了一个 `QProgressBar` 控件和一个 `QPushButton` 控件,用于启动线程。当用户点击 `Start` 按钮时,我们启动了 `Worker` 线程。我们还定义了一个 `update_progress` 方法,用于更新进度条的值。在 `MainWindow` 类的构造函数中,我们连接了 `Worker` 的 `progress_updated` 信号到 `update_progress` 方法。 最后,我们创建了一个 `QApplication` 实例,并且创建了一个 `MainWindow` 实例并显示。当用户关闭窗口时,`app.exec_()` 方法会返回并退出应用程序。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值