Bug #5915
closedArray#join with explicit nil should not use $,
Added by john_firebaugh (John Firebaugh) almost 14 years ago. Updated over 13 years ago.
Description
=begin
The documentation for Array#join says that the default argument is $,. But:
$, = ","
[1,2,3].join(nil) #=> "1,2,3"
Based on the documentation (and the POLS), I would expect "123".
=end
        
           Updated by naruse (Yui NARUSE) almost 14 years ago
          
          
        
        
          
            Actions
          
          #1
            [ruby-core:42200]
          Updated by naruse (Yui NARUSE) almost 14 years ago
          
          
        
        
          
            Actions
          
          #1
            [ruby-core:42200]
        
      
      John Firebaugh wrote:
The documentation for Array#join says that the default argument is $,. But:
$, = ","
[1,2,3].join(nil) #=> "1,2,3"Based on the documentation (and the POLS), I would expect "123".
nil is not "", so there is no such documentation.
        
           Updated by john_firebaugh (John Firebaugh) almost 14 years ago
          
          
        
        
          
            Actions
          
          #2
            [ruby-core:42201]
          Updated by john_firebaugh (John Firebaugh) almost 14 years ago
          
          
        
        
          
            Actions
          
          #2
            [ruby-core:42201]
        
      
      On Fri, Jan 20, 2012 at 8:19 PM, Yui NARUSE naruse@airemix.jp wrote:
nil is not "", so there is no such documentation.
I don't understand your point.
To quote from array.c:
/*
- call-seq:
- 
ary.join(sep=$,) -> str
- Returns a string created by converting each element of the array to
- a string, separated by +sep+.
- 
[ "a", "b", "c" ].join #=> "abc"
- 
[ "a", "b", "c" ].join("-") #=> "a-b-c"
*/
Clearly $, is documented as being the default. If that is not the
desired behavior, the documentation should be changed. I would
suggest:
/*
- call-seq:
- 
ary.join(sep=nil) -> str
- Returns a string created by converting each element of the array to
- a string, separated by +sep+. If +sep+ is nil, $, is used instead.
- 
[ "a", "b", "c" ].join #=> "abc"
- 
[ "a", "b", "c" ].join("-") #=> "a-b-c"
*/
But I would prefer it to work as currently documented, because that
behavior is (slightly) less complex in that it doesn't require
explicitly specifying that "If +sep+ is nil, $, is used instead."
        
           Updated by john_firebaugh (John Firebaugh) almost 14 years ago
          
          
        
        
          
            Actions
          
          #3
            [ruby-core:42203]
          Updated by john_firebaugh (John Firebaugh) almost 14 years ago
          
          
        
        
          
            Actions
          
          #3
            [ruby-core:42203]
        
      
      $, = nil
[1, 2, 3].join
$, = ","
[1, 2, 3].join(nil)
According to how #join is currently documented, these should produce the same result.
        
           Updated by marcandre (Marc-Andre Lafortune) almost 14 years ago
          
          
        
        
          
            Actions
          
          #4
            [ruby-core:42211]
          Updated by marcandre (Marc-Andre Lafortune) almost 14 years ago
          
          
        
        
          
            Actions
          
          #4
            [ruby-core:42211]
        
      
      I feel that both Yui and John are right.
As Yui points out, the documentation doesn't state explicitly how nil is treated.
On the other hand, I agree with John that the current behaviour is not intuitive and that either the documentation should be clarified or the behaviour should be changed.
- To explicitly describe the current behavior, the documentation needs to state that if sepisnil, then$,is used and that if$,isnil, then no separator is used. This last part is also needed, because currentlyArray#joindoes not call#to_sonsep, sonilis actually a special value.
Or else:
2) The current behavior so that sep == nil means that there will be no separator. Documentation could state this explicitly by saying that "... separated by +sep+ (if non-nil)". Clearly, the shorter documentation reflects the fact that this is simpler. I believe this is what John suggests.
- The behaviour could also be changed so that Array#joincalls#to_sonsep, like it does on its members. Most Ruby functions only callto_stron the arguments, butArray#joincallsto_son the elements of the array, so it's not clear (to me) why it doesn't onsep(maybe Matz has an idea?). In that case, the behavior would match what John and I favor, and the documentation doesn't really need to change or could state that "elements andsepwill be converted to strings if need be by callingto_s".
I personally favor (2) or (3), so that:
$, = "foo"
[4,2].join(nil) == "42"
Yui, do you prefer the current behavior where the result is "4foo2"?
        
           Updated by naruse (Yui NARUSE) almost 14 years ago
          
          
        
        
          
            Actions
          
          #5
            [ruby-core:42218]
          Updated by naruse (Yui NARUSE) almost 14 years ago
          
          
        
        
          
            Actions
          
          #5
            [ruby-core:42218]
        
      
      - Status changed from Open to Assigned
- Assignee set to matz (Yukihiro Matsumoto)
Marc, thank you for clarification.
Marc-Andre Lafortune wrote:
As Yui points out, the documentation doesn't state explicitly how
nilis treated.
Yes.
On the other hand, I agree with John that the current behaviour is not intuitive and that either the documentation should be clarified or the behaviour should be changed.
- To explicitly describe the current behavior, the documentation needs to state that if
sepisnil, then$,is used and that if$,isnil, then no separator is used. This last part is also needed, because currentlyArray#joindoes not call#to_sonsep, sonilis actually a special value.
Yes, there is this option.
People  must show the reason against this.
Or else:
2) The current behavior so thatsep == nilmeans that there will be no separator. Documentation could state this explicitly by saying that "... separated by +sep+ (if non-nil)". Clearly, the shorter documentation reflects the fact that this is simpler. I believe this is what John suggests.
- The behaviour could also be changed so that
Array#joincalls#to_sonsep, like it does on its members. Most Ruby functions only callto_stron the arguments, butArray#joincallsto_son the elements of the array, so it's not clear (to me) why it doesn't onsep(maybe Matz has an idea?). In that case, the behavior would match what John and I favor, and the documentation doesn't really need to change or could state that "elements andsepwill be converted to strings if need be by callingto_s".I personally favor (2) or (3), so that:
$, = "foo" [4,2].join(nil) == "42"Yui, do you prefer the current behavior where the result is "4foo2"?
nil is a special value so that it can occur some special behavior.
The point seems the difference between $,=nil and sep=nil, feel strange.
        
           Updated by john_firebaugh (John Firebaugh) over 13 years ago
          
          
        
        
          
            Actions
          
          #6
            [ruby-core:42214]
          Updated by john_firebaugh (John Firebaugh) over 13 years ago
          
          
        
        
          
            Actions
          
          #6
            [ruby-core:42214]
        
      
      Array#join actually tries three coercions on elements: #to_str, followed by
#to_ary (for nested joins), followed by #to_s.
The separator is coerced with #to_str if it is not nil and not a
String. Note that this matters only for an explicitly passed parameter, as
assignment to $, is restricted to Strings and nil. I would favor lifting
this restriction, as the current coercion logic is applied to $, -- it
just happens to be a no-op given the assignment restriction.
        
           Updated by naruse (Yui NARUSE) over 13 years ago
          
          
        
        
          
            Actions
          
          #7
            [ruby-core:46427]
          Updated by naruse (Yui NARUSE) over 13 years ago
          
          
        
        
          
            Actions
          
          #7
            [ruby-core:46427]
        
      
      - Category set to doc
- Assignee changed from matz (Yukihiro Matsumoto) to naruse (Yui NARUSE)
- Target version set to 2.0.0
matz said this is documentation bug.
I'll fix it.
        
           Updated by naruse (Yui NARUSE) over 13 years ago
          
          
        
        
          
            Actions
          
          #8
            [ruby-core:46533]
          Updated by naruse (Yui NARUSE) over 13 years ago
          
          
        
        
          
            Actions
          
          #8
            [ruby-core:46533]
        
      
      - Status changed from Assigned to Closed
Added in r36421.
        
           Updated by headius (Charles Nutter) over 13 years ago
          
          
        
        
          
            Actions
          
          #9
            [ruby-core:46738]
          Updated by headius (Charles Nutter) over 13 years ago
          
          
        
        
          
            Actions
          
          #9
            [ruby-core:46738]
        
      
      Is there a test for this? If not, perhaps one should be added.
I'm fixing http://jira.codehaus.org/browse/JRUBY-6776 and would like to know that the behavior is being tested.