Feature #15897
closed`it` as a default block parameter
Description
How about considering "it" as a keyword for the block parameter only if it is the form of a local varaible reference and if there is no variable named "it"?
[1, 2, 3].map { it.to_s } #=> ["1", "2", "3"]
If you are familiar with Ruby's parser, this explanation is more useful: NODE_VCALL to "it" is considered as a keyword.
Examples:
public def it(x = "X")
x
end
[1, 2, 3].map { it.to_s } #=> ["1", "2", "3"]
[1, 2, 3].map { self.it } #=> ["X", "X", "X"] # a method call because of a receiver
[1, 2, 3].map { it() } #=> ["X", "X", "X"] # a method call because of parentheses
[1, 2, 3].map { it "Y" } #=> ["Y", "Y", "Y"] # a method call because of an argument
[1, 2, 3].map { it="Y"; it } #=> ["Y", "Y", "Y"] # there is a variable named "it" in this scope
it = "Z"
[1, 2, 3].map { it.to_s } #=> ["Z", "Z", "Z"] # there is a variable named "it" in this scope
Pros:
- it is the best word for the feature (according to @matsuda (Akira Matsuda))
- it is reasonably compatible; RSpec won't break because their "it" requires an argument
Cons:
- it actually brings incompatibility in some cases
- it is somewhat fragile; "it" may refer a wrong variable
- it makes the language semantics dirty
Fortunately, it is easy to fix the incompatible programs: just replace it
with it()
. (Off topic: it is similar to super()
.)
Just inserting an assignment to a variable "it" may affect another code. This is a bad news, but, IMO, a variable named "it" is not so often used. If this proposal is accepted, I guess people will gradually avoid the variable name "it" (like "p").
The dirtiness is the most serious problem for me. Thus, I don't like my own proposal so much, honestly. But it would be much better than Perlish @1
. (Note: I don't propose the removal of @1
in this ticket. It is another topic.) In any way, I'd like to hear your opinions.
An experimental patch is attached. The idea is inspired by @jeremyevans0's proposal of @
.
P.S. It would be easy to use _
instead of it
. I'm unsure which is preferable.
Files
Updated by mame (Yusuke Endoh) over 5 years ago
- Related to Misc #15723: Reconsider numbered parameters added
Updated by Hanmac (Hans Mackowiak) over 5 years ago
_
can't be used as default block parameter because it already has a special meaning when using block variables like {|a,_,_,b| }
Updated by mame (Yusuke Endoh) over 5 years ago
@hanmac,
You cannot use both it
and the ordinal parameter |a,_,b|
simultaneously. It will cause a SyntaxError like this.
$ ./miniruby -e '1.times {|a,_,b| it }'
-e:1: ordinary parameter is defined
Updated by Hanmac (Hans Mackowiak) over 5 years ago
@mame (Yusuke Endoh) you got me wrong, i mean you might not use _
for this like you said in your P.S.
Updated by shevegen (Robert A. Heiler) over 5 years ago
I was about to write a very lengthy reply, but I think it would be too difficult to read for others, so this is a somewhat shorter variant. So just the main gist:
(1) I don't quite like the name "it", mostly due to semantics (the name does not tell me much at all), but one advantage is that "it" is short to type. I do not have a good alternative name, though. _ as a name would be even shorter, and avoids some of the semantic-meaning problem, but may have other small issues - see a bit later what I mean here.
(2) Even though I do not like the name "it", to me personally, it would be better to see that BOTH @1 @2 and "it" would be added, in the sense that then people could just pick what they prefer. Obviously I see no problem with @1 @2 at all, so I am biased. But that way people could just use whatever they prefer. (There could be another name, of course ... I thought about some way for acessing block/procs data ... such as BLOCK or ProcData or something like that. One problem is that this is all longer to type than e. g. "it" or @1 but perhaps we could use some generic way to access what a proc/block represents anyway, even aside from the proposal itself, similar to ... LINES or method or FILE or so. Just so that we may have a concept for it; although it will probably not be widely used. But I have not really thought about this much. Note that I was wondering about something like this for case/when structures as well, so that we could access them more easily, also from "outside" of methods, but I digress here).
IF only "it" alone were to be added, then I would rather prefer that neither @1 @2 nor "it" would be added, and we'd all only use the oldschool way (which is fine, too). But as said, I am biased so I think @1 @2 etc... are perfectly fine.
I think this is also a problem I have with the "this is perl" comments in general - to me it is not like perl anywhere. And the old variant such as:
foo.each {|a,b,c|
just continue to work fine; so to me the change is primarily about adding more flexibilty. This was a problem I have had with the use cases that were mentioned - people would be very eager to point out what they dislike (understandably so), but at the same time would not want to mention use cases that may be useful. So I think in general, it would be better to be more objective in giving examples, even if a certain functionality is disliked. Use cases should ideally be extensive, not just focusing on what the ruby user at hand may dislike the most (since they may tend to focus on that first, which is understandable, and then ignore anything else; I tend to do so myself sometimes).
To clarify this - my primary problem with "it" is the name itself, not the functionality that is associated with it.
Using _ avoids the name/semantic issue a bit, to some extent, but I think _ has some slight other issues.
For example, the _ variable is used quite a lot in ruby code out there, or at the least in some code bases, so I am not sure it would be a good name here. Note that I use _ as "throwaway" variable a lot in my own code. This should be kept in mind, at the least for when ruby users do something similar (I have no idea whether it is common or not, though). I think hanmac sort of reffered to this too in a way.
Since I like _ a lot, I'd rather see "it" be added than _, because I will most likely not use "it" in my own code ;D , whereas I would still use _ fine, possibly even within blocks, and possibly @1 @2 too, at the least for quick debugging, if it were to stay/remain, which I hope it will. But I am biased. ;)
I should also note that while I think @1 @2 are perfectly fine, I also don't have a big problem if it were not to be added permanently, even though I think it is fine if it would, evidently. The oldschool way is the best.
Since I see @1 @2 mostly as a convenience feature, though, I can continue to work with ruby just fine. I just don't think that all prior statements in particular in the other thread(s) made a whole lot of sense; and I have no problem if "it" would be added either as well, since I can avoid it, and just use @1 @2 for debugging. ;)
I think, realistically, I assume that most ruby users will continue to just use ruby code like it used to be, like:
foobar.some_method {|a, b, c, d, _, f|
I am quite certain that I will keep on using the above variant, and that neither "it" nor @1 @2 would persist in my own code - but for quick debugging, in particular for longer names, I think @1 @2 is really great. I don't think "it" would be equivalent to this, though; at the least to me, "it" is not the same as e. g. @1 @2 in several ways. But as said, I have no problem at all if both variants would be added.
Of course I am not naive - when features are added/offered, people will use it and play around with it; adults are kids after all, just play with different things. ;) I just don't think that the primary focus for dislike should be limited to just some use cases, without considering situations such as e. g. @1 @2 not be used in production code, but just for debugging purposes alone. I don't have a
problem with the scenario where we can avoid naming parameters, but to me this is not the primary use case I would like to focus myself - for me the "pp @1; pp @3" variant really is the more important aspect of the suggestion. When you come from this point of view then I think it is easy to understand that "it", aside from the name, is not exactly the same.
Last but not least, as mame wrote - I think if you have not yet commented on either @1 @2 (in other issues) and/or "it" (here in this proposal), it may be good to comment on the suggestion/idea itself here. Matz actually asked for feedback before, not only in the other thread but also the old(er) ones predating these.
Updated by phluid61 (Matthew Kerwin) over 5 years ago
Hanmac (Hans Mackowiak) wrote:
@mame (Yusuke Endoh) you got me wrong, i mean you might not use
_
for this like you said in your P.S.
Isn't {|_| _ }
no more or less conflicting than {|it| it }
? You can either use positional args, or a default arg, but not both. So {|_| _ }
means what it currently means, irrespective of this proposal.
Unless you mean there's a chance of an outer scope that already uses _
as an arg, creating a conflict in the inner scope?
Updated by phluid61 (Matthew Kerwin) over 5 years ago
shevegen (Robert A. Heiler) wrote:
I was about to write a very lengthy reply, but I think it would be too difficult to read for others, so this is a somewhat shorter variant.
Good grief.
Updated by mikegee (Michael Gee) over 5 years ago
RSpec won't break because their "it" requires an argument
Unfortunately this is not accurate. RSpec has a shorthand style like this:
subject { fortytwo } it { is_expected.to eq 42 }
Updated by jeremyevans0 (Jeremy Evans) over 5 years ago
mikegee (Michael Gee) wrote:
RSpec won't break because their "it" requires an argument
Unfortunately this is not accurate. RSpec has a shorthand style like this:
subject { fortytwo } it { is_expected.to eq 42 }
That's a block argument :). In any case, the parser treats it differently as NODE_ITER/NODE_FCALL, not as NODE_VCALL:
RubyVM::AbstractSyntaxTree.parse("it").children
# => [[], nil, #<RubyVM::AbstractSyntaxTree::Node:VCALL@1:0-1:2>]
RubyVM::AbstractSyntaxTree.parse("it{}").children
# => [[], nil, #<RubyVM::AbstractSyntaxTree::Node:ITER@1:0-1:4>]
RubyVM::AbstractSyntaxTree.parse("it{}").children.last.children
# => [#<RubyVM::AbstractSyntaxTree::Node:FCALL@1:0-1:2>, #<RubyVM::AbstractSyntaxTree::Node:SCOPE@1:2-1:4>]
Regarding the proposal itself, the dirtying of the semantics bothers me about this as well. However, I can see where people would find it
cleaner than @
in terms of syntax, so this is really a tradeoff between the cleanliness of semantics and syntax. I don't have a strong opinion on it
compared to @
, but I think either is preferable to @1
or _
.
Updated by Eregon (Benoit Daloze) over 5 years ago
I like the proposal and I think it reads nicer than @
.
It's a bit magical that giving it an argument changes the semantics, but that's somewhat similar to having a p
local variable, and I think it's worth the better readability and syntax.
Not so many people seem confused by p
so I guess it would not be too surprising and just intuitive in common cases.
I also like _
because _
is "unnamed" (rather than abstract like it
) and a "placeholder for the missing argument name" and this whole feature is about removing the need to name the block argument.
Showing them in code for an easy comparison:
[1, 2, 3].map { @ * 3 }
[1, 2, 3].map { @1 * 3 }
[1, 2, 3].map { _ * 3 }
[1, 2, 3].map { it * 3 }
[1, 2, 3].map { |n| n * 3 }
In the case of nested unnamed block arguments ([1].map { _ * 3.then { _ } }
), both _
and it
would refer to the outer block's argument, and consider the inner block(s) have no arguments.
That could be confusing, it might be worth warning or rejecting such cases, although they are probably rare.
But it would be much better than Perlish
@1
Strongly agreed.
Updated by shugo (Shugo Maeda) over 5 years ago
Thus, I don't like my own proposal so much, honestly. But it would be much better than Perlish @1.
I don't like both proposals, but I prefer @1
to it
because @1
looks ugly and may help prevent overuse.
Furthermore, the proposed it
may be more Perlish in the sense that it depends on the context.
Updated by Eregon (Benoit Daloze) over 5 years ago
shugo (Shugo Maeda) wrote:
I don't like both proposals, but I prefer
@1
toit
because@1
looks ugly and may help prevent overuse.
I think we should never purposefully introduce something ugly in the language.
Preventing overuse is I think best done by limiting to a single argument (as argued in #15723).
Updated by shugo (Shugo Maeda) over 5 years ago
Eregon (Benoit Daloze) wrote:
shugo (Shugo Maeda) wrote:
I don't like both proposals, but I prefer
@1
toit
because@1
looks ugly and may help prevent overuse.I think we should never purposefully introduce something ugly in the language.
So let's reject both proposals.
Preventing overuse is I think best done by limiting to a single argument (as argued in #15723).
I guess it
will be overused when a block takes only one argument.
Updated by Eregon (Benoit Daloze) over 5 years ago
shugo (Shugo Maeda) wrote:
I think we should never purposefully introduce something ugly in the language.
So let's reject both proposals.
That's not what I meant. I'd rather not have something ugly in the language at all.
But I think we can make it not ugly, either with _
or it
proposed here.
I think readability matters a lot to many people, we typically read code more often than we write.
_
or it
seem much better for readability than @
or @1
.
Preventing overuse is I think best done by limiting to a single argument (as argued in #15723).
I guess
it
will be overused when a block takes only one argument.
Maybe, but that harm would be IMHO very little, because it
and _
read nicely and easily,
compared to spreading multiple numbered Perlish variables in Ruby code.
Updated by janosch-x (Janosch Müller) over 5 years ago
Kotlin has implemented it
like this (docs).
From purely personal experience, after doing just a little bit of Kotlin, I often feel a temptation to use it
when writing Ruby, just to notice that I can't. I found it
in Kotlin natural, easy to get used to, and easy to parse visually and understand when re-reading my code after a while.
Updated by shugo (Shugo Maeda) over 5 years ago
Eregon (Benoit Daloze) wrote:
shugo (Shugo Maeda) wrote:
I think we should never purposefully introduce something ugly in the language.
So let's reject both proposals.
That's not what I meant. I'd rather not have something ugly in the language at all.
But I think we can make it not ugly, either with_
orit
proposed here.
it
doesn't look ugly at first glance, but it
makes the language semantics dirty as mame admitted in his proposal.
I think readability matters a lot to many people, we typically read code more often than we write.
_
orit
seem much better for readability than@
or@1
.
If it
is a normal reserved word, I agree with you.
However, the semantics of it
depends on the context, and therefore @1
is more readable for me.
Updated by sawa (Tsuyoshi Sawada) over 5 years ago
I propose to use a new keyword item
.
-
I feel that using a keyword spelt in letters is the right way here since keywords like
self
are used in other cases where we reach for things out of the blue without receiving them through argument signature. -
"Item" is close enough to "it", so we may achieve sympathy from some of the people opting for "it", but is not "it", so it does not have the problem that "it" has.
-
\item
is used in LaTeX as a command to introduce bullet points in listed structures, which is analogous to elements in a block led byeach
,map
and their kins.[1, 2, 3].map{item ** 2} # => [1, 4, 9]
-
At the same time, "item" does not exclusively mean "element". It is a neutral term regarding that. So it would not be unnatural to be used in blocks led by methods like
then
,tap
and their kins."foo".then{item + "bar" + item} # => "foobarfoo"
-
I have a concern that "it" somewhat implies the receiver since it means something whose referent has been fixed in the context. In fact, the method
itself
returns the receiver.
Updated by janosch-x (Janosch Müller) over 5 years ago
sawa (Tsuyoshi Sawada) wrote:
I propose to use a new keyword
item
.
I think that is a great proposal.
it
is nice to read when passed to methods of other objects or when used with binary operators:
strings.each { puts it }
pathnames.map { File.read it }
numbers.map { it + 2 }
unfortunately, it
is quite awkward to read when calling its own methods (which is probably the more common case in Ruby):
strings.each { it.chomp!('foo') }
pathnames.map { it.read }
numbers.map { it.next.next }
item
works well for both cases:
strings.each { puts item }
pathnames.map { File.read item }
numbers.map { item + 2 }
strings.each { item.chomp!('foo') }
pathnames.map { item.read }
numbers.map { item.next.next }
Updated by mame (Yusuke Endoh) over 5 years ago
Don't think that this proposal can be applied to any words. A common name is much more dangerous than a pronoun like it
because it is much more frequently used as a method name.
Actually, the count of "def item()" is 20 times more than "def it()" in gem-codesearch result.
$ csearch "^\s*def it\(?\)?$" | wc -l
12
$ csearch "^\s*def item\(?\)?$" | wc -l
225
Furthermore, I found some codes that will be broken if "item" becomes a soft keyword.
https://github.com/ginty/cranky/blob/ca7176da2b8e69c37669afa03fee1a242338e690/lib/cranky/job.rb#L49
https://github.com/carlosipe/mercado-libre/blob/ebb912a7c4e942eb38e649d8e11a005c288ebc92/test/mercadolibre.rb#L44
Updated by akim (Akim Demaille) over 5 years ago
FWIW, wrt "This is Perlish": at least one modern language has adopted a similar feature: Swift. I've used it in practice, and it is really nice to use.
This page has a running example expressed in different ways: https://docs.swift.org/swift-book/LanguageGuide/Closures.html
func backward(_ s1: String, _ s2: String) -> Bool {
return s1 > s2
}
var reversedNames = names.sorted(by: backward)
reversedNames = names.sorted(by: { (s1: String, s2: String) -> Bool in
return s1 > s2
})
This one shows the common inheritance from Smalltalk.
reversedNames = names.sorted(by: { s1, s2 in return s1 > s2 } )
reversedNames = names.sorted(by: { s1, s2 in s1 > s2 } )
reversedNames = names.sorted(by: { $0 > $1 } )
reversedNames = names.sorted(by: >)
Updated by Benoit_Tigeot (Benoit Tigeot) over 5 years ago
jeremyevans0 (Jeremy Evans) wrote:
mikegee (Michael Gee) wrote:
RSpec won't break because their "it" requires an argument
Unfortunately this is not accurate. RSpec has a shorthand style like this:
subject { fortytwo } it { is_expected.to eq 42 }
That's a block argument :). In any case, the parser treats it differently as NODE_ITER/NODE_FCALL, not as NODE_VCALL:
Thanks for the clarification. As member of the RSpec team but expressing my own view I was worried about a strong conflict. I still think this not a good idea to use "it" because it is used a lot in the tests your write using RSpec. It can easily lead to confusion. I think we should send other proposal for this. Sorry Mame.
Updated by Benoit_Tigeot (Benoit Tigeot) over 5 years ago
- File mame_its_proposal.patch mame_its_proposal.patch added
I added an updated patch version. The patch has been tested with existing examples that mame shared initially.
Also I tested the patch against rspec-core without any issues. https://github.com/rspec/rspec-core/issues/2645
Updated by Eregon (Benoit Daloze) over 5 years ago
Benoit_Tigeot (Benoit Tigeot) wrote:
Thanks for the clarification. As member of the RSpec team but expressing my own view I was worried about a strong conflict. I still think this not a good idea to use "it" because it is used a lot in the tests your write using RSpec. It can easily lead to confusion. I think we should send other proposal for this. Sorry Mame.
What do you think about _
?
I'm unsure if it's confusing in practice, they are used in fairly different contexts, and it
is never used alone (no args, no block) by RSpec AFAIK.
It's pretty clear in this example, isn't it?
describe "Array#map" do
it "should transform the values" do
expect([1,2,3].map { it * 2 }).to eq([2,4,6]) # RSpec 3+ style
[1,2,3].map { it * 2 }.should == [2,4,6] # MSpec/RSpec 2 style
end
end
Updated by JonRowe (Jon Rowe) over 5 years ago
👋Hello! As the current maintainer of RSpec I'm concerned about the confusion here.
RSpec.describe do
it "will do the thing" do
it # now refers to the first argument to the block? (Which ironically is the example itself)
end
it # creates a pending example with no implementation
end
So if I get a vote I'd love this to have a different name. As pointed out I think _
is confusing as it's being reused from elsewhere, but what about _1
or $1
or some other special syntax?
Updated by Benoit_Tigeot (Benoit Tigeot) over 5 years ago
Eregon (Benoit Daloze) wrote:
What do you think about
_
?I'm unsure if it's confusing in practice, they are used in fairly different contexts, and
it
is never used alone (no args, no block) by RSpec AFAIK.
I think Jon express my feelings with a clear example.
For _
same thing. It is already use and it can lead to an issue: https://bugs.ruby-lang.org/issues/15897#note-3
Updated by Eregon (Benoit Daloze) over 5 years ago
JonRowe (Jon Rowe) wrote:
it # creates a pending example with no implementation
Is that used in practice though? I know pending examples, but I would expect they at least have a description.
FWIW, MSpec's it
requires the description argument, so it cannot ever be ambiguous for the user.
https://github.com/ruby/mspec/blob/ca2bc422cd8f8ddb23629e972372cf8e49f992fc/lib/mspec/runner/object.rb#L14
Benoit_Tigeot (Benoit Tigeot) wrote:
For
_
same thing. It is already use and it can lead to an issue: https://bugs.ruby-lang.org/issues/15897#note-3
I think that comment is unclear.
It just means named and unnamed arguments are incompatible, just like currently in trunk
, foo { |named_arg| @1 }
is a SyntaxError.
foo { |_| _ * 3 }
and foo { _ * 3 }
would both work fine.
Updated by JonRowe (Jon Rowe) over 5 years ago
Is that used in practice though? I know pending examples, but I would expect they at least have a description.
No but as Ruby doesn't allow method overloading you do create that ambiguity, and the possibility for future bugs due to precedence rules.
FWIW, MSpec's it requires the description argument, so it cannot ever be ambiguous for the user.
Yes it can, as you are creating method / variable shadowing in built to the language, so its always ambiguous, is this the method or the variable.
Updated by Eregon (Benoit Daloze) over 5 years ago
JonRowe (Jon Rowe) wrote:
FWIW, MSpec's it requires the description argument, so it cannot ever be ambiguous for the user.
Yes it can, as you are creating method / variable shadowing in built to the language, so its always ambiguous, is this the method or the variable.
I think this ambiguity is fairly intuitive and easy to resolve:
it's exactly the same rule as for method calls without arguments, if we consider it
is always defined as a local variable.
I.e., if there are no arguments, it's always the variable/default block parameter, otherwise it's a method call with the given arguments.
I think it's going to be extremely rare for it
to be confusing in practice, once people know about the default block parameter feature.
Updated by joallard (Jonathan Allard) over 5 years ago
Eregon (Benoit Daloze) wrote:
I think this ambiguity is fairly intuitive and easy to resolve:
[...] if there are no arguments, it's always the variable/default block parameter, otherwise it's a method call with the given arguments.
I think it's going to be extremely rare for
it
to be confusing in practice, once people know about the default block parameter feature.
I would echo Benoit's comments here. As an avid RSpec user, I employ it "does something"
without a block fairly often for prototyping. However, I virtually never use a completely naked it
, no arguments or block, which seems to be the issue here. Even then, if I really need a naked 'it', I could always write it()
.
The following, taken from Jon Rowe's example above, makes total sense to me:
RSpec.describe do
it "will do the thing" do
it # without arguments, we're referring to a thing whose meaning is given by the context
end
it # meaning given by context: first block argument
it() # empty, description-less example (not affected)
it{} # also unaffected
end
While I do empathize with our RSpec folks, I don't see the value in preventing an expressive keyword to exist (which would be useful in a lot of cases) in order to save the ability to use the empty it
, which has a pretty limited use. Especially considering if one really needs the latter, they may just write it()
, a simple workaround.
The value of an expressive, language-based keyword is something I value and makes sense to me in Ruby's design based on natural language.
With that said, I'd support finding an alternative keyword, but it seems that based on previous discussions, we've not been able to find one that makes better consensus. I haven't either. Besides, we could make similar criticisms about those other keywords (item
, this/that
, one/other
, ...)
Updated by JonRowe (Jon Rowe) over 5 years ago
Its worth pointing out that this would always have to be lower priority than methods and other such locals defined in order to allow code to work. If precedence for this was changed to later override existing definitions of it, or simply ban them as is the case with other ruby keywords, it would break RSpec and Mspec. Consider:
it = 'is my string'
something do
it # what whould this be? the argument to the block or the string?
end
From my experience maintaining RSpec I think newcomers would either never know about it, or struggle to realise what is going on.
All just my opinions of course :)
Updated by mame (Yusuke Endoh) over 5 years ago
@JonRowe Thank you for your opinion, but I'd be happy if you could read my original proposal carefully. I've already pointed out the issue (and said my opinion against the issue).
Cons:
- it is somewhat fragile; "it" may refer a wrong variable
Just inserting an assignment to a variable "it" may affect another code. This is a bad news, but, IMO, a variable named "it" is not so often used. If this proposal is accepted, I guess people will gradually avoid the variable name "it" (like "p").
I hear from some people that they are actually using a variable it
. That's unfortunate, but I still think that a soft keyword "it
" is the best solution, as long as we need to add a something like @1
(and matz strongly wants to add something).
Updated by JonRowe (Jon Rowe) over 5 years ago
@mame (Yusuke Endoh) I did, I apologise for not making it clear, I'm reiterating it to add weight to the con, its not just a simple "it is somewhat fragile, it may refer to a wrong variable" its a "this shadows the most commonly used method in the most downloaded rubygem"1, its not one or two people this will affect.
I welcome the feature, I just think the name is wrong, something special like _1
etc would be better.
Updated by Eregon (Benoit Daloze) over 5 years ago
JonRowe (Jon Rowe) wrote:
its a "this shadows the most commonly used method in the most downloaded rubygem"[1], its not one or two people this will affect.
It only "shadows" (by that I understand "no longer works in that case") for the case of it
without any description, block and parenthesis, right?
Which seems extremely rare as said before.
So I expect only people using a plain it
from RSpec would be affected, and I would expect very few people use that (it doesn't even seem tested in RSpec's test suite).
If using a local variable named it
, it
continues just referring to that local variable in that method.
If precedence for this was changed to later override existing definitions of it
I believe it will never change if introduced, breaking RSpec is not an option.
Updated by Eregon (Benoit Daloze) over 5 years ago
I think we should listen to RSpec users here like @joallard and would welcome more users to reply on this thread.
People who maintain software defining it
probably have a different view than their users.
If RSpec users understand it
well, then I think that addresses the "potential confusion" point for RSpec.
Updated by ivoanjo (Ivo Anjo) over 5 years ago
As a user of both Kotlin and RSpec, here's my 0.01 cents: I've been using Kotlin for almost 1.5 years, and the implicit it
is really nice shortcut, without making the code too harder to understand. Even newcomers seem to get it
pretty quickly. I'd definitely use it if I had it in Ruby (and I do miss it when I switch between Ruby and Kotlin).
On the other hand the coincidence with RSpec's it
DSL is rather unfortunate. Would be better if this could be made a keyword, but that's clearly not an option, neither for it
nor for other options on this thread.
Would it be reasonable to perhaps emit a warning when it
is used naked in a scope where it
is defined as a method that can be called with zero args? That way we could slowly push away any remaining confusing issues, and perhaps the RSpec maintainers may consider dropping zero args it
in a future 4.x release?
Updated by shevegen (Robert A. Heiler) over 5 years ago
Just a very brief comment since it was discussed in the last developer meeting, solely on the
syntax issue of %1, %2, versus @1, @2 and :1, :2:
To me personally, %1, %2 is almost the same as @1, @2. I'd still prefer @1 @2 etc... but I
don't have a huge problem with %1, although I find @1 more elegant than %1, purely syntax-wise.
The idea for :1, :2 on the other hand, though, looks somewhat strange to me.
It reminds me more of a symbol than @1 reminds me of an instance variable, oddly enough.
If @1 would not be considered to be good from a syntax point (although I find it just
fine), then I think %1 would be better than :1.
To those who would like to have a look at the developer meeting log, the summary was
provided by mame (I think) here:
https://docs.google.com/document/d/1K61SGIwp8_rNsPyhmayUcERu71vt_etDjXdhqrLmBVY/edit#
Updated by ko1 (Koichi Sasada) over 5 years ago
I'm against on this proposal because of compatibility issue.
I think current workaround introduces new confusion.
Updated by joallard (Jonathan Allard) almost 5 years ago
It seems like this issue is stagnant. Is there anything needed for this to go forward?
Updated by jeremyevans0 (Jeremy Evans) almost 5 years ago
- Status changed from Open to Closed
joallard (Jonathan Allard) wrote in #note-38:
It seems like this issue is stagnant. Is there anything needed for this to go forward?
The default block parameter became the numbered block parameter and was added in 2.7. So you can now do:
[1, 2, 3].map { _1.to_s } #=> ["1", "2", "3"]
I'm going to close this feature since numbered block parameters were introduced to address it.
Updated by joallard (Jonathan Allard) over 4 years ago
jeremyevans0 (Jeremy Evans) wrote in #note-39:
I'm going to close this feature since numbered block parameters were introduced to address it.
Noted. But I think there might have been some confusion.
In my opinion, the spirit of this ticket was to adopt it
as a syntax rather than solve the underlying function with any combination of non-word symbols (eg. _1, @1, or $_0, _[0], ...) It seems to me like there's a fair amount of agreement about it.
Having slept on it for a few months now (pun intended) and as a daily user of Ruby for soon 10 years – and seeing it's in usage in other language –, I can say that adopting the keyword it
is still relevant and a good idea to me.
If I may argue on the substance just a bit more, I see the very essence of Ruby in it
. Not in _1
, unfortunately. Just like I see Ruby in array.first
rather than array[0]
. I am reminded of Ruby's readability-before-purity philsophy and "Write the code you wish you could".
Moreover, the one-arg block (obj.method{ |one| one... }
) pattern is, in my experience, used quite frequently. it
could make that both shorter and more expressive. That is why I find importance in this proposal.
When I asked whether there was anything needed to move forward, that's what it was about. The _1
syntax does not address the needs for a natural, expressive, terse block param syntax.
Of course, seeing the current resolution, I'm not having unrealistic expectations, but I just wanted to be clear with what the proposal was for and make the case.
If the right way is to open another ticket or there's another way forward, I'm happy to walk the walk or help.
Updated by k0kubun (Takashi Kokubun) over 2 years ago
- Has duplicate Feature #18980: `it` as a default block parameter added