多线程踩坑记录:
1、多线程切记不可以同时操作同一个原子数据。解释:存在一个条数据库A数据,不可以在2个或2个以上的线程中同时操作A数据。会引发重复操作。
2、多线程操作方法不要加synchronized同步关键字,这失去了多线程的意义。解释:会是多线程按照同步线程执行。
3、调用第三方接口时尽量不要使用多线程。解释:因为你不知道第三方接口是否针对高并发作了处理。如果第三方接口没有针对高并发作处理,自己写的多线程逻辑也没有处理好,会导致重复操作成功。后果很难处理。
总结:
作为一只菜鸟遇到过的问题。给自己提醒下。

处理一批数据的多线程方法:
使用同步线程获取批量数据,在逻辑上将这批数据分成不同部分,每一部分使用多线程处理。这样可以避免多个线程同时操作统一个数据。

多线程的实例synchronized对比:

线程执行方法:

import threads.TaskRunnable;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; public class ThreadsMain {
public static void main(String[] args) {
//创建线程池对象 参数5,代表有5个线程的线程池
ExecutorService service = Executors.newFixedThreadPool(5);
//创建Runnable线程任务对象
TaskRunnable task = new TaskRunnable();
//从线程池中获取线程对象
service.submit(task);
System.out.println("----------------------");
//再获取一个线程对象
service.submit(task);
//关闭线程池
service.shutdown();
}
}

带有synchronized的执行方法:

public class TaskRunnable implements Runnable{
@Override
public synchronized void run() {
for (int i = 0; i < 10; i++) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("自定义线程任务在执行"+i);
}
}
}

执行结果:

----------------------
自定义线程任务在执行0
自定义线程任务在执行1
自定义线程任务在执行2
自定义线程任务在执行3
自定义线程任务在执行4
自定义线程任务在执行5
自定义线程任务在执行6
自定义线程任务在执行7
自定义线程任务在执行8
自定义线程任务在执行9
自定义线程任务在执行0
自定义线程任务在执行1
自定义线程任务在执行2
自定义线程任务在执行3
自定义线程任务在执行4
自定义线程任务在执行5
自定义线程任务在执行6
自定义线程任务在执行7
自定义线程任务在执行8
自定义线程任务在执行9

Process finished with exit code 0

没有synchronized的执行方法:

public class TaskRunnable implements Runnable{
@Override
public void run() {
for (int i = 0; i < 10; i++) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("自定义线程任务在执行"+i);
}
}
}

执行结果:

----------------------
自定义线程任务在执行0
自定义线程任务在执行0
自定义线程任务在执行1
自定义线程任务在执行1
自定义线程任务在执行2
自定义线程任务在执行2
自定义线程任务在执行3
自定义线程任务在执行3
自定义线程任务在执行4
自定义线程任务在执行4
自定义线程任务在执行5
自定义线程任务在执行5
自定义线程任务在执行6
自定义线程任务在执行6
自定义线程任务在执行7
自定义线程任务在执行7
自定义线程任务在执行8
自定义线程任务在执行8
自定义线程任务在执行9
自定义线程任务在执行9

Process finished with exit code 0

对比结论:

加入synchronized执行结果会跟同步方法相似。就失去了多线程的意义

最新文章

  1. linux零基础入门总结
  2. 开源通信(C#)__
  3. [转]Vs解决方案的目录结构设置和管理
  4. OpenCV的基本数据结构
  5. Servlet3.0-使用注解定义Servlet
  6. 基于ASP.NET的comet简单实现
  7. oracle技巧-持续更新
  8. 获取CPU使用情况信息(转)
  9. Python的学习
  10. basename $0的用法
  11. 断开/删除 SVN 链接(.svn)的几种方法
  12. 2015年5月9日 student information management system
  13. Oracle数据库常用关键字以及函数
  14. [译]ASP.NET Core 2.0 会话状态
  15. 关于myeclipse8.6的优化设置
  16. 学习pthreads,管理线程的栈
  17. baseFileWriter.go
  18. windows2012的服务器远程桌面提示内部错误的问题解决方法
  19. 深入C#
  20. 国内的Android SDK镜像

热门文章

  1. awk.sed.grep三剑客详解
  2. CSS基础2——选择器
  3. 第3章 如何编写函数定义 3.7 if特殊表
  4. centos 6.9使用Rsync+Inotify-tools实现数据实时同步
  5. 【Mac系统】之fiddler下载和安装
  6. 【Python+selenium Wendriver API】之操作警告和弹出框
  7. 2016 acm香港网络赛 B题. Boxes
  8. cocos2d-x 3.0rc1 使用iconv库 解决UTF8乱码问题
  9. ios推送服务,php服务端
  10. 论JavaWeb前后端分离放弃jsp