1

函数编码类型(即嵌套柯里化函数)在 Javascript 中有一些缺点:

  • 它们在开发控制台中的表示是模糊的(例如[Some(5), None]显示为[f, f]
  • 没有什么能阻止您将组合器应用于错误的类型(例如eitherMap(f) (Some(5))
  • 你不能检查他们的自由变量
  • 你甚至不能躲避打字

这些缺点使它们在现实世界的应用中毫无用处。

我想知道是否有办法克服这些缺点并想出了以下草图:

const tag = (name, x, k) => {
  const Cons =
    Function(`return function ${name}() {}`) ();

  Cons.prototype[Symbol.toStringTag] = name;
  Cons.prototype["run" + name] = k;
  Cons.prototype.tag = name;

  const o = new Cons();
  Object.defineProperty(o, "value", {get: () => x});
  return o;
};

const Some = x =>
  tag("Option", x, def =>
    tag("Option", x, k => k(x)));

const None = tag("Option", null, def =>
  tag("Option", def, k => def));

const option = def => k => fx =>
  fx.runOption(def).runOption(k);

const safeInc = option(0) (n => n + 1);

safeInc(Some(5)); // 6
safeInc(None); // 0

const xs = [Some("foo"), None]; // [Option, Option]

/* 
  expanded dev console display:

  1: Option {value: ...} --> expands to "foo"
  2: Otpion {value: ...} --> expands to null
*/

请注意,我对原型继承根本不感兴趣。

这种方法既乏味又可能很慢,因为我应用了Function构造函数,这使得代码更难预测。有没有更好的方法给 curried 函数一个类型(或者更确切地说是 JS 中的标签),以便消除列出的缺点?

4

1 回答 1

1

我稍微改进了我的方法,摆脱了Function调用和重复创建构造函数。它适用于我的特定用例(函数编码类型),但我未能解决更普遍的任意函数的柯里化形式,因为它仍然太乏味。无论如何,这里是:

const tag = Cons => (k, ...args) => {
  const o = new Cons();

  Object.defineProperties(o, {
    "value": {get: () => args},
    "runOpt": {value: k}});

  return o;
};


const Opt = tag(
  function Option() {
    Option.prototype[Symbol.toStringTag] = "Option";
    Option.prototype.tag = "Option";
  });


const Some = x =>
  Opt(def => Opt(k => k(x), x), x);

const None = Opt(def => Opt(k => def, def));

const option = def => k => fx =>
  fx.runOpt(def).runOpt(k);

const safeInc = option(0) (n => n + 1);

safeInc(Some(5)); // 6
safeInc(None); // 0

Some(5); // Option {runOption}
[Some("foo"), None]; // [Option, Option]

/*
  expanded dev console display:

  1: Option {runOpt: f, value: ...} --> expands to ["foo"]
  2: Otpion {runOpt: f, value: ...} --> expands to []
*/
于 2018-11-25T20:21:17.140 回答