• 解决node项目一个极度困难的捕获异常却无法读取异常信息的问题


    这个项目是集成了第三方NeteaseCloudMusicApi项目的接口代码,我没有直接使用它的接口,因为需要再跑一个npm run开个端口,感觉很麻烦。

    所以下定决心,使用拆分代码的方式,硬生生将这个api项目的部分api接口代码集成到了我自己的项目。当然前提是必须能看懂它的代码,然后才能做拆分,否则你根本无从下手

    最后终于都搞定了,这个过程很复杂,把它的api代码拆分到了我自己的项目中。

    但是有个问题,有时候请求歌曲的时候会报404错误

     但是这个404错误报的明显不正确,因为我已经在代码里打了埋点

     上面证明:这个接口请求实际已经匹配到路由了,所以绝不应该是404错误!

    然后排查代码发现这个第三方api的错误处理代码居然写的是404!

    这就不对了,肯定起码应该是个500,但是这还不够,起码得知道是什么错误,

    所以,把它的msg的值写成捕获到的err

     本以为这样处理就应该能看到具体的异常信息了,但是实际返回结果却还是空对象:

     问题变得棘手了,异常明明捕获到了啊可是为什么打印不出来呢!

    看来首先得看看这个err的原始信息是什么? 于是在catch代码块的开头先console.log一下这个err:

    然后这个异常信息被打印出来了:

    报错如下:

    get Err==> TypeError: Cannot read property 'sort' of undefined
        at Object.module.exports [as module] (D:\VueCode\VueProject\myqqmusic\src\registerRouter\NetEaseApi\modules\song_url.js:33:10)
        at processTicksAndRejections (internal/process/task_queues.js:93:5)
        at async D:\VueCode\VueProject\myqqmusic\src\registerRouter\NetEaseApi\routes.js:69:40
     

     但是为什么把err赋值给msg就取不到值呢,然后断点调试看了一下这个error的数据结构:

     然后结合GPT分析了一下,按理说: 通过err.message和err.stack应该可以获取到异常的具体信息:

     然后确实看到了异常的具体信息了,这回没有报空值{}:

     

     好了,小结一下:Javascript中Error对象必须通过message或者stack属性来获取到具体信息,

    也就是err.message或者err.stack这种形式,不能把Error对象直接赋值给变量!!!否则变量获取的值就是空对象!

    但是这种报错信息太难看了,应该根据报错指引,找到异常被触发的地方:

    get Err==> TypeError: Cannot read property 'sort' of undefined
        at Object.module.exports [as module] (D:\VueCode\VueProject\myqqmusic\src\registerRouter\NetEaseApi\modules\song_url.js:33:10)
        at processTicksAndRejections (internal/process/task_queues.js:93:5)
        at async D:\VueCode\VueProject\myqqmusic\src\registerRouter\NetEaseApi\routes.js:69:40

     根据上面报错信息,找到song_url.js第33行:

    这个地方说result没有sort属性,那么result本身极有可能也是个undefined!

    所以非常有必要看看它的父级res到底是个什么东西

    然后通过断点调试可以看到res底下果然没有data属性,那么result就必然是个undefined了,

    难怪result.sort会触发Cannot read property 'sort' of undefined异常了

    而且这里可以看到引发异常的真实原因:

    // 这个异常信息是放在res.body里面的

     {
      code: -460,
      message: "网络太拥挤,请稍候再试!",
    }

     所以在通过res.body.data获取result之前,要先判断一下res.body是否有data这个属性

    或者说判断res的返回状态是否正常,因为仅仅根据res.body是否有data属性来判断返回是否正常有点太片面了

    然后,我找了一个请求成功的例子:

    可以看到请求成功的时候:不仅res.body有data属性,res.body.code还等于200,

    所以应该增加以下代码:在res请求失败的时候主动抛出异常,并将res.body作为异常信息:

    if (res.body.code != 200 ) {    

        throw new Error(res.body)   // 因为异常的详情信息是存储在res.body里面

      }

    但是在前面说的外部异常拦截的地方(catch err代码块),得到结果是:msg没能正确解构出异常的信息,显示的是: [object Object]

     

     

     关于这个问题,我查了很久也查了很多资料,尝试了很多次,都没有能够解决问题。

    可以说:这是这次问题排查中难度最大的地方

    然后呢,我就尝试着从chatgpt对话,希望能找到答案。

    最初按照它给的方法试了好几次,也还是不行。

    最后我问了一个关键的神提问:

    你没懂我意思,比如原始异常的信息是res.body="{"code":-460,"message":"网络太拥挤,请稍候再试!"}" ,然后我throw new Error(JSON.stringify(res.body)) 抛出了这个异常,我问catch的时候怎么才能把这json格式的异常信息取出来

     然后chatgpt终于给出了正确的回答:

    我明白你的问题了。如果你的异常信息是一个JSON字符串,你可以在catch块中使用JSON.parse来解析它,以获取原始的JSON对象。以下是一个示例:

    1. try {
    2. const res = { body: '{"code":-460,"message":"网络太拥挤,请稍候再试!"}' };
    3. throw new Error(JSON.stringify(res.body));
    4. } catch (error) {
    5. try {
    6. // 尝试将异常消息解析为JSON
    7. const errorData = JSON.parse(error.message);
    8. console.log("Error code:", errorData.code);
    9. console.log("Error message:", errorData.message);
    10. } catch (parseError) {
    11. // 处理解析JSON失败的情况
    12. console.error("Error parsing JSON:", parseError);
    13. }
    14. }

    在这个示例中,我们首先抛出一个异常,异常消息是JSON字符串。然后在catch块中,我们尝试使用JSON.parse将异常消息解析为JSON对象,并访问其中的属性。如果解析失败,我们捕获到解析错误并进行处理。

    这样你就可以获取到原始的JSON对象信息了。但要小心处理JSON解析失败的情况,因为异常消息不一定总是有效的JSON字符串。

    这里的关键有两点:

    1.  自定义异常信息时: new Error构造函数通常接收一个字符串作为错误信息的参数,而不是一个对象

    如果是接收参数是对象必须通过JSON.stringify()方法转换成json字符串

    2. 当已预知捕获的error是json字符串时,那么此时必须使用JSON.parse()方法把error.message转换成object对象,否则打印出来的就是[object Object]

     所以,代码需要这样修改:

    异常主动抛出的部分:

      if (res.body.code != 200 ) {

        // 注意这里new Error的参数不能是对象,而必须是字符串!!!

        // 所以res.body必须通过JSON.stringify转成字符串

        throw new Error(JSON.stringify(res.body))  

      }

     异常捕获的地方:

    catch (err) {

                    // 因为err.message是JSON字符串,必须通过JSON.parse转换成对象,

                   // 否则msg: errData赋值时就会变成: Error: [object Object]

                    // 这样msg就获取不到值,变成了空值了

                    const errData = JSON.parse(err.message)  //关键点 :必须使用 JSON.parse方法

                    console.log("Error message:", errData.message);       

                    console.log("get Err==>", err)

                    console.log('[ERR]', decode(req.originalUrl), {

                        status: err.status,

                        body: err.body,

                    })

                    if (!err.body) {

                        res.status(500).send({

                            code: 500,

                            data: null,

                            msg: errData,                    

                        })

                        return

                    }

    问题解决:前台msg信息终于正常显示出来了,不再是[object Object]

     

     总结:

    1. Javascript中的Error对象必须通过message或者stack属性来获取到它的具体信息,

    也就是err.message或者err.stack这种形式。

    不能把整个Error对象直接赋值给变量!!!否则变量获取的值就是空对象!

    2.  自定义异常信息时: new Error构造函数通常接收一个字符串作为错误信息的参数,而不是一个对象

    如果是接收参数是对象必须通过JSON.stringify()方法把该对象转换成json字符串

    // 主动抛出异常的部分

    // 如果是接收参数是对象必须通过JSON.stringify()方法把该对象转换成json字符串

    new Error(JSON.stringify(err_obj))

    3. 当已预知捕获的error是json字符串时,那么此时必须使用JSON.parse()方法把error.message转换成object对象,否则打印出来的就是[object Object]

    // 捕获异常的部分

    catch(err) {

    // 当已知err是json字符串时,则必须使用JSON.parse()方法把error.message转换成object对象,否则打印出来的就是[object Object]

       errData = JSON.parse(err.message)  // 使用JSON.parse方法是关键

    }

  • 相关阅读:
    【Window10 】删除‘设备和驱动器’中的百度网盘、酷狗音乐、迅雷下载等
    MySQL(5)
    UnrealEngine创建自定义资产类型
    继承【C++】
    知识分享|分段函数线性化及matlab测试
    SpringCloud与云原生
    排序-选择类排序
    自动控制原理3.5---线性系统的稳定性分析
    python可视化分析(五)-绘制边缘箱线图
    字节有『芯』在跳动,了吗?YOLOv7目标检测实现:确实挺好;伯克利博士找工作的6个月;软件工程资源大列表 | ShowMeAI资讯日报
  • 原文地址:https://blog.csdn.net/jiaohuizhuang6019/article/details/134070952