Webpack 代码拆分“加载块失败”错误文件路径错误

IT技术 reactjs typescript webpack
2021-04-12 18:23:03

我正在将 React + Typescript 与 Webpack 一起使用,并且我正在尝试在实际需要时加载我的一些 React 组件。

问题是,当通过延迟加载请求块时,我收到以下错误:

未捕获的错误:加载块 1 失败。(错误:http://localhost:58988/1.chunk.js)在 HTMLScriptElement.onScriptComplete (bootstrap:114)

这个块生成成功并且没有任何错误,只是路径错误 -http://localhost:58988/1.chunk.js应该是http://localhost:58988/dist/1.chunk.js

我也尝试使用最新React.lazy的延迟加载react组件,但我遇到了同样的问题。那么,有没有办法告诉编译器如何解析这些文件路径?

这是一些代码:

我的组件.tsx

import * as React from "react";
import * as ES5Promise from "promise";

export class MyComponent extends React.Component<{}, {}> {
    constructor(props) {
        super(props);
    }

    private readonly onLoadModuleClickHandler = (): void => {
        import("./Button").then((module) => {
            console.log(module);
        });
    }

    render() {
        return (
            <div>
                <input type="button" value="Load another module" onClick={this.onLoadModuleClickHandler}/>
            </div>
        );
    }
}

配置文件

{
  "compilerOptions": {
    "moduleResolution": "node",
    "noImplicitAny": false,
    "noEmitOnError": true,
    "module": "esnext",
    "removeComments": false,
    "sourceMap": false,
    "target": "es5",
    "jsx": "react",
    "noEmit": true,
    "importHelpers": true,
    "lib": ["dom", "es5", "es2015.promise"]
  },
  "exclude": [
    "node_modules",
    "wwwroot"
  ]
}

webpack.config.js

module.exports = {
    entry: {
        "app": "./src/App.tsx"
    },
    output: {
        path: path.resolve(__dirname, 'wwwroot/dist'),
        filename: "[name].bundle.js",
        chunkFilename: "[name].chunk.js"
    },
    module: {
        rules: [
            {
                test: /\.(ts|tsx)?$/,
                use: "awesome-typescript-loader",
                exclude: /node_modules/
            },
            {
                test: /\.(css|less)?$/,
                use: [{
                    loader: "style-loader"
                }, {
                    loader: "css-loader?modules&localIdentName=[local]--[hash:base64:5]"
                }, {
                    loader: "less-loader"
                }]
            },
        ]
    },
    resolve: {
        extensions: [".js", ".jsx", ".ts", ".tsx", ".css", ".less"]
    }
};
6个回答

发生这种情况是因为output.publicPath默认情况下是/.

只需更新output.publicPath以指向您想要的位置 => /dist/

非常感谢,这为我修好了!
2021-05-23 18:23:03
这确实需要在拆分块文档中,因为正如@icl7126 所说,这并不总是/默认情况下,可能会令人困惑。
2021-05-27 18:23:03
在我的情况下,实际上设置值来/解决问题。我想默认值并不总是/. 谢谢!
2021-06-04 18:23:03
ChunkLoadError: Loading chunk 'xx' failed当我将脚本存储在 cdn 上时,我不断收到此错误
2021-06-05 18:23:03

重新部署时,在重建应用程序包时,请确保不要使用先前的块文件清理输出文件夹,因为已经加载了应用程序的用户将尝试获取不再存在的先前块文件。

最好的办法是进行应用版本跟踪(使用 AJAX 并从数据库或动态配置文件中读取),当应用检测到更新版本时,向用户发送消息并要求他们重新加载页面。

+1 这有点晚了,但我仍然记得当我真正让这些块工作时,我在尝试延迟加载组件时经常会收到 404 错误,所以最终我选择退出这种延迟加载的东西。当我现在阅读您的评论时,它实际上很有意义。
2021-06-13 18:23:04
我过去遇到过这样的问题,今年看到你的问题后我写了这个很晚的答案,让人们知道生成的块文件不应该被删除,或者至少在新的块文件构建后保留很长时间. 是的,如果你仔细想想,这是有道理的。人们在加载应用程序时将使用捆绑映射运行应用程序,因此如果此映射中的某些文件被删除,那么当延迟加载启动并尝试获取它们时,他们将收到 404。
2021-06-18 18:23:04

让我们不要过分关注这一点。它只发生了5次。我相信有人已经打开了一段时间的 excalidraw,我们在此期间推送了一个新版本,当他们回来时,他们尝试加载一些懒惰的块,但它不再存在。

我们在 Facebook 解决这个问题的方法是我们将之前的块保留在我们的 CDN 上一周,如果捆绑包超过一周,我们会强制为人们刷新应用程序。这样我们就不需要关心非常旧的版本了。

以上回复来自vjeux

我的计划是监听window.onerror中的错误,然后提示用户刷新浏览器。

window.addEventListener('error', e => {
  // prompt user to confirm refresh
  if (/Loading chunk [\d]+ failed/.test(e.message)) {
    window.location.reload();
  }
});
可以确认,如果您不使用计数器,我们尝试了这个技巧并让我们陷入无限循环。
2021-05-24 18:23:04
想知道这是否会导致无限循环......猜测可能需要像本地存储计数器或其他东西,因为页面可能不需要大部分工作
2021-06-10 18:23:04

在我的情况下window.addEventListener('error'...没有触发所以我重写console.error方法如下:

    const origin = console.error;
    console.error = (error) => {
      if (/Loading chunk [\d]+ failed/.test(error.message)) {
        alert('A new version released. Need to relaod the page to apply changes.')
        window.location.reload();
      } else {
        origin(error);
      }
    }

output.publicPathin的值设置webpack.config.js/将解决问题。