Что в Ruby противоположно chr ()?

100

Во многих языках есть пара функций, chr()и ord(), которые преобразуют числа в символьные значения. В некоторых языках ord()называется asc().

Ruby имеет Integer#chr, который отлично работает:

>> 65.chr
A

Справедливо. Но как пойти другим путем?

"A".each_byte do |byte|
   puts byte
end

печатает:

65

и это довольно близко к тому, что я хочу. Но я бы предпочел избежать цикла - я ищу что-то достаточно короткое, чтобы его можно было прочитать при объявлении const.

RJHunter
источник

Ответы:

84

Если String # ord не существовал в 1.9, он есть в 2.0:

"A".ord #=> 65
Роб Кэмерон
источник
33

В Ruby до серии 1.8 включительно следующие строки будут производить 65 (для ASCII):

puts ?A
'A'[0]

Поведение было изменено в Ruby 1.9, оба вышеперечисленных будут возвращать «A». Правильный способ сделать это в Ruby 1.9:

'A'[0].ord

К сожалению, этого ordметода нет в Ruby 1.8.

Роберт Гэмбл
источник
К сожалению, «правильный» путь в Ruby 1.9 такой длинный, но, по крайней мере, он будет легче обнаруживаться при поиске «ord». Спасибо за очень подробный ответ.
RJHunter
13

Пытаться:

'A'.unpack('c')
dylanfm
источник
1
Теперь, когда Ruby 1.9 изменил значение 'A' [0], это более переносимый метод.
AShelly
10

Я хотел бы +1 к комментарию dylanfm и AShelly, но добавить [0]:

'A'.unpack('C')[0]

Вызов unpack возвращает массив, содержащий одно целое число, которое не всегда принимается там, где требуется целое число:

$ ruby ​​-e 'printf ("0x% 02X \ n", "A" .unpack ("C"))'
-e: 1: в printf: невозможно преобразовать массив в целое число (TypeError)
    от -e: 1
$ ruby ​​-e 'printf ("0x% 02X \ n", "A" .unpack ("C") [0])'
0x41
$ 

Я пытаюсь написать код, работающий на Ruby 1.8.1, 1.8.7 и 1.9.2.

Отредактировано, чтобы передать C для распаковки в верхнем регистре, потому что unpack ("c") дает мне -1, где ord () дает мне 255 (несмотря на то, что он работает на платформе, где C подписан char).

Мартин Дори
источник
4

Только что наткнулся на это, когда собирал чистую версию Stringprep на Ruby через RFC.

Остерегайтесь того, что chrвыходит из строя за пределами [0,255], вместо этого используйте переносимые замены 1.9.x - 2.1.x:

[22] pry(main)> "\u0221".ord.chr
RangeError: 545 out of char range
from (pry):2:in 'chr'
[23] pry(main)> x = "\u0221".unpack('U')[0]
=> 545
[24] pry(main)> [x].pack('U')
=> "ȡ"
[25] pry(main)>

источник
Спасибо, это, кажется, единственный ответ, который дает charи его обратный в случае Unicode правильно
Munyari
3

Кроме того, если у вас есть символ в строке и вы хотите декодировать его без цикла:

puts 'Az'[0]
=> 65
puts 'Az'[1]
=> 122
Кент Фредрик
источник
2

У вас могут быть такие:

65.chr.ord
'a'.ord.chr
Эдуардо Сантана
источник
2

Если вы не против вытащить значения из массива, вы можете использовать "A".bytes

Кларк
источник
0

Я пишу код для 1.8.6 и 1.9.3, и мне не удалось заставить ни одно из этих решений работать в обеих средах :(

Однако я столкнулся с другим решением: http://smajnr.net/2009/12/ruby-1-8-nomethoderror-undefined-method-ord-for-string.html

У меня это тоже не сработало, но я адаптировал его для себя:

unless "".respond_to?(:ord)
  class Fixnum
    def ord
      return self
    end
  end
end

После этого в обеих средах будет работать следующее:

'A'[0].ord

Хантсколин
источник
Когда user18096 написал свой ответ, он "A".unpack("C")[0]был нацелен на Ruby 1.8.1, Ruby 1.8.7 и Ruby 1.9.2. Это не работает в вашей среде? Что за неудача?
RJHunter
Привет, RJHunter, я пытаюсь преобразовать определенный символ в строке в его числовое значение. Следующий код работает в 1.9.3, но не в 1.8.6. self.status = tagAccountString[4].unpack('C')[0] В 1.8.6 я получаю Exception undefined method unpack 'для 0: Fixnum обрабатывает данные основного буферизованного тега - exit` Следующий код работает (с моим предложенным решением) в обеих средах. self.status = tagAccountString[4].ord Любые советы (например, лучшее решение) более чем приветствуются
hantscolin
tagAccountString[4]возвращает String в новых Ruby, но используется для возврата Fixnum в Ruby 1.8. Вот почему вы увидели ошибку undefined method unpack for 0:Fixnum. Вы можете использовать status = tagAccountString[4,1].unpack('C')[0]или даже status, = tagAccountString.unpack('xxxxC')если вы всегда хотите игнорировать четыре символа и преобразовать следующий.
RJHunter
Спасибо RJHunter за объяснение и альтернативные решения. Однако, поскольку «мое» решение более читабельно и многоразово, я буду придерживаться его (если нет веской причины?)
hantscolin