webpack导入的模块不是构造函数

Lir*_*n H 9 javascript babel npm webpack es6-modules

我创建了一个小JS模块,打算制作一个npm包,但目前仅在GitHub上。该模块是用ES6和SCSS编写的,因此依赖于webpack和babel进行编译。

为了测试它,我创建了一个具有类似设置(webpack和babel)的单独项目。npm安装模块后,尝试将其导入到index.js中时,我在Chrome开发者工具中收到以下错误:(其中x为模块名称)

index.js:11 Uncaught TypeError: x__WEBPACK_IMPORTED_MODULE_1___default.a is not a constructor
    at eval (index.js:11)
    at Object../src/index.js (main.js:368)
    at __webpack_require__ (main.js:20)
    at eval (webpack:///multi_(:8081/webpack)-dev-server/client?:2:18)
    at Object.0 (main.js:390)
    at __webpack_require__ (main.js:20)
    at main.js:69
    at main.js:72
Run Code Online (Sandbox Code Playgroud)

我浏览了无数的答案,并尝试了无数的解决方案,但没有成功。我的模块的设置如下。

.babelrc

{
  "presets": [
    ["env", {
      "targets": {
        "browsers": ["ie >= 11"]
      }
    }]
  ],
  "plugins": [
    "transform-es2015-modules-commonjs",
    "transform-class-properties"
  ]
}
Run Code Online (Sandbox Code Playgroud)

webpack.common.js

const path = require('path')
const ExtractTextPlugin = require('extract-text-webpack-plugin')
const HtmlWebpackPlugin = require('html-webpack-plugin')
const cleanWebpackPlugin = require('clean-webpack-plugin')

const baseSCSS = new ExtractTextPlugin('main/_base.css')
const themeSCSS = new ExtractTextPlugin('main/_theme.css')

module.exports = {
  entry: {
    example: [
      path.join(__dirname, 'src', 'example', 'index.js')
    ],
    main: [
      'idempotent-babel-polyfill',
      path.join(__dirname, 'src', 'index.js')
    ]
  },
  output: {
    path: path.join(__dirname, 'dist'),
    filename: path.join('[name]', 'index.js')
  },
  module: {
    rules: [
      {
        test: /\.js$/,
        exclude: /node_modules/,
        use: {
          loader: 'babel-loader',
        }
      },
      {
        test: /\.scss$/,
        use: ExtractTextPlugin.extract(
          {
            fallback: 'style-loader',
            use: ['css-loader', 'sass-loader']
          }
        )
      },
      {
        test: /\_base-scss$/,
        use: baseSCSS.extract(
          {
            fallback: 'style-loader',
            use: ['css-loader', 'sass-loader']
          }
        )
      },
      {
        test: /\_theme-scss$/,
        use: themeSCSS.extract(
          {
            fallback: 'style-loader',
            use: ['css-loader', 'sass-loader']
          }
        )
      }
    ]
  },
  plugins: [
    new cleanWebpackPlugin('dist', {}),
    new ExtractTextPlugin({ filename: path.join('example', 'style.css') }),
    baseSCSS,
    themeSCSS,
    new HtmlWebpackPlugin({
      inject: false,
      hash: true,
      template: path.join(__dirname, 'src', 'example', 'index.html'),
      filename: path.join('example', 'index.html')
    })
  ]
}
Run Code Online (Sandbox Code Playgroud)

webpack.prod.js

const merge = require('webpack-merge')
const UglifyJSPlugin = require('uglifyjs-webpack-plugin')
const webpack = require('webpack')
const common = require('./webpack.common.js')

module.exports = merge(common, {
  plugins: [
    new UglifyJSPlugin({
      sourceMap: true
    }),
    new webpack.DefinePlugin({
      'process.env.NODE_ENV': JSON.stringify('production')
    })
  ],
  mode: 'production'
})
Run Code Online (Sandbox Code Playgroud)

package.json

{
  "name": "my-module-name",
  "version": "1.0.0-beta.1",
  "description": "",
  "main": "dist/main/index.js",
  "scripts": {
    "start": "webpack-dev-server --config webpack.dev.js",
    "server": "node src/server",
    "format": "prettier-standard 'src/**/*.js'",
    "lint": "eslint src",
    "build": "webpack --config webpack.prod.js",
    "test": "echo \"Error: no test specified\" && exit 1"
  },
  "author": "Liran",
  "license": "ISC",
  "devDependencies": {
    "babel-core": "^6.26.0",
    "babel-eslint": "^8.2.3",
    "babel-loader": "^7.1.4",
    "babel-plugin-transform-class-properties": "^6.24.1",
    "babel-plugin-transform-es2015-modules-commonjs": "^6.26.2",
    "babel-preset-env": "^1.7.0",
    "clean-webpack-plugin": "^0.1.19",
    "css-loader": "^0.28.11",
    "eslint": "^4.19.1",
    "extract-text-webpack-plugin": "^4.0.0-beta.0",
    "html-webpack-plugin": "^3.2.0",
    "idempotent-babel-polyfill": "^0.1.1",
    "node-sass": "^4.9.0",
    "prettier-standard": "^8.0.1",
    "sass-loader": "^7.0.1",
    "style-loader": "^0.21.0",
    "uglifyjs-webpack-plugin": "^1.2.5",
    "webpack": "^4.6.0",
    "webpack-cli": "^2.0.15",
    "webpack-dev-middleware": "^3.1.3",
    "webpack-dev-server": "^3.1.3",
    "webpack-merge": "^4.1.2"
  }
}
Run Code Online (Sandbox Code Playgroud)

任何帮助/指针将不胜感激。如果您需要更多信息,请告诉我。

Dav*_*oun 40

如果您不是库的作者并且在使用另一个库时遇到问题,您可能会看到如下错误:

TypeError: [LIBRARY_NAME]__WEBPACK_IMPORTED_MODULE_3__ is not a constructor
Run Code Online (Sandbox Code Playgroud)

如果是这种情况,您可能在代码中错误地导入了库(这可能是默认导出的问题)。仔细检查库文档以了解使用情况。

这可能就像改变这个一样简单:

TypeError: [LIBRARY_NAME]__WEBPACK_IMPORTED_MODULE_3__ is not a constructor
Run Code Online (Sandbox Code Playgroud)

对此:

import Foo from 'some-library/Foo';
Run Code Online (Sandbox Code Playgroud)

  • @brance 当然。第一个没有大括号的示例称为“默认导出”,其中整个导出库可通过“Foo”变量使用(例如“Foo.doSomething()”)。第二个带有大括号的示例是“模块导出”模式,其中模块是从库中逐个明确挑选的,这对于 treeshaking 来说是理想的。这个想法是,如果一个库有 100 个模块,而您只想使用 1 个,则此模式使其变得明确,因此模块捆绑器知道他们不需要引入其他 99 个模块,就像使用默认导出示例一样。 (5认同)
  • @DavidCalhoun 非常感谢您的解释! (2认同)

Pla*_*256 10

它不起作用,因为它缺少libraryTarget and library属性。通过该webpack知道要创建哪种模块格式,即:commonjs(module.exports)或es(export)。

我会做类似的事情:

...
  output: {
    path: path.join(__dirname, 'dist'),
    filename: path.join('[name]', 'index.js'),
    library: "my-library",
    libraryTarget: "umd" // exposes and know when to use module.exports or exports.
  },
...
Run Code Online (Sandbox Code Playgroud)


Bur*_*Leo 6

除了设置 之外libraryTarget,可能还需要将exportJavaScript 文件中的 移至默认值。

function MyClassName() {
  ...
}

export default MyClassName;
Run Code Online (Sandbox Code Playgroud)

然后在 webpack 配置中库类型umd...

(请注意,我使用了较新的library.type而不是较旧的libraryTarget(请参阅https://webpack.js.org/configuration/output/#outputlibrarytarget)。

 const path = require('path');
 
 module.exports = {
    mode: "production",
    entry: '../wherever/MyClassName.js',
    
    output: {
        library: {
          name: "MyClassName",
          type: "umd",  // see https://webpack.js.org/configuration/output/#outputlibrarytype
          export: "default",  // see https://github.com/webpack/webpack/issues/8480
        },
        filename: 'MyClassName.min.js',
        path: path.resolve(__dirname, '../wherever/target/')
    },
    optimization: {
        minimize: true
    }
 };
Run Code Online (Sandbox Code Playgroud)

export default使得该类在 JavaScript 中可用,就像直接嵌入文件一样,即

<script type="text/javascript" src="MyClassName.min.js"></script>
<script type="text/javascript">
<!--

var myInstance = new MyClassName();

// -->
</script>
Run Code Online (Sandbox Code Playgroud)

免责声明:我添加了这个答案,尽管原来的问题已经存在三年了。在遇到“不是构造函数”问题后,我花了几个小时才找到解决方案default。那是第二次,我搜索并找到了它:D