V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
nyse
V2EX  ›  Node.js

NodeJS 程序,执行完后没有自动退出,可能是什么原因造成的?

  •  
  •   nyse · 2019-05-29 16:17:59 +08:00 · 8908 次点击
    这是一个创建于 2061 天前的主题,其中的信息可能已经有所发展或是发生改变。

    NodeJS 写了一个程序,循环读取一个全局数组变量,当数组不为空时,把条目 POP 下来,提交给处理函数,处理函数使用 Promise 实现。

    执行到最后,当数组没有条目了,程序就停止那里不动,没有自动退出。

    有遇到过这种情况的吗?这有可能是什么原因造成的?

    14 条回复    2020-05-22 20:42:43 +08:00
    mooncakejs
        1
    mooncakejs  
       2019-05-29 16:23:17 +08:00 via iPhone   ❤️ 1
    望闻问切,好歹来个代码啊
    nyse
        2
    nyse  
    OP
       2019-05-29 16:31:03 +08:00
    @mooncakejs 好大一个项目,我都不知道从哪查起。。。

    就想问一下有什么情况会让 NodeJS 不自动退出。。。
    yangg
        3
    yangg  
       2019-05-29 16:33:19 +08:00   ❤️ 1
    process.on('uncaughtException', ...)
    airyland
        4
    airyland  
       2019-05-29 16:54:54 +08:00 via iPhone   ❤️ 1
    看看有没有 setInterval
    v2nika
        5
    v2nika  
       2019-05-29 16:58:00 +08:00   ❤️ 1
    Node.js 中的 io 句柄和 setTimeout/setInterval/setImmediate 默认都会阻止进程退出, 所以你的脚本应该是没跑完, 如果项目太大的话, 可以用 inspect 看一下
    maichael
        6
    maichael  
       2019-05-29 16:58:55 +08:00   ❤️ 1
    有写退出的逻辑吗?当数组为空时是怎么处理的。
    AlloVince
        7
    AlloVince  
       2019-05-29 17:11:17 +08:00   ❤️ 1
    如果使用了连接 redis 或者 mysql 的库,一般需要在程序执行完后手动断开连接,否则会保持连接,程序不会退出
    winglight2016
        8
    winglight2016  
       2019-05-29 21:49:45 +08:00   ❤️ 1
    使用了 promise 或者 async/wait 这种异步代码是有可能因为某个异常没有捕获,或者资源没有释放而阻塞,用 try/catch/final 处理一下吧。
    nondanee
        9
    nondanee  
       2019-05-30 02:41:40 +08:00 via Android   ❤️ 1
    可能是还有 .on('xxx-event', () => {}) 事件监听没 end 吧
    jiejiss
        10
    jiejiss  
       2019-05-30 07:37:45 +08:00   ❤️ 1
    如果还有 task 没有处理完就不会退出
    不过一般来讲都是 macrotask 的锅,因为 microtask 从注册到释放不会超过两个 time loop
    libook
        11
    libook  
       2019-05-30 12:47:59 +08:00   ❤️ 1
    比如监听端口:
    const http = require('http');
    const server = http.createServer(console.log);
    server.listen(8080);
    console.log(`Server is running.`);

    又比如 Promise 预期会执行 resolve 或 reject,只不过暂时还没有执行的时候:
    new Promise((res, rej) => {
    setTimeout(res, 5000);
    })

    死循环也会出现进程一直没退出的现象。
    flyingfz
        12
    flyingfz  
       2019-05-30 17:22:41 +08:00   ❤️ 1
    process.exit(0);
    nyse
        13
    nyse  
    OP
       2019-05-31 08:40:33 +08:00
    问题找到了,确实是应为没断开 MySQL 连接造成的。

    感谢大家提供的思路。
    Lws
        14
    Lws  
       2020-05-22 20:42:43 +08:00
    @winglight2016 确实如此,promise 造成了我的代理一直不退出。我这里的 promise 太多,写捕获还是有点麻烦的。看了下 node 有提供--unhandled-rejections=strict 在遇到未捕获时自动退出程序
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   实用小工具   ·   2582 人在线   最高记录 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 21ms · UTC 02:28 · PVG 10:28 · LAX 18:28 · JFK 21:28
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.