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


184

Сценарій: розмір різних файлів зберігається в базі даних у вигляді байтів. Який найкращий спосіб відформатувати інформацію про цей розмір у кілобайти, мегабайти та гігабайти? Наприклад, у мене є MP3, який Ubuntu відображає як "5,2 Мб (5445632 байт)". Як я можу відобразити це на веб-сторінці як "5,2 МБ" ТА мати файли менше одного мегабайт у форматі KB, а файли - один гігабайт і вище як ГБ?


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 дійсно 8857 :)
Ман

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
Мені довелося кинути розмір $ в подвійний, щоб змусити його працювати. ось що для мене працювало: format formatBytes ($ size, $ precision = 2) {$ base = log (floatval ($ size)) / log (1024); $ суфікси = масив ('', 'k', 'M', 'G', 'T'); повернення круглий (pow (1024, $ база - підлога ($ база)), $ точність). $ суфікси [пол ($ база)]; }
Крістофер Грей

formatBytes(259748192, 3)повертає, 259748192 MBщо не вірно
Фліп

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, тому що вони перевіряють позицію. Десяткове форматування також є надмірним.
Gus Neves


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';
}

EDIT: Я оновив свою публікацію, щоб включити виправлення, запропоноване 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] "зробити" @ $ S [$ F-1] ".
camomileCase

@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% швидше, ніж інші рішення, які використовують функцію журналу (називається функцією 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

Я зробив це перетворення всього вводу в байт і так перетворення на будь-який необхідний вихід. Також я використав допоміжну функцію, щоб отримати базову 1000 або 1024, але залишив її гнучкою, щоб вирішити використовувати 1024 за популярним типом (без 'я', як МБ замість 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

Ця робота з останнім 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

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.
Девід Кері

Дякую! Я додав заклик струн і виправив помилку зміщення :)
Крістіан

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

Ця функція приймає будь-яку абревіатуру розміру пам'яті, наприклад "Мегабайт, МБ, Мб, мб, м, кілобайт, К, КБ, b, Терабайт, Т ....", тому вона є безпечною для друку.


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.