• 我的物联网项目(九)久违的死循环


    旧业务不断的调整,新的需求不断的开发,版本不断的迭代,这个是当前项目的一个暂时不可改变的现状。再加上每个开发人员写代码的风格和层次不一样,所以有很多本来可以在写代码过程中避免的问题非要通过线上的报警才能发觉。

    最近两天线上linux服务器发现java进程CPU不断的飙升,新发的包过一会儿CPU就慢慢上涨,感觉很奇怪,之前没有这种情况,应该开发人员新写的代码所导致,排查问题如下:

    一 使用top命令

    使用top命令

    top

    结果如下:

    过了一会儿,继续top查看

    二 查询具体的线程CPU占用率

    使用命令 ps -mp 5910 -o THREAD,tid,time

    ps -mp 5910 -o THREAD,tid,time

    结果如下:

    可以看到5910号进程产生了大量的线程,继续查看该线程的具体执行情况。

    先通过 printf "%x " 18082拿到十进制

    printf "%x
    " 18082

    结果是46a2

    三 使用jstack查看具体线程

    使用命令 jstack 5910 | grep 46a2-a 100

    jstack 5910 | grep 46a2-a 100

    发现如下:

    找来开发人员一起仔细想想写的代码在什么地方用到了线程池之类的,果然真有这样的代码,一起喵了眼代码,发现了问题。

    注:以下代码是模拟代码

    @Scheduled(fixedDelay = 1000)
        public void doJob() {
        	System.out.println("###sync start:"+ dateFormat.format(new Date()) + "###");
        	ExecutorService executorService = Executors.newCachedThreadPool();
    		for (int i = 0; i < Integer.parseInt(threadnum); i++) {
    			executorService.execute(createTask("test"+i));
    		}
    		System.out.println("###sync end:" + dateFormat.format(new Date()) + "###");
        }
    private Runnable createTask(final String key) {
    		return new Runnable() {
    			public void run() {
    				while (true) {
    					try {
    						disposeOrder(key);
    					} catch (Exception e1) {
    						System.out.println("ScheduledTasks disposeOrderException:" + e1);
    						e1.printStackTrace();
    					}			
    				}
    			}
    		};
    	}
    private void disposeOrder(String key){
    		System.out.println("key="+key);
    		try {
    			Thread.sleep(10000);
    		} catch (InterruptedException e) {
    			e.printStackTrace();
    		}
    	}

    问题其实就在createTask的方法里面while (true)进入了死循环,不断的产生线程导致(前面的线程没释放,后面的不断在产生)。

    其实事情到这里还未完,我看到代码里面有用到newCachedThreadPool,它是创建一个可缓存线程池,如果线程池长度超过处理需要,可灵活回收空闲线程,若无可回收,则新建线程。也就是说如果主线程提交任务的速度高于线程池中处理任务的速度时,CachedThreadPool会不断创建新线程。极端情况下,CachedThreadPool会因为创建过多线程而耗尽CPU资源。所以这块也需要注意先备注下后面再优化。

     
  • 相关阅读:
    垃圾回收的整个过程
    实验:老年代空闲空间放不下minorgc晋升的对象发生fullgc
    实验: survivor放不下的对象进入老年代
    实验: 动态对象年龄判定
    永久代、栈内存大小怎么设置
    如何设置高并发系统的jvm堆内存大小
    Bin、App_Data等文件夹详述(转自http://blog.csdn.net/zzjiadw/article/details/6801506)
    谈谈并行、并发或多线程
    SQL---存储过程---存储过程编写案例
    SQL---存储过程---sp_addextendedproperty表字段加描述
  • 原文地址:https://www.cnblogs.com/dgcjiayou/p/7886701.html
Copyright © 2020-2023  润新知