使用多线程解决读写数据不一致的问题

2024-09-05 11:04

本文主要是介绍使用多线程解决读写数据不一致的问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

引言

在工作中我们经常会遇到从一个文件中读取数据,然后去做另一个操作,最近小编在日常工作中遇到一个问题:从一个json文件中读取参数化数据循环对比配置,发现生成的配置都是取得最后一个参数所生成的,在这里小编就很疑惑了,查看日志,入参都是没有问题的,是什么原因引起的呢?小编排查日志很久都没发现啥问题,于是我用单个参数进行更新发现每个参数都是OK的,不会存在更新失败的问题,所以参数是没有问题的,那么是哪里的问题呢。

然后我注意到执行循环对比时,多个配置进行对比程序运行完时间竟然还比单个文件运行时间短,这明显不合理啊,于是我突发奇想:是不是程序没执行完就往下循环了,因为小编刚开始是用的单线程。

初始代码

问题代码如下

# 省略库导入,代码仅供参考current_dir = os.getcwd()
config_dir = os.path.join(current_dir, 'configs')
app_file = os.path.join(config_dir, "app.json")
app_list = []
diff_result = Nonedef test_batch_diff():logging.info(f"{config_dir},{app_file}")with open(app_file, "r") as f:app_list = json.load(f)logging.info(f"{json.dumps(app_list, indent=2)}")for app_value in app_list:logging.info(f'app_value:{app_value}')app = app_value.get("app")a = app_value.get("a")e = app_value.get("e")version = app_value.get("version")app_dict = dict(app=app, a=a, e=e, version=version)logging.info(f'app_dict:{app_dict}')if app and a and e and version:# update_config_psa_data(**app_dict)diff_result = single_configs_compare_v2(app_dict)if diff_result:write_diff_results(app, diff_result)# 示例调用
if __name__ == "__main__":test_batch_diff()

为啥这个写入的diff_result都是psa_list最后的一个数据?小编查阅了下资料确认是:

可能出在single_configs_compare_v2(psa_dict)函数耗时较长,导致在该函数执行期间程序进入下一次循环,从而导致每次写入的diff_result都是最后一个数据。

执行时间较长,然后没完全执行完就开始下个循环了,导致参数化结果都是一样的,这显然是不可接受的。在这里可以将耗时较长的任务放在另外一个线程中进行处理,经测试,将耗时长的这个函数single_configs_compare_v2放到另外一个线程中执行问题就解决了。

使用多线程优化后的代码


# 省略库导入,代码仅供参考
current_dir = os.getcwd()
config_dir = os.path.join(current_dir, 'configs')
app_file = os.path.join(config_dir, "app.json")
app_list = []
diff_result = Nonedef process_app(app_dict):diff_result = single_configs_compare_v2(app_dict)if diff_result:write_diff_results(app_dict["app"], diff_result)def test_batch_diff():logging.info(f"{config_dir},{app_file}")with open(app_file, "r") as f:app_list = json.load(f)logging.info(f"{json.dumps(app_list, indent=2)}")threads = []for app_value in app_list:logging.info(f'app_value:{app_value}')app = app_value.get("app")a = app_value.get("a")e = app_value.get("e")version = app_value.get("version")app_dict = dict(app=app, a=a, e=e, version=version)logging.info(f'app_dict:{app_dict}')if app and a and e and version:thread = threading.Thread(target=process_app, args=(app_dict,))threads.append(thread)thread.start()# 确保所有线程都执行完毕for thread in threads:thread.join()# 示例调用
if __name__ == "__main__":test_batch_diff()

在这个修改后的代码中,我们将single_configs_compare_v2(app_dict)的执行放在了一个单独的线程中,这样可以避免阻塞主循环。通过使用多线程,可以让每次对比操作在独立的线程中进行,从而避免因为耗时操作导致的数据混乱问题。

总结

多线程是一种常见的并发编程模型,在许多应用场景中具有显著的优势。下面详细介绍多线程的使用场景及其优势。

多线程的使用场景

  1. I/O 密集型任务

    • 当程序需要频繁进行 I/O 操作(如磁盘读写、网络通信等),并且这些操作耗时较长时,使用多线程可以让其他线程继续执行,提高程序的整体响应速度和效率。
  2. CPU 密集型任务

    • 对于需要大量计算的任务,多线程可以充分利用多核处理器的并行计算能力,加快计算速度。但是需要注意的是,多线程在 CPU 密集型任务上的优势取决于具体的应用场景和硬件配置。
  3. 用户界面交互

    • 在图形用户界面(GUI)应用程序中,主线程通常负责处理用户交互事件,而其他线程则可以用来执行耗时的后台任务,避免用户界面冻结。
  4. 网络服务

    • 在服务器端开发中,多线程可以用来处理并发的客户端请求,每个客户端连接由一个独立的线程处理,提高服务器的响应能力和吞吐量。
  5. 批处理任务

    • 对于需要处理大量数据或任务的情况,可以将任务分解成多个子任务,并在多个线程中并行处理,以提高整体处理速度。

多线程的优势

  1. 提高响应速度

    • 多线程可以提高程序的响应速度,尤其是在处理 I/O 密集型任务时,可以让程序在等待 I/O 操作完成的同时继续执行其他任务。
  2. 提高资源利用率

    • 多线程可以更好地利用计算机系统的资源,尤其是多核处理器。每个线程都可以在不同的核心上并行执行,从而提高整体性能。
  3. 改善用户体验

    • 在 GUI 应用程序中,多线程可以避免由于长时间运行的任务导致的界面冻结,提供更好的用户体验。
  4. 简化复杂任务的处理

    • 对于复杂的任务,可以将其拆分成多个子任务,并在多个线程中并行执行,简化任务处理的难度。
  5. 提高系统吞吐量

    • 在服务器端应用中,多线程可以处理更多的并发请求,提高系统的吞吐量。

多线程的注意事项

尽管多线程有很多优势,但也存在一些需要注意的问题:

  1. 线程安全

    • 多线程环境下,多个线程可能会同时访问共享资源,因此需要确保对共享资源的访问是线程安全的,通常通过使用锁(如 threading.Lock)来实现。
  2. 死锁

    • 如果多个线程持有不同的锁,并且都在等待对方释放自己所需的锁,就会发生死锁。需要合理设计线程间的锁获取顺序来避免死锁。
  3. 资源开销

    • 创建和销毁线程会有一定的资源开销,因此在频繁创建和销毁线程时需要注意性能问题。
  4. 调试难度

    • 多线程程序的调试通常比单线程程序更加复杂,因为线程之间的执行顺序往往是不确定的,可能会引入难以复现的 bug。

总之,多线程是一种强大的工具,可以显著提高程序的性能和响应速度,但在使用时需要谨慎处理线程间的数据共享和同步问题。在适当的应用场景下,合理利用多线程可以带来极大的性能提升。

这篇关于使用多线程解决读写数据不一致的问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1138780

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Linux join命令的使用及说明

《Linuxjoin命令的使用及说明》`join`命令用于在Linux中按字段将两个文件进行连接,类似于SQL的JOIN,它需要两个文件按用于匹配的字段排序,并且第一个文件的换行符必须是LF,`jo... 目录一. 基本语法二. 数据准备三. 指定文件的连接key四.-a输出指定文件的所有行五.-o指定输出

Linux jq命令的使用解读

《Linuxjq命令的使用解读》jq是一个强大的命令行工具,用于处理JSON数据,它可以用来查看、过滤、修改、格式化JSON数据,通过使用各种选项和过滤器,可以实现复杂的JSON处理任务... 目录一. 简介二. 选项2.1.2.2-c2.3-r2.4-R三. 字段提取3.1 普通字段3.2 数组字段四.

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

k8s按需创建PV和使用PVC详解

《k8s按需创建PV和使用PVC详解》Kubernetes中,PV和PVC用于管理持久存储,StorageClass实现动态PV分配,PVC声明存储需求并绑定PV,通过kubectl验证状态,注意回收... 目录1.按需创建 PV(使用 StorageClass)创建 StorageClass2.创建 PV

IDEA和GIT关于文件中LF和CRLF问题及解决

《IDEA和GIT关于文件中LF和CRLF问题及解决》文章总结:因IDEA默认使用CRLF换行符导致Shell脚本在Linux运行报错,需在编辑器和Git中统一为LF,通过调整Git的core.aut... 目录问题描述问题思考解决过程总结问题描述项目软件安装shell脚本上git仓库管理,但拉取后,上l

Redis 基本数据类型和使用详解

《Redis基本数据类型和使用详解》String是Redis最基本的数据类型,一个键对应一个值,它的功能十分强大,可以存储字符串、整数、浮点数等多种数据格式,本文给大家介绍Redis基本数据类型和... 目录一、Redis 入门介绍二、Redis 的五大基本数据类型2.1 String 类型2.2 Hash

Redis中Hash从使用过程到原理说明

《Redis中Hash从使用过程到原理说明》RedisHash结构用于存储字段-值对,适合对象数据,支持HSET、HGET等命令,采用ziplist或hashtable编码,通过渐进式rehash优化... 目录一、开篇:Hash就像超市的货架二、Hash的基本使用1. 常用命令示例2. Java操作示例三