Ответ 1
Как насчет размера файла? Кажется, он способен конвертировать из байтов (и других форматов) в довольно печатные значения:
Пример:
Filesize.from("12502343 B").pretty # => "11.92 MiB"
Я пытаюсь создать метод, который преобразует целое число, представляющее байты в строку с форматом "pretedied up".
Здесь моя полузаработная попытка:
class Integer
def to_filesize
{
'B' => 1024,
'KB' => 1024 * 1024,
'MB' => 1024 * 1024 * 1024,
'GB' => 1024 * 1024 * 1024 * 1024,
'TB' => 1024 * 1024 * 1024 * 1024 * 1024
}.each_pair { |e, s| return "#{s / self}#{e}" if self < s }
end
end
Что я делаю неправильно?
Как насчет размера файла? Кажется, он способен конвертировать из байтов (и других форматов) в довольно печатные значения:
Пример:
Filesize.from("12502343 B").pretty # => "11.92 MiB"
Если вы используете его с Rails - как насчет стандартного помощника числа Rails?
number_to_human_size(number, options = {})
?
Я согласен с @David, что лучше всего использовать существующее решение, но чтобы ответить на ваш вопрос о том, что вы делаете неправильно:
s
на self
, а не наоборот.s
, поэтому разделите s
на 1024.Итак:
class Integer
def to_filesize
{
'B' => 1024,
'KB' => 1024 * 1024,
'MB' => 1024 * 1024 * 1024,
'GB' => 1024 * 1024 * 1024 * 1024,
'TB' => 1024 * 1024 * 1024 * 1024 * 1024
}.each_pair { |e, s| return "#{(self.to_f / (s / 1024)).round(2)}#{e}" if self < s }
end
end
позволяет:
1.to_filesize # => "1.0B" 1020.to_filesize # => "1020.0B" 1024.to_filesize # => "1.0KB" 1048576.to_filesize # => "1.0MB"
Опять же, я не рекомендую это делать, но, кажется, стоит исправить ошибки.
Вы получаете баллы за добавление метода к Integer, но это кажется более специфичным для файла, поэтому я предлагаю обезьяну с файлом, скажем, добавив метод в файл с именем .prettysize().
Но вот альтернативное решение, которое использует итерацию, и позволяет печатать одиночные байты как float: -)
def format_mb(size)
conv = [ 'b', 'kb', 'mb', 'gb', 'tb', 'pb', 'eb' ];
scale = 1024;
ndx=1
if( size < 2*(scale**ndx) ) then
return "#{(size)} #{conv[ndx-1]}"
end
size=size.to_f
[2,3,4,5,6,7].each do |ndx|
if( size < 2*(scale**ndx) ) then
return "#{'%.3f' % (size/(scale**(ndx-1)))} #{conv[ndx-1]}"
end
end
ndx=7
return "#{'%.3f' % (size/(scale**(ndx-1)))} #{conv[ndx-1]}"
end
Решение @Darshan Computing здесь только частичное. Поскольку хеш-ключи не гарантируются, этот подход не будет работать надежно. Вы можете исправить это, выполнив что-то вроде этого внутри метода to_filesize,
conv={
1024=>'B',
1024*1024=>'KB',
...
}
conv.keys.sort.each { |s|
next if self >= s
e=conv[s]
return "#{(self.to_f / (s / 1024)).round(2)}#{e}" if self < s }
}
Это то, что я сделал для аналогичного метода внутри Float,
class Float
def to_human
conv={
1024=>'B',
1024*1024=>'KB',
1024*1024*1024=>'MB',
1024*1024*1024*1024=>'GB',
1024*1024*1024*1024*1024=>'TB',
1024*1024*1024*1024*1024*1024=>'PB',
1024*1024*1024*1024*1024*1024*1024=>'EB'
}
conv.keys.sort.each { |mult|
next if self >= mult
suffix=conv[mult]
return "%.2f %s" % [ self / (mult / 1024), suffix ]
}
end
end
Это мое решение:
def filesize(size)
units = ['B', 'KiB', 'MiB', 'GiB', 'TiB', 'Pib', 'EiB']
return '0.0 B' if size == 0
exp = (Math.log(size) / Math.log(1024)).to_i
exp = 6 if exp > 6
'%.1f %s' % [size.to_f / 1024 ** exp, units[exp]]
end
По сравнению с другими решениями он более простой, более эффективный и генерирует более правильный вывод.
Оба to_filesize
и to_human
имеют проблемы с большими числами. format_mb
имеет странный случай, когда, например, '1 MiB' считается "1024 KiB", что-то, что некоторые могут захотеть, но, конечно, не я.
origin: filesize to_filesize format_mb to_human
0 B: 0.0 B 0.0B 0 b 0.00 B
1 B: 1.0 B 1.0B 1 b 1.00 B
10 B: 10.0 B 10.0B 10 b 10.00 B
1000 B: 1000.0 B 1000.0B 1000 b 1000.00 B
1 KiB: 1.0 KiB 1.0KB 1024 b 1.00 KB
1.5 KiB: 1.5 KiB 1.5KB 1536.0 b 1.50 KB
10 KiB: 10.0 KiB 10.0KB 10.000 kb 10.00 KB
100 KiB: 100.0 KiB 100.0KB 100.000 kb 100.00 KB
1000 KiB: 1000.0 KiB 1000.0KB 1000.000 kb 1000.00 KB
1 MiB: 1.0 MiB 1.0MB 1024.000 kb 1.00 MB
1 Gib: 1.0 GiB 1.0GB 1024.000 mb 1.00 GB
1 TiB: 1.0 TiB 1.0TB 1024.000 gb 1.00 TB
1 PiB: 1.0 Pib ERROR 1024.000 tb 1.00 PB
1 EiB: 1.0 EiB ERROR 1024.000 pb 1.00 EB
1 ZiB: 1024.0 EiB ERROR 1024.000 eb ERROR
1 YiB: 1048576.0 EiB ERROR 1048576.000 eb ERROR
Кроме того, он имеет лучшую производительность.
user system total real
filesize: 2.740000 0.000000 2.740000 ( 2.747873)
to_filesize: 3.560000 0.000000 3.560000 ( 3.557808)
format_mb: 2.950000 0.000000 2.950000 ( 2.949930)
to_human: 5.770000 0.000000 5.770000 ( 5.783925)
Я тестировал каждую реализацию с реалистичным генератором случайных чисел:
def numbers
Enumerator.new do |enum|
1000000.times do
exp = rand(5)
num = rand(1024 ** exp)
enum.yield num
end
end
end