-
Notifications
You must be signed in to change notification settings - Fork 25
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Argument against this proposal #39
Comments
Wouldn't this very proposal let you do the same thing? return foo |> bar(?) |> baz(?);
In some functional languages, currying happens by default. This proposal is just trying to make that more natural to write. An example of a curry-by-default language let add a b = a + b
let increment = add 1
printfn "%d" (increment 8) Simulating the same behaviour with this proposal const add = (a, b) => a + b
const increment = add(1, ?);
console.log(increment(8))
I think that example is broken. |
There are cases where being explicit with the helper function argument names is warranted and there are cases where I think they bring little to no value. const parsedValues = userInputs.map(n => parseInt(n))
const parsedValues = userInputs.map(parseInt(?)) I don't think the latter of those is any less readable than the former. On the contrary, I find the extra const log = console.log({ toString() { return [${new Date()}] } }, ?); That example just creates a { toString() { return `[${new Date()}]` } } and second argument is passed in later when const log = console.log(`[${new Date()}]`, ?) would evaluate that eagerly and all |
Personally I liked partial application, but I think the example of const log = console.log({ toString() { return [${new Date()}] } }, ?); vs const log = x => console.log(`[${new Date()}]`, x) Even the partial application version worked as expect, I think the arrow function version is much easy to write, read and understand. So maybe we'd better think the whole things again, do the proposals (not only F# style + partial application, but also Hack/smart style) really add enough benefit? |
Whenever there's a new proposal, comments start popping where rationales such as this creep in:
Well, you'd have an idea if you'd learn it as part of the language. I mean, look at this code.
What is Heck, even
Of course it took you a second. It takes everyone a second because this is the first time we're looking at this new syntax. It used to take me more than a second to understand what It took me a second to realize exactly what I understand that discussing the syntax and pointing out footguns is a good thing, but the whole rationale here is basically "I'd rather write an arrow function because I'm used to it and there are cases where it makes code easier to understand". For example, I have no idea what Hack is and seeing this
I'm so confused. What's
Funny to say that since you use anonymous functions in the code. Why not swallow the bullet and write a good name for a function?
Now, back to
What's
|
@lazarljubenovic thanks for your thoughtful comment which clearly showcases that I have not explained my argument well enough. I'll try to elaborate and perhaps it'll help get the point across. My point isn't that That is not my point. My point is that in an expression: a |> b(#) I know what is being passed to In an expression b(?) I don't know what is or can be passed to This is why it's beneficial, for code readability, to have the expression written as someName => b(someName) because this gives me a better chance to understand the code. So I hope this clarifies my point. Now to your last point, which boils down "You don't have to use the syntax when it's added", this is not really a valid argument for a language syntax discussion. One could use this argument for any backwards-compatible piece of syntax. If we followed this logic we would end up with a language with an infinite number of features. Besides the cost associated with each feature (complications to compilers and interpreters, and to tooling), the most important counter argument is that I do not choose how others will write code. So the possibility of writing a piece of code in less confusing syntax is not an argument for adding that confusing piece of syntax. |
@xixixao Instead of comparing |
@ducaale I am specifically calling out that this proposal should not move forward if the Hack-style pipeline proposal (or for that matter the smart pipeline proposal) moves forward. So my only concern is the non-pipeline use of this new syntax. |
@xixixao Partial Application is a good addition and can stand on its own. But if BOTH pipeline styles move forward, then partial application becomes more important: tc39/proposal-pipeline-operator#167 (comment) |
@xixixao My interpretation of this argument is that
Is that interpretation correct? If so, I strongly disagree. On point 1.I don't see this being a real issue in practice. When a function is in-scope, it's because it's either:
If For example: export const multiplyInt = (x, y) => {
if (typeof x !== 'bigint' || typeof y !== 'bigint') {
throw new TypeError('multiplyInt only accepts integers');
}
return x * y;
};
import { multiplyInt } from './multiplyInt.js';
// `double` is ambiguous; `n` implies the argument could be a `number`
const double = n => multiplyInt(2, n);
console.log(double(4));
// TypeError: multiplyInt only accepts integers In this case, I'd argue that the partial application proposal would remove the ambiguity: // `double` is clear; it accepts only what `multiplyInt` accepts as its second argument
const double = multiplyInt(2, ?); On the other hand, if For example: export const handleEach = (handler, iterable) => {
for (const item of iterable) {
// No way to know what `handler` accepts here (or even if its a function, really) without looking at the call sites of `handleEach`
handler(item);
}
}; Overall, the concern of On point 2.I don't agree that it should be mandatory to explicitly name the parameters when defining a function. Firstly, as I alluded to above, when you define a function In other words, the partial application proposal allows you to make less assumptions when defining certain classes of functions. Secondly, the declarations
In other words, the omission of the parameter name can itself be an important piece of information. Thirdly, it's already possible to define functions without naming the parameters. For example: const bar = foo.bind(null, x); This is already something that many developers (including myself) do with varying degrees of regularity. There are notable downsides to this approach, however:
In other words, we can already define functions in this manner, however the means of doing so results in less clarity and less usability than the partial application proposal. Finally, requiring explicit parameter names when defining functions carries the assumption that these names are the only (or even the primary) means by which developers describe function contracts. With the emergence of static typing in the JS ecosystem, this is no longer the case. TypeScript for instance allows for the definition of functions with highly specific, strongly enforced contracts that go well beyond simple naming. In a statically typed world, parameter names don't matter nearly as much for either code clarity or code correctness. For example: // `multiplyInt` is of type `(x: bigint, y: bigint) => bigint`
const multiplyInt = (x: bigint, y: bigint): bigint => x * y;
// `double` would be inferred as type `(y: bigint) => bigint`
const double = multiplyInt(2, ?); In other words, the lack of explicit parameter names is even less likely to be a problem in a statically typed world, allowing TypeScript users to get the benefits of the partial application proposal with even less potential downsides. On point 3.For what it's worth, I personally prefer point-free style in my functional code, so I disagree with avoiding it. Frankly, preferences around things like point-free style are almost entirely subjective, so debating them is questionable IMO. What I will say though is that you personally not liking or not seeing value in point-free style doesn't mean that others feel the same. Nor does it mean that a partial application operator wouldn't be a useful or "good" addition to the language. There's something to be said for enabling alternative programming styles and increasing language diversity, even if you personally don't have an interest in those particular areas. |
While I agree entirely with your arguments (in fact I maintain a library dedicated to FP in JS and remain a big fan of point-free programming), I would like to note that there is also something important to be said against "enabling alternative programming styles and increasing language diversity." That is the matter of adding complexity. There is a tricky balance to maintain between adding useful features and bloating a language with too many competing ideas. I come down firmly on the side of this being a useful feature, in this guise or some alternative. But it does add conceptual weight to the language and makes it some degree more difficult to add future features. We should always keep this in mind alongside our considerations of utility. |
@CrossEye, when the context of this issue is about not needing partial application if smart/hack pipelines are added, then it is true, there will be one additional syntax. On the other hand, if you consider that with both styles of pipelines, every programmer needs to learn practically the same concept weight anyway, the syntax cost by itself is negligible. There will actually be confusion if users see the partial application pattern and find out that it isn't usable outside pipelines. Pipelines without any kind of partial application capability would be unwieldy for all but few coding styles, strongly favouring them. Prematurely killing off partial application proposal would deal a severe blow to minimal & F# style pipelines. Just like getting partial application to the spec first would favour both minimal & F# style pipelines. @xixixao, attacking one part of two complementary proposals by claiming that another, competing proposal, if implemented, would take care of it, is in my opinion an unfair move. The fact that F# pipelines and partial application don't technically require each other means that passing it through TC39 as a single proposal would be unlikely. I believe that either success of Hack/Smart pipelines or failure of partial application proposal likely results in the failure of both sides of the F#+partial application combo. I'm ready to see this thread again when something passes TC39 stage 3 or 4. Just a chance of it happening shouldn't be enough! |
I enjoy the point-free style because the code I write leans functional even in JS. When you're doing FP functions are everything. They take the place of objects and are just as easily configured. For example, a function with 3 arguments would likely take some payload (the primary bit of data the function acts upon) and two arguments configuring the details of that action. When this is prevalent in what you do, as it is in what I do, there is no tool so useful as partial application. You plug functions and compose them together in and out of pipelines. It is easy to write a For what it's worth, though I know many have argued of the importance of naming vars, I find that in a functional style, functions tend to be tiny and obvious and that tiny names work just fine. That is, I have never felt I suffered from losing the descriptive var names as cited in the OP. I find that I become familiar with the functions I use because of their frequent use, and for less familiar functions I look them up when needed. Partial application syntax is a greater boon to my daily programming than the pipeline operator is, although that's a close second. |
I am trying to revive pipeline proposal and have it get going with Hack-style only version. See tc39/proposal-pipeline-operator#167 (comment)
I'd like to add for your consideration why I think we should not be moving forward with this proposal - not any of its particular variants, but in general (if we decide to move forward with any Hack-style pipeline supporting proposal). This expands on the point 5 in the comment linked above.
What are Pipelines?
First, let me explain the point of the pipeline operator: It allows you to skip naming things. Every pipeline can be rewritten:
like this:
Now if that's the case, you might be wondering what's my gripe with this proposal. What's the difference, isn't this proposal also allowing you to skip naming things?
It is! But with one major, major difference.
In the case of the pipeline, you can see what you're not naming.
I can see
foo
andbar(#)
in the example above, so I have some idea of whatbar(foo)
is.A more realistic example from Hack:
It doesn't help this code to add variables
unique_test_users_not_sorted
andunique_test_users_sorted
. I can see what's going on just as well from the pipeline code.Partial Application
Now let's look at this proposal:
What is
?
? I have no idea. I'll have to inspectfoo
to find out, orfooBar
needs to be superbly named. What's the alternative? Functions!Even a super undescriptive name like
n
gives me more information (the argument is probably a number).More examples directly from readme:
I have no idea what the second arg is intended to do.
I have no idea what is being passed to
onClick
(if I pretend I don't know whataddEventListener
does - but even knowing it took me a second to wrap my head around this). Now I try to remember what coding was like when I got started, and I really would not want to be learning from such code (or wishing it upon anyone else).There are other arguments against this proposal, like its weirdly restricted nature to only some forms of expressions (why not others? we could argue for years).
Just like in the discussion on pipelines I will advise: Swallow the bullet and add those 5 characters. If you're not writing an entirely throw-away code you will want to have a good name for that argument. This proposal is way too complicated for the little benefit of saving those 5 characters in the few special cases it applies.
In summary, this proposal got started to support pipelines, pipelines don't need it if we go with Hack-style only, and in that case we should drop this proposal.
cc @codehag @ducaale @aadamsx who are advocating the other direction in #36
The text was updated successfully, but these errors were encountered: