Формат байтов в килобайтах, мегабайтах, гигабайтах


184

Сценарий: размер различных файлов хранится в базе данных в байтах. Как лучше всего отформатировать информацию о размере в килобайтах, мегабайтах и ​​гигабайтах? Например, у меня есть MP3, который Ubuntu отображает как «5,2 МБ (5445632 байт)». Как бы я отобразил это на веб-странице как «5,2 МБ» И если файлы размером менее одного мегабайта отображаются в КБ, а файлы размером в один гигабайт и выше отображаются в ГБ?


3
Я верю, что вы должны создать функцию, которая делает это. Просто разделите число на 1024 и посмотрите на результат. Если его больше, чем 1024, то снова делим.
Иван Невоструев

Ответы:


319
function formatBytes($bytes, $precision = 2) { 
    $units = array('B', 'KB', 'MB', 'GB', 'TB'); 

    $bytes = max($bytes, 0); 
    $pow = floor(($bytes ? log($bytes) : 0) / log(1024)); 
    $pow = min($pow, count($units) - 1); 

    // Uncomment one of the following alternatives
    // $bytes /= pow(1024, $pow);
    // $bytes /= (1 << (10 * $pow)); 

    return round($bytes, $precision) . ' ' . $units[$pow]; 
} 

(Взято с php.net , там много других примеров, но мне больше нравится этот :-)


8
Если бы вы использовали $bytes /= (1 << (10 * $pow))или что-то подобное, мне бы это понравилось больше. :-P
Крис Шестер-Янг

5
Вот и вы: D (лично мне не нравится побитовая арифметика, потому что трудно понять, если вы к этому не привыкли)
Лев

3
@Justin, потому что 9287695/1024/1024 - это действительно 8 857 :)
Mahn

30
На самом деле, это KiB, MiB, GiBи TiBтак как вы деля 1024. Если бы вы поделились на 1000это было бы без i.
Деватор

8
Uncomment one of the following alternativesбыло чем-то, чего я не замечал в течение 5 минут ...
Арнис Юрага

211

Это реализация Криса Джестера-Янга, самая чистая из всех, что я когда-либо видел, в сочетании с php.net и точным аргументом.

function formatBytes($size, $precision = 2)
{
    $base = log($size, 1024);
    $suffixes = array('', 'K', 'M', 'G', 'T');   

    return round(pow(1024, $base - floor($base)), $precision) .' '. $suffixes[floor($base)];
}

echo formatBytes(24962496);
// 23.81M

echo formatBytes(24962496, 0);
// 24M

echo formatBytes(24962496, 4);
// 23.8061M

8
в нем 2 ошибки - добавьте 1 к (как минимум небольшому) размеру файла - не работает с 0 (возврат NAN)
maazza

Хороший. Есть ли версия этого наоборот?
Люк

3
я мечтаю : $suffixes = array('B', 'kB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'); я хочу жесткий диск Yottabyte! :-P
SpYk3HH

1
Я должен был привести к двойному размеру $, чтобы заставить его работать. Вот что сработало для меня: функция formatBytes ($ size, $ precision = 2) {$ base = log (floatval ($ size)) / log (1024); $ суффиксы = массив ('', 'k', 'M', 'G', 'T'); возвратный раунд (pow (1024, $ base - floor ($ base)), $ precision). $ суффиксы [этаж ($ основа)]; }
Кристофер Грей

formatBytes(259748192, 3)возвращается, 259748192 MBчто не правильно
Flip

97

псевдокод:

$base = log($size) / log(1024);
$suffix = array("", "k", "M", "G", "T")[floor($base)];
return pow(1024, $base - floor($base)) . $suffix;

Microsoft и Apple используют 1024, это зависит от вашего варианта использования.
Парса

15

Это реализация Kohana , вы можете использовать его:

public static function bytes($bytes, $force_unit = NULL, $format = NULL, $si = TRUE)
{
    // Format string
    $format = ($format === NULL) ? '%01.2f %s' : (string) $format;

    // IEC prefixes (binary)
    if ($si == FALSE OR strpos($force_unit, 'i') !== FALSE)
    {
        $units = array('B', 'KiB', 'MiB', 'GiB', 'TiB', 'PiB');
        $mod   = 1024;
    }
    // SI prefixes (decimal)
    else
    {
        $units = array('B', 'kB', 'MB', 'GB', 'TB', 'PB');
        $mod   = 1000;
    }

    // Determine unit to use
    if (($power = array_search((string) $force_unit, $units)) === FALSE)
    {
        $power = ($bytes > 0) ? floor(log($bytes, $mod)) : 0;
    }

    return sprintf($format, $bytes / pow($mod, $power), $units[$power]);
}

Их идея иметь опцию между 1024 и 1000 мощностями хороша. Но эта реализация действительно странная. $force_unitи, $siкажется, делают то же самое. Вы также можете передать любую строку с «i» $force_unit, потому что они проверяют позицию. Десятичное форматирование также излишне.
Гас Невес

14

Просто разделите его на 1024 для КБ, 1024 ^ 2 для МБ и 1024 ^ 3 для ГБ. Так просто, как, что.


8

Просто моя альтернатива, короткая и чистая

/**
 * @param int $bytes Number of bytes (eg. 25907)
 * @param int $precision [optional] Number of digits after the decimal point (eg. 1)
 * @return string Value converted with unit (eg. 25.3KB)
 */
function formatBytes($bytes, $precision = 2) {
    $unit = ["B", "KB", "MB", "GB"];
    $exp = floor(log($bytes, 1024)) | 0;
    return round($bytes / (pow(1024, $exp)), $precision).$unit[$exp];
}

или, более глупый и эффективный:

function formatBytes($bytes, $precision = 2) {
    if ($bytes > pow(1024,3)) return round($bytes / pow(1024,3), $precision)."GB";
    else if ($bytes > pow(1024,2)) return round($bytes / pow(1024,2), $precision)."MB";
    else if ($bytes > 1024) return round($bytes / 1024, $precision)."KB";
    else return ($bytes)."B";
}

7

используйте эту функцию, если вы хотите короткий код

bcdiv ()

$size = 11485760;
echo bcdiv($size, 1048576, 0); // return: 10

echo bcdiv($size, 1048576, 2); // return: 10,9

echo bcdiv($size, 1048576, 2); // return: 10,95

echo bcdiv($size, 1048576, 3); // return: 10,953

6

Я знаю, что может быть немного поздно ответить на этот вопрос, но больше данных не собирается кого-то убивать. Вот очень быстрая функция:

function format_filesize($B, $D=2){
    $S = 'BkMGTPEZY';
    $F = floor((strlen($B) - 1) / 3);
    return sprintf("%.{$D}f", $B/pow(1024, $F)).' '.@$S[$F].'B';
}

РЕДАКТИРОВАТЬ: я обновил свой пост, чтобы включить исправление, предложенное camomileCase:

function format_filesize($B, $D=2){
    $S = 'kMGTPEZY';
    $F = floor((strlen($B) - 1) / 3);
    return sprintf("%.{$D}f", $B/pow(1024, $F)).' '.@$S[$F-1].'B';
}

1
Вы получаете двойной B (BB) для небольших значений $ B, в качестве обходного пути вы можете сделать "$ S = 'kMGTPEZY'", а вместо "@ $ S [$ F]" do "@ $ S [$ F-1]».
РомашкаCase

@camomileCase Два с половиной года спустя - я обновил свой ответ. Спасибо.
Дэвид Беланджер

4

Простая функция

function formatBytes($size, $precision = 0){
    $unit = ['Byte','KiB','MiB','GiB','TiB','PiB','EiB','ZiB','YiB'];

    for($i = 0; $size >= 1024 && $i < count($unit)-1; $i++){
        $size /= 1024;
    }

    return round($size, $precision).' '.$unit[$i];
}

echo formatBytes('1876144', 2);
//returns 1.79 MiB

3

Гибкое решение:

function size($size, array $options=null) {

    $o = [
        'binary' => false,
        'decimalPlaces' => 2,
        'decimalSeparator' => '.',
        'thausandsSeparator' => '',
        'maxThreshold' => false, // or thresholds key
        'suffix' => [
            'thresholds' => ['', 'K', 'M', 'G', 'T', 'P', 'E', 'Z', 'Y'],
            'decimal' => ' {threshold}B',
            'binary' => ' {threshold}iB',
            'bytes' => ' B'
        ]
    ];

    if ($options !== null)
        $o = array_replace_recursive($o, $options);

    $base = $o['binary'] ? 1024 : 1000;
    $exp = $size ? floor(log($size) / log($base)) : 0;

    if (($o['maxThreshold'] !== false) &&
        ($o['maxThreshold'] < $exp)
    )
        $exp = $o['maxThreshold'];

    return !$exp
        ? (round($size) . $o['suffix']['bytes'])
        : (
            number_format(
                $size / pow($base, $exp),
                $o['decimalPlaces'],
                $o['decimalSeparator'],
                $o['thausandsSeparator']
            ) .
            str_replace(
                '{threshold}',
                $o['suffix']['thresholds'][$exp],
                $o['suffix'][$o['binary'] ? 'binary' : 'decimal']
            )
        );
}

var_dump(size(disk_free_space('/')));
// string(8) "14.63 GB"
var_dump(size(disk_free_space('/'), ['binary' => true]));
// string(9) "13.63 GiB"
var_dump(size(disk_free_space('/'), ['maxThreshold' => 2]));
// string(11) "14631.90 MB"
var_dump(size(disk_free_space('/'), ['binary' => true, 'maxThreshold' => 2]));
// string(12) "13954.07 MiB"

2

Мне удалось с помощью следующей функции,

    function format_size($size) {
        $mod = 1024;
        $units = explode(' ','B KB MB GB TB PB');
        for ($i = 0; $size > $mod; $i++) {
            $size /= $mod;
        }
        return round($size, 2) . ' ' . $units[$i];
    }

2
Осторожно: K для Кельвина и k для килограммов.
ZeWaren

2

Мой подход

    function file_format_size($bytes, $decimals = 2) {
  $unit_list = array('B', 'KB', 'MB', 'GB', 'PB');

  if ($bytes == 0) {
    return $bytes . ' ' . $unit_list[0];
  }

  $unit_count = count($unit_list);
  for ($i = $unit_count - 1; $i >= 0; $i--) {
    $power = $i * 10;
    if (($bytes >> $power) >= 1)
      return round($bytes / (1 << $power), $decimals) . ' ' . $unit_list[$i];
  }
}

2

Я не знаю, почему вы должны сделать это так сложно, как другие.

Следующий код гораздо проще для понимания и примерно на 25% быстрее, чем другие решения, использующие функцию log (называемую функцией 20 миллионов раз с другими параметрами).

function formatBytes($bytes, $precision = 2) {
    $units = ['Byte', 'Kilobyte', 'Megabyte', 'Gigabyte', 'Terabyte'];
    $i = 0;

    while($bytes > 1024) {
        $bytes /= 1024;
        $i++;
    }
    return round($bytes, $precision) . ' ' . $units[$i];
}

2

Эта работа с последним PHP

function formatBytes($bytes, $precision = 2) { 
    $units = array('B', 'KB', 'MB', 'GB', 'TB'); 

    $bytes = max($bytes, 0); 
    $pow = floor(($bytes ? log($bytes) : 0) / log(1024)); 
    $pow = min($pow, count($units) - 1); 

    $bytes /= pow(1024, $pow); 

    return round($bytes, $precision) . ' ' . $units[$pow]; 
} 

Все, что было сделано, - это то же самое точное копирование примера из PHP.net, которое было сделано главным ответчиком еще в 2010 году, только 8 лет спустя.
JakeGould

2

Я сделал это, преобразовав все входные данные в байты и, таким образом, преобразовав их в любой необходимый вывод. Кроме того, я использовал вспомогательную функцию для получения базы 1000 или 1024, но оставил ее гибкой, чтобы решить использовать 1024 для популярного типа (без «i», например, MB вместо MiB).

    public function converte_binario($size=0,$format_in='B',$format_out='MB',$force_in_1024=false,$force_out_1024=false,$precisao=5,$return_format=true,$decimal=',',$centena=''){
    $out = false;

    if( (is_numeric($size)) && ($size>0)){
        $in_data = $this->converte_binario_aux($format_in,$force_in_1024);
        $out_data = $this->converte_binario_aux($format_out,$force_out_1024);

        // se formato de entrada e saída foram encontrados
        if( ((isset($in_data['sucesso'])) && ($in_data['sucesso']==true)) && ((isset($out_data['sucesso'])) && ($out_data['sucesso']==true))){
            // converte formato de entrada para bytes.
            $size_bytes_in = $size * (pow($in_data['base'], $in_data['pot']));
            $size_byte_out = (pow($out_data['base'], $out_data['pot']));
            // transforma bytes na unidade de destino
            $out = number_format($size_bytes_in / $size_byte_out,$precisao,$decimal,$centena);
            if($return_format){
                $out .= $format_out;
            }
        }
    }
    return $out;
}

public function converte_binario_aux($format=false,$force_1024=false){
    $out = [];
    $out['sucesso'] = false;
    $out['base'] = 0;
    $out['pot'] = 0;
    if((is_string($format) && (strlen($format)>0))){
        $format = trim(strtolower($format));
        $units_1000 = ['b','kb' ,'mb' ,'gb' ,'tb' ,'pb' ,'eb' ,'zb' ,'yb' ];
        $units_1024 = ['b','kib','mib','gib','tib','pib','eib','zib','yib'];
        $pot = array_search($format,$units_1000);
        if( (is_numeric($pot)) && ($pot>=0)){
            $out['pot'] = $pot;
            $out['base'] = 1000;
            $out['sucesso'] = true;
        }
        else{
            $pot = array_search($format,$units_1024);
            if( (is_numeric($pot)) && ($pot>=0)){
                $out['pot'] = $pot;
                $out['base'] = 1024;
                $out['sucesso'] = true;
            }
        }
        if($force_1024){
            $out['base'] = 1024;
        }
    }
    return $out;
}

1

попробуй это ;)

function bytesToSize($bytes) {
                $sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB'];
                if ($bytes == 0) return 'n/a';
                $i = intval(floor(log($bytes) / log(1024)));
                if ($i == 0) return $bytes . ' ' . $sizes[$i]; 
                return round(($bytes / pow(1024, $i)),1,PHP_ROUND_HALF_UP). ' ' . $sizes[$i];
            }
echo bytesToSize(10000050300);

1
function changeType($size, $type, $end){
    $arr = ['B', 'KB', 'MB', 'GB', 'TB'];
    $tSayi = array_search($type, $arr);
    $eSayi = array_search($end, $arr);
    $pow = $eSayi - $tSayi;
    return $size * pow(1024 * $pow) . ' ' . $end;
}

echo changeType(500, 'B', 'KB');

1
function convertToReadableSize($size)
{
  $base = log($size) / log(1024);
  $suffix = array("B", "KB", "MB", "GB", "TB");
  $f_base = floor($base);
  return round(pow(1024, $base - floor($base)), 1) . $suffix[$f_base];
}

Просто вызовите функцию

echo convertToReadableSize(1024); // Outputs '1KB'
echo convertToReadableSize(1024 * 1024); // Outputs '1MB'

1

Хотя эта библиотека немного устарела, она предлагает проверенный и надежный API преобразования:

https://github.com/gabrielelana/byte-units

После установки:

\ByteUnits\Binary::bytes(1024)->format();

// Output: "1.00KiB"

И конвертировать в другом направлении:

\ByteUnits\Binary::parse('1KiB')->numberOfBytes();

// Output: "1024"

Помимо базового преобразования, он предлагает методы сложения, вычитания, сравнения и т. Д.

Я никак не связан с этой библиотекой.


0
function byte_format($size) {
    $bytes = array( ' KB', ' MB', ' GB', ' TB' );
    foreach ($bytes as $val) {
        if (1024 <= $size) {
            $size = $size / 1024;
            continue;
        }
        break;
    }
    return round( $size, 1 ) . $val;
}

0

Вот упрощенная реализация функции Drupal format_size :

/**
 * Generates a string representation for the given byte count.
 *
 * @param $size
 *   A size in bytes.
 *
 * @return
 *   A string representation of the size.
 */
function format_size($size) {
  if ($size < 1024) {
    return $size . ' B';
  }
  else {
    $size = $size / 1024;
    $units = ['KB', 'MB', 'GB', 'TB'];
    foreach ($units as $unit) {
      if (round($size, 2) >= 1024) {
        $size = $size / 1024;
      }
      else {
        break;
      }
    }
    return round($size, 2) . ' ' . $unit;
  }
}

0

Немного поздно, но чуть более быстрый вариант принятого ответа приведен ниже:

function formatBytes($bytes, $precision)
{
    $unit_list = array
    (
        'B',
        'KB',
        'MB',
        'GB',
        'TB',
    );

    $bytes = max($bytes, 0);
    $index = floor(log($bytes, 2) / 10);
    $index = min($index, count($unit_list) - 1);
    $bytes /= pow(1024, $index);

    return round($bytes, $precision) . ' ' . $unit_list[$index];
}

Это более эффективно благодаря выполнению одной операции log-2 вместо двух операций log-e.

Однако на самом деле быстрее сделать более очевидное решение ниже:

function formatBytes($bytes, $precision)
{
    $unit_list = array
    (
        'B',
        'KB',
        'MB',
        'GB',
        'TB',
    );

    $index_max = count($unit_list) - 1;
    $bytes = max($bytes, 0);

    for ($index = 0; $bytes >= 1024 && $index < $index_max; $index++)
    {
        $bytes /= 1024;
    }

    return round($bytes, $precision) . ' ' . $unit_list[$index];
}

Это связано с тем, что индекс рассчитывается одновременно со значением количества байтов в соответствующем блоке. Это сократило время выполнения примерно на 35% (увеличение скорости на 55%).


0

Другая сжатая реализация, которая может переводиться в основную 1024 (двоичную) или базовую 1000 (десятичную), а также работает с невероятно большими числами, следовательно, об использовании библиотеки bc:

function renderSize($byte,$precision=2,$mibi=true)
{
    $base = (string)($mibi?1024:1000);
    $labels = array('K','M','G','T','P','E','Z','Y');
    for($i=8;$i>=1;$i--)
        if(bccomp($byte,bcpow($base, $i))>=0)
            return bcdiv($byte,bcpow($base, $i), $precision).' '.$labels[$i-1].($mibi?'iB':'B');
    return $byte.' Byte';
}

Просто небольшая заметка; bcpow()вызовет исключение TypeError, если $baseи $iне являются строковыми значениями. Проверено на PHP версии 7.0.11.
Дэвид Кери

Спасибо! Я добавил заклинатель строк и исправил ошибку смещения :)
Christian

0

Я решил добавить сетку из двух отправителей кода (используя код Джона Химмельмана, который находится в этой теме и используя код Евгения Кузьменко ), который я использую.

function swissConverter($value, $format = true, $precision = 2) {
    //Below converts value into bytes depending on input (specify mb, for 
    //example)
    $bytes = preg_replace_callback('/^\s*(\d+)\s*(?:([kmgt]?)b?)?\s*$/i', 
    function ($m) {
        switch (strtolower($m[2])) {
          case 't': $m[1] *= 1024;
          case 'g': $m[1] *= 1024;
          case 'm': $m[1] *= 1024;
          case 'k': $m[1] *= 1024;
        }
        return $m[1];
        }, $value);
    if(is_numeric($bytes)) {
        if($format === true) {
            //Below converts bytes into proper formatting (human readable 
            //basically)
            $base = log($bytes, 1024);
            $suffixes = array('', 'KB', 'MB', 'GB', 'TB');   

            return round(pow(1024, $base - floor($base)), $precision) .' '. 
                     $suffixes[floor($base)];
        } else {
            return $bytes;
        }
    } else {
        return NULL; //Change to prefered response
    }
}

Это использует код Евгения для форматирования $valueв байты (я храню свои данные в МБ, поэтому он преобразует мои данные: 10485760 MBв 10995116277760) - затем он использует код Джона для преобразования его в правильное отображаемое значение ( 10995116277760в10 TB ).

Я нашел это действительно полезным - так что я благодарен двум представителям!


0

Чрезвычайно простая функция, чтобы получить человеческий размер файла.

Первоисточник: http://php.net/manual/de/function.filesize.php#106569

Скопируйте / вставьте код:

<?php
function human_filesize($bytes, $decimals = 2) {
  $sz = 'BKMGTP';
  $factor = floor((strlen($bytes) - 1) / 3);
  return sprintf("%.{$decimals}f", $bytes / pow(1024, $factor)) . @$sz[$factor];
}
?>

0

Я разработал свою собственную функцию, которая преобразует читаемый человеком размер памяти в разные размеры.

function convertMemorySize($strval, string $to_unit = 'b')
{
    $strval    = strtolower(str_replace(' ', '', $strval));
    $val       = floatval($strval);
    $to_unit   = strtolower(trim($to_unit))[0];
    $from_unit = str_replace($val, '', $strval);
    $from_unit = empty($from_unit) ? 'b' : trim($from_unit)[0];
    $units     = 'kmgtph';  // (k)ilobyte, (m)egabyte, (g)igabyte and so on...


    // Convert to bytes
    if ($from_unit !== 'b')
        $val *= 1024 ** (strpos($units, $from_unit) + 1);


    // Convert to unit
    if ($to_unit !== 'b')
        $val /= 1024 ** (strpos($units, $to_unit) + 1);


    return $val;
}


convertMemorySize('1024Kb', 'Mb');  // 1
convertMemorySize('1024', 'k')      // 1
convertMemorySize('5.2Mb', 'b')     // 5452595.2
convertMemorySize('10 kilobytes', 'bytes') // 10240
convertMemorySize(2048, 'k')        // By default convert from bytes, result is 2

Эта функция принимает любое сокращение объема памяти, такое как «Мегабайт, МБ, МБ, МБ, М, КБ, К, КБ, В, Терабайт, Т ....», поэтому она безопасна при опечатке.


0

Основываясь на ответе Лео , добавьте

  • Поддержка негатива
  • Поддержка 0 <значение <1 (например, 0,2, приведет к записи (значение) = отрицательное число)

Если вы хотите, чтобы максимальная единица измерения была Мега, измените на $units = explode(' ', ' K M');


function formatUnit($value, $precision = 2) {
    $units = explode(' ', ' K M G T P E Z Y');

    if ($value < 0) {
        return '-' . formatUnit(abs($value));
    }

    if ($value < 1) {
        return $value . $units[0];
    }

    $power = min(
        floor(log($value, 1024)),
        count($units) - 1
    );

    return round($value / pow(1024, $power), $precision) . $units[$power];
}
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.