Is there a better way to do this?

  • Thread starter Michael W. Ryder
  • Start date
M

Michael W. Ryder

As part of my learning Ruby I am trying to learn how to format strings.
The following is an example for formatting a U.S. phone number:

a = "1234567890"
b = "(000) 000-0000"
ai = 0

for i in 0..(b.length) -1
if b[i,1] == "0"
b[i,1] = a[ai,1]
ai += 1
end
end
puts b

Is there a better (more Rubyish) way to do this?
Obviously I will want to add more code in the future to handle
exceptions like phone numbers that are not 10 characters long or are
longer, but this will get me started.
 
H

Harry Kakueki

As part of my learning Ruby I am trying to learn how to format strings.
The following is an example for formatting a U.S. phone number:

a = "1234567890"
b = "(000) 000-0000"
ai = 0

for i in 0..(b.length) -1
if b[i,1] == "0"
b[i,1] = a[ai,1]
ai += 1
end
end
puts b

Is there a better (more Rubyish) way to do this?
Obviously I will want to add more code in the future to handle
exceptions like phone numbers that are not 10 characters long or are
longer, but this will get me started.
I'm sure there are better ways but here is something to look at.

a = "1234567890"
area = a.slice(0..2)
exc = a.slice(3..5)
num = a.slice(6..9)
tel = "(#{area}) #{exc}-#{num}"

p tel

Harry
 
M

Michael W. Ryder

Harry said:
As part of my learning Ruby I am trying to learn how to format strings.
The following is an example for formatting a U.S. phone number:

a = "1234567890"
b = "(000) 000-0000"
ai = 0

for i in 0..(b.length) -1
if b[i,1] == "0"
b[i,1] = a[ai,1]
ai += 1
end
end
puts b

Is there a better (more Rubyish) way to do this?
Obviously I will want to add more code in the future to handle
exceptions like phone numbers that are not 10 characters long or are
longer, but this will get me started.
I'm sure there are better ways but here is something to look at.

a = "1234567890"
area = a.slice(0..2)
exc = a.slice(3..5)
num = a.slice(6..9)
tel = "(#{area}) #{exc}-#{num}"

p tel

Harry
I am trying to come up with a "generic" formatting routine so that I
could feed it something like sform("123456789", "000-00-0000") or
sform("1234567890", "(000) 000-0000") or sform("123456", "00/00/00") and
it would work. I could easily do something like you suggest, and for
some cases it might be better, but I want something I can for any number
of formats. Thanks for the input.
 
D

Dan Zwell

I am trying to come up with a "generic" formatting routine so that I
could feed it something like sform("123456789", "000-00-0000") or
sform("1234567890", "(000) 000-0000") or sform("123456", "00/00/00") and
it would work. I could easily do something like you suggest, and for
some cases it might be better, but I want something I can for any number
of formats. Thanks for the input.

welcome.


require 'enumerator'

def sform(num, fmt)
# convert num to array (of one digit strings):
num = num.to_enum:)each_byte).map { |code| code.chr }

# for each zero, replace it with a digit popped off the
# front of the array of numbers (or characters):
fmt.gsub(/0/) { num.shift }
end


dan
 
H

Harry Kakueki

I am trying to come up with a "generic" formatting routine so that I
could feed it something like sform("123456789", "000-00-0000") or
sform("1234567890", "(000) 000-0000") or sform("123456", "00/00/00") and
it would work. I could easily do something like you suggest, and for
some cases it might be better, but I want something I can for any number
of formats. Thanks for the input.
Well, that's different :)
So, you want to grab the first 10 digits and ignore beyond that?
Look into regular expressions.
Harry
 
M

Michael W. Ryder

Dan said:
welcome.


require 'enumerator'

def sform(num, fmt)
# convert num to array (of one digit strings):
num = num.to_enum:)each_byte).map { |code| code.chr }

# for each zero, replace it with a digit popped off the
# front of the array of numbers (or characters):
fmt.gsub(/0/) { num.shift }
end


dan

Your method is much better than my C style one. With a little work to
handle exceptions it should generally work. The only problem I have
found so far is that it doesn't handle periods in the number string
properly -- i.e. sform("12345.67", "$00,000.00") returns $12,345..6"
instead of $12,345.67". Something for me to work on. Thanks for the code.
 
W

William James

welcome.

require 'enumerator'

def sform(num, fmt)
# convert num to array (of one digit strings):
num = num.to_enum:)each_byte).map { |code| code.chr }

# for each zero, replace it with a digit popped off the
# front of the array of numbers (or characters):
fmt.gsub(/0/) { num.shift }
end

dan

Without 'enumerator':

def sform( str, fmt )
ary = str.split(//)
fmt.gsub( /0/ ){ ary.shift }
end
 
H

Harry Kakueki

Well, that's different :)
So, you want to grab the first 10 digits and ignore beyond that?
Look into regular expressions.
Harry
This will grab digits and then you can do what you want with them.

arr = ["1234567890", "(123) 867-5309","12/34/56/78/90444"]

arr.each do |x|
num = x.scan(/\d/).join
p num
puts
end

Harry
 
W

William James

Your method is much better than my C style one. With a little work to
handle exceptions it should generally work. The only problem I have
found so far is that it doesn't handle periods in the number string
properly -- i.e. sform("12345.67", "$00,000.00") returns $12,345..6"
instead of $12,345.67". Something for me to work on. Thanks for the code.

I made some changes to handle the decimal point
and the case when there are fewer digits in the
string than in the format.

# The part on the left of the decimal point.
def sform_left( str, fmt )
result = ''
fmt = fmt.split(//)
str.split(//).reverse_each{|d|
while fmt.last != '0' do
result = fmt.pop + result
end
fmt.pop
result = d + result
}
result = fmt.first + result if fmt.first != '0'
result
end

# The part on the right of the decimal point.
def sform_right( str, fmt )
ary = str.split(//)
fmt.gsub( /0/ ){ ary.shift || '0' }
end

def sform( str, fmt )
str = str.split('.')
fmt = fmt.split('.')
result = sform_left( str[0], fmt[0])
if fmt[1]
result += "." + sform_right( str.last, fmt.last)
end
result
end

puts sform("12345", "$00,000")
puts sform("1234", "$00,000")
puts sform("123", "$00,000")
puts sform("12345.6", "$00,000.00")
puts sform("12345.678", "$00,000.00")
puts sform("12345.678", "$00,000")

--- output ---
$12,345
$1,234
$123
$12,345.60
$12,345.67
$12,345
 
H

Harry Kakueki

I am trying to come up with a "generic" formatting routine so that I
could feed it something like sform("123456789", "000-00-0000") or
sform("1234567890", "(000) 000-0000") or sform("123456", "00/00/00") and
it would work. I could easily do something like you suggest, and for
some cases it might be better, but I want something I can for any number
of formats. Thanks for the input.

I was thinking about your question again and came up with this.
Just before I posted I saw that William had taken a similar approach
and offered it in a more compact form.
Anyway , for what it's worth.
I was thinking the dots should only be in the formatting, not in the
input so I stripped them out of the input.

#arr = ["123456", "00/00/00"]
arr = ["12345.67", "$00,000.00"]
#arr = ["1234567890", "(000) 000-0000"]

inp_arr = arr[0].delete(".").split(//)
fmt_arr = arr[1].split(//)
str = ""
fmt_arr.each do |x|
str << inp_arr.shift if x =~ /\d/
str << x if x !~ /\d/
end
p str

Harry
 
M

Michael W. Ryder

William said:
Your method is much better than my C style one. With a little work to
handle exceptions it should generally work. The only problem I have
found so far is that it doesn't handle periods in the number string
properly -- i.e. sform("12345.67", "$00,000.00") returns $12,345..6"
instead of $12,345.67". Something for me to work on. Thanks for the code.

I made some changes to handle the decimal point
and the case when there are fewer digits in the
string than in the format.

# The part on the left of the decimal point.
def sform_left( str, fmt )
result = ''
fmt = fmt.split(//)
str.split(//).reverse_each{|d|
while fmt.last != '0' do
result = fmt.pop + result
end
fmt.pop
result = d + result
}
result = fmt.first + result if fmt.first != '0'
result
end

# The part on the right of the decimal point.
def sform_right( str, fmt )
ary = str.split(//)
fmt.gsub( /0/ ){ ary.shift || '0' }
end

def sform( str, fmt )
str = str.split('.')
fmt = fmt.split('.')
result = sform_left( str[0], fmt[0])
if fmt[1]
result += "." + sform_right( str.last, fmt.last)
end
result
end

puts sform("12345", "$00,000")
puts sform("1234", "$00,000")
puts sform("123", "$00,000")
puts sform("12345.6", "$00,000.00")
puts sform("12345.678", "$00,000.00")
puts sform("12345.678", "$00,000")

--- output ---
$12,345
$1,234
$123
$12,345.60
$12,345.67
$12,345
Thank you for the ideas. I figured there had to be a better way to
accomplish what I wanted without doing it like I would do it in C. With
a little work I think I can make these do everything I want to do. The
last step I need to work on is to allow it to handle formats that
replace leading zeros with blanks. This will allow the display of
dollar amounts in neat columns.
 
B

Brian Candler

As part of my learning Ruby I am trying to learn how to format strings.
The following is an example for formatting a U.S. phone number:

a = "1234567890"
b = "(000) 000-0000"
ai = 0

for i in 0..(b.length) -1
if b[i,1] == "0"
b[i,1] = a[ai,1]
ai += 1
end
end
puts b

Is there a better (more Rubyish) way to do this?

I have a vague recollection that Perl has a specific feature along these
lines: ah yes, see "man perlform". But I've never used it, and I think this
is one Perlism that Ruby hasn't copied.

It sounds to me like you actually want two different types of format:

format("000 000-0000","1234567890") # => "123 456-7890"
format("000000.00","1234.4") # => " 1234.40"

People have given you several solutions for the former. The latter is most
easily handled by sprintf (or format % [values]) if the value is numeric.

Actually, you could bend sprintf to do the former too:

val = "1234567890"
fmt = "(000) 000-0000"
res = sprintf(fmt.gsub(/0/,"%s"), *val.split(//))

Regards,

Brian.
 
B

Brian Candler

Actually, you could bend sprintf to do the former too:

val = "1234567890"
fmt = "(000) 000-0000"
res = sprintf(fmt.gsub(/0/,"%s"), *val.split(//))

which of course can be shrunk to

val = "1234567890"
fmt = "(000) 000-0000"
res = fmt.gsub(/0/,"%s") % val.split(//)
 
R

Robert Klemme

As part of my learning Ruby I am trying to learn how to format strings.
The following is an example for formatting a U.S. phone number:

a = "1234567890"
b = "(000) 000-0000"
ai = 0

for i in 0..(b.length) -1
if b[i,1] == "0"
b[i,1] = a[ai,1]
ai += 1
end
end
puts b

Is there a better (more Rubyish) way to do this?
Obviously I will want to add more code in the future to handle
exceptions like phone numbers that are not 10 characters long or are
longer, but this will get me started.

I'd start by having the parts of the number separate. Then you can
easily use sprintf for the formatting. For example, with

Phone = Struct.new:)country, :area, :number) do
def to_s
sprintf("(%d) %d-%d", country, area, number)
end
end

You can do

irb(main):006:0> a = Phone.new 123, 456, 7890
=> #<struct Phone country=123, area=456, number=7890>
irb(main):007:0> puts a
(123) 456-7890
=> nil
irb(main):008:0>

You can even add exceptions and special handling if one of the parts is
missing etc. If you need leading zeros, that's easy with sprintf as well.

Kind regards

robert
 
H

Harry Kakueki

I'd start by having the parts of the number separate. Then you can
easily use sprintf for the formatting. For example, with

Phone = Struct.new:)country, :area, :number) do
def to_s
sprintf("(%d) %d-%d", country, area, number)
end
end

You can do

irb(main):006:0> a = Phone.new 123, 456, 7890
=> #<struct Phone country=123, area=456, number=7890>
irb(main):007:0> puts a
(123) 456-7890
=> nil
irb(main):008:0>

You can even add exceptions and special handling if one of the parts is
missing etc. If you need leading zeros, that's easy with sprintf as well.

Kind regards

robert

I answered that question, too.
But then he added more requirements.
He not only wants to deal with phone number formats, but any format.

Harry
 
R

Robert Klemme

I answered that question, too.
But then he added more requirements.
He not only wants to deal with phone number formats, but any format.

That somehow eluded me. Thanks for pointing it out!

Kind regards

robert
 
K

Ken Bloom

As part of my learning Ruby I am trying to learn how to format strings.
The following is an example for formatting a U.S. phone number:

a = "1234567890"
b = "(000) 000-0000"
ai = 0

for i in 0..(b.length) -1
if b[i,1] == "0"
b[i,1] = a[ai,1]
ai += 1
end
end
puts b

Is there a better (more Rubyish) way to do this? Obviously I will want
to add more code in the future to handle exceptions like phone numbers
that are not 10 characters long or are longer, but this will get me
started.

puts a.sub(/(...)(...)(....)/,'(\1) \2-\3')

when you have other formats, use if a.length == whatever to decide which
pattern to apply.

--Ken
 
M

Michael W. Ryder

Ken said:
As part of my learning Ruby I am trying to learn how to format strings.
The following is an example for formatting a U.S. phone number:

a = "1234567890"
b = "(000) 000-0000"
ai = 0

for i in 0..(b.length) -1
if b[i,1] == "0"
b[i,1] = a[ai,1]
ai += 1
end
end
puts b

Is there a better (more Rubyish) way to do this? Obviously I will want
to add more code in the future to handle exceptions like phone numbers
that are not 10 characters long or are longer, but this will get me
started.

puts a.sub(/(...)(...)(....)/,'(\1) \2-\3')

when you have other formats, use if a.length == whatever to decide which
pattern to apply.

--Ken
That looks very interesting! Thanks for the code. I think it will work
great for parts of the method. With very little work this I can make
this work for formatting dollar amounts, etc.
 

Ask a Question

Want to reply to this thread or ask your own question?

You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.

Ask a Question

Members online

Forum statistics

Threads
473,770
Messages
2,569,584
Members
45,075
Latest member
MakersCBDBloodSupport

Latest Threads

Top