Formate bytes para kilobytes, megabytes, gigabytes

184

Cenário: o tamanho de vários arquivos é armazenado em um banco de dados como bytes. Qual é a melhor maneira de formatar essas informações de tamanho em kilobytes, megabytes e gigabytes? Por exemplo, eu tenho um MP3 que o Ubuntu exibe como "5,2 MB (5445632 bytes)". Como eu exibia isso em uma página da Web como "5,2 MB" E arquivos com menos de um megabyte como KB e arquivos com um gigabyte e acima como GB?

leepowers
fonte
3
Eu acredito que você deve criar uma função fazendo isso. Apenas divida o número por 1024 e veja o resultado. Se for mais de 1024, divida novamente.
Ivan Nevostruev 24/03

Respostas:

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

(Extraído do php.net , existem muitos outros exemplos, mas eu gosto mais deste :-)

Leo
fonte
8
Se você usou $bytes /= (1 << (10 * $pow))ou algo parecido, eu poderia gostar melhor. :-P
Chris Jester-Young
5
Lá vai você: D (pessoalmente, eu não gosto de aritmética bit a bit, porque é difícil de entender se você não está acostumado a isso ...)
Leo
3
@Justin isso é porque 9287695/1024/1024 é de fato 8.857 :)
Mahn
30
Na verdade, é KiB, MiB, GiBe TiBdesde que você está dividindo por 1024. Se você dividir por 1000seria sem o i.
Devator 16/05
8
Uncomment one of the following alternativesera algo que eu não percebi por 5 minutos ...
Arnis Juraga
211

Esta é a implementação de Chris Jester-Young, a mais limpa que eu já vi, combinada com o php.net e um argumento de precisão.

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
John Himmelman
fonte
8
tem 2 erros - adicionar 1 a (pelo menos pequenos) arquivos de tamanho - não trabalhar com 0 (retorno NAN)
maazza
Agradável. Existe uma versão disso ao contrário?
Lucas
3
um lil sonhando : $suffixes = array('B', 'kB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'); eu quero um disco rígido de Yottabyte! :-P
SpYk3HH
1
eu tive que converter o tamanho de $ para o dobro para fazê-lo funcionar. aqui está o que funcionou para mim: função formatBytes ($ tamanho, $ precisão = 2) {$ base = log (valor do float ($ tamanho)) / log (1024); $ sufixos = array ('', 'k', 'M', 'G', 'T'); round de retorno (pow (1024, $ base - base ($ base)), $ precisão). $ sufixos [floor ($ base)]; }
Christopher Gray
formatBytes(259748192, 3)retornos 259748192 MBque não é certo
Virar
97

Pseudo-código:

$base = log($size) / log(1024);
$suffix = array("", "k", "M", "G", "T")[floor($base)];
return pow(1024, $base - floor($base)) . $suffix;
Chris Jester-Young
fonte
A Microsoft e a Apple usam 1024, isso depende do seu caso de uso.
Parsa Yazdani
15

Esta é a implementação do Kohana , você pode usá-lo:

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]);
}
ryeguy
fonte
A ideia deles de ter uma opção entre 1024 e 1000 de potência é boa. Mas essa implementação é realmente estranha. $force_unite $siparece fazer a mesma coisa. Você também pode passar qualquer string com um "i" para $force_unit, porque eles testam a posição. A formatação decimal também é um exagero.
Gus Neves
14

Apenas divida-o por 1024 para kb, 1024 ^ 2 para mb e 1024 ^ 3 para GB. Tão simples como isso.

Vonder
fonte
8

Apenas minha alternativa, curta e limpa:

/**
 * @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];
}

ou, mais estúpido e eficaz:

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";
}
guari
fonte
7

use esta função se desejar um código curto

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
Yanni
fonte
6

Sei que talvez seja um pouco tarde para responder a essa pergunta, mas mais dados não vão matar alguém. Aqui está uma função muito rápida:

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: eu atualizei minha postagem para incluir a correção proposta por 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';
}
David Bélanger
fonte
1
Você recebe um B (BB) duplo para valores pequenos de $ B, pois, como alternativa, você poderia criar "$ S = 'kMGTPEZY'" e, em vez de "@ $ S [$ F]" do "@ $ S [$ F-1] ".
camomileCase
@camomileCase Dois anos e meio depois - atualizei minha resposta. Obrigado.
David Bélanger
4

Função simples

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
SebHallin
fonte
3

Solução flexível:

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"
Pavel Tzonkov
fonte
2

Eu consegui com a seguinte função,

    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];
    }
Janith Chinthana
fonte
2
Cuidado: K é para Kelvin e k é para quilos.
ZeWaren
2

Minha abordagem

    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];
  }
}
user24087
fonte
2

Não sei por que você deve tornar as coisas tão complicadas quanto as outras.

O código a seguir é muito mais simples de entender e cerca de 25% mais rápido que as outras soluções que usam a função de log (chamada de função 20 milhões de vezes com parâmetros diferentes)

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];
}
ZettiCaletti
fonte
2

Fiz isso convertendo todas as entradas em bytes e, assim, convertendo para qualquer saída necessária. Além disso, usei uma função auxiliar para obter a base 1000 ou 1024, mas a deixei flexível para decidir usar 1024 no tipo popular (sem 'i', como MB em vez de 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;
}
Gabriel Barcellos
fonte
1

tente isso;)

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);
Yahia Mgarrech
fonte
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');
Kerem Çakır
fonte
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];
}

Basta chamar a função

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

Este trabalho com o último 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]; 
} 
Josué Leo Moreno
fonte
Tudo o que foi feito lá é a mesma cópia exata de um exemplo do PHP.net, que foi feito pelo responsável pela resposta em 2010, apenas 8 anos depois.
JakeGould 31/03
1

Embora um pouco obsoleta, essa biblioteca oferece uma API de conversão robusta e testada :

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

Uma vez instalado:

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

// Output: "1.00KiB"

E para converter na outra direção:

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

// Output: "1024"

Além da conversão básica, oferece métodos de adição, subtração, comparação, etc.

Eu não sou de forma alguma afiliado a esta biblioteca.

Ben Johnson
fonte
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;
}
assustador
fonte
0

Aqui está a implementação simplificada da função 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;
  }
}
ya.teck
fonte
0

É um pouco tarde, mas uma versão um pouco mais rápida da resposta aceita está abaixo:

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

É mais eficiente, devido à realização de uma única operação log-2 em vez de duas operações log-e.

Na verdade, é mais rápido fazer a solução mais óbvia abaixo:

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

Isso ocorre porque o índice é calculado ao mesmo tempo que o valor do número de bytes na unidade apropriada. Isso reduziu o tempo de execução em cerca de 35% (um aumento de velocidade de 55%).

user3690595
fonte
0

Outra implementação condensada que pode ser traduzida para a base 1024 (binária) ou a base 1000 (decimal) e também funciona com números incrivelmente grandes, portanto, do uso da biblioteca 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';
}
cristão
fonte
Apenas uma pequena nota lateral; bcpow()lançará a exceção TypeError se $basee $inão for valores de sequência. Testado no PHP versão 7.0.11.
David Cery
Obrigado! Eu adicionei o rodízio de cordas e fixa um bug compensar :)
Christian
0

Imaginei que adicionaria uma malha de código de dois remetentes (usando o código de John Himmelman, que está neste segmento, e usando o código de Eugene Kuzmenko ) que estou usando.

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

Isso usa o código de Eugene para formatar os $valuebytes (eu mantenho meus dados em MB e os converte: 10485760 MBem 10995116277760) - depois usa o código de John para convertê-los no valor de exibição adequado ( 10995116277760em10 TB ).

Achei isso muito útil - então, meus agradecimentos aos dois participantes!

EML
fonte
0

Função extremamente simples para obter o tamanho do arquivo humano.

Fonte original: http://php.net/manual/de/function.filesize.php#106569

Copie / cole o código:

<?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];
}
?>
John Erck
fonte
0

Eu desenvolvi minha própria função que converte o tamanho da memória legível humana em tamanhos diferentes.

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

Essa função aceita qualquer abreviação de tamanho de memória como "Megabyte, MB, Mb, mb, m, kilobyte, K, KB, b, Terabyte, T ....", portanto, é seguro contra erros de digitação.

Juan Lago
fonte
0

Com base na resposta de Leo , adicione

  • Suporte para negativos
  • Suporte 0 <valor <1 (Ex: 0,2, causará log (valor) = número negativo)

Se você deseja unidade máxima para Mega, mude para $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];
}
Asa de aço
fonte