8

由于我在 Java 中进行大部分编程,我发现在 Node.js 模块中导出类而不是对象实例是很有吸引力的,例如:

class Connection {
    constructor(db) {
        this.db = db;
    }
    connect(connectionString) {
        this.db.connect(connectionString);
    }
}
exports.Connection = Connection;

由于这样做需要跨依赖模块多次实例化该类,因此我仍然需要导出一个已经存在的实例以用于其余的生产代码。我在同一个模块中执行此操作:

exports.connection = new Connection(require("mongoose"));

这允许一些可测试性,因为可以在测试中交换真正的依赖关系:

const Connection = require("./connection").Connection;

describe("Connection", () => {
    it("connects to a db", () => {
        const connection = new Connection({connect: () => {}});
        // ...
    });
});

这种方法有效,但它有一种奇怪的感觉,因为我在这里混合了两种模式:导出原型(用于单元测试)和实例(用于生产代码)。这可以接受吗?我应该继续这样做还是换成不同的东西?如果是这样,首选模式是什么?

4

1 回答 1

0

你是对的,这是一种糟糕的编码风格,但实际上你可以编写一个函数,根据接收到的参数,返回单个实例(用于整个应用程序)或类本身(用于测试)。像这样的东西:

class MyClass() {}

const instance = new MyClass();

function getInstanceOrClass(isTesting) {
    if(isTesting) {
        return MyClass;
    } else {
        return instance;
    }
}

exports.getInstanceOrClass = getInstanceOrClass;

// in other files

const getInstanceOrClass = require('./yourFileName');

const classSingletonInstance = getInstanceOrClass();

// in test files

const getInstanceOrClass = require('./yourFileName');

const MyClass = getInstanceOrClass(true);
于 2022-01-19T15:52:07.823 回答