maneira mais rápida de corda convertido para inteiro em PHP
-
04-07-2019 - |
Pergunta
Usando PHP, qual é o caminho mais rápido para converter uma string como esta:? "123"
para um inteiro
Por que é que determinado método o mais rápido? O que acontece se ele fica de entrada inesperado, como "hello"
ou um array?
Solução
Acabei de configurar um exercício de benchmarking rápida:
Function time to run 1 million iterations
--------------------------------------------
(int) "123": 0.55029
intval("123"): 1.0115 (183%)
(int) "0": 0.42461
intval("0"): 0.95683 (225%)
(int) int: 0.1502
intval(int): 0.65716 (438%)
(int) array("a", "b"): 0.91264
intval(array("a", "b")): 1.47681 (162%)
(int) "hello": 0.42208
intval("hello"): 0.93678 (222%)
Em média, chamando intval () é duas vezes e meia mais lento, ea diferença é maior se a sua entrada já é um inteiro.
Eu estaria interessado em saber por embora.
Update: Eu executar os testes novamente, desta vez com a coerção (0 + $var)
| INPUT ($x) | (int) $x |intval($x) | 0 + $x |
|-----------------|------------|-----------|-----------|
| "123" | 0.51541 | 0.96924 | 0.33828 |
| "0" | 0.42723 | 0.97418 | 0.31353 |
| 123 | 0.15011 | 0.61690 | 0.15452 |
| array("a", "b") | 0.8893 | 1.45109 | err! |
| "hello" | 0.42618 | 0.88803 | 0.1691 |
|-----------------|------------|-----------|-----------|
Adenda: Acabei de se deparar com um comportamento um pouco inesperado que você deve estar ciente de quando se escolhe um dos seguintes métodos:
$x = "11";
(int) $x; // int(11)
intval($x); // int(11)
$x + 0; // int(11)
$x = "0x11";
(int) $x; // int(0)
intval($x); // int(0)
$x + 0; // int(17) !
$x = "011";
(int) $x; // int(11)
intval($x); // int(11)
$x + 0; // int(11) (not 9)
Testado usando PHP 5.3.1
Outras dicas
Pessoalmente, sinto fundição é a mais bonita.
$iSomeVar = (int) $sSomeOtherVar;
Se um string como 'Olá' ser enviado, ele será convertido para integer 0. Para uma string como '22 anos', ele vai ser convertido para inteiro 22. Qualquer coisa que não pode analisar a um número torna-se 0.
Se você realmente precisa fazer a velocidade, eu acho que as outras sugestões aqui são correto em assumir que a coerção é o mais rápido.
Faça um teste.
string coerce: 7.42296099663
string cast: 8.05654597282
string fail coerce: 7.14159703255
string fail cast: 7.87444186211
Este foi um teste que correu cada cenário 10.000.000 vezes. : -)
Co-ercion é 0 + "123"
A carcaça é (integer)"123"
Eu acho Co-ercion é um pouco mais rápido. Oh, e tentando 0 + array('123')
é um erro fatal em PHP. Você pode querer o seu código para verificar o tipo do valor fornecido.
O meu código de teste está abaixo.
function test_string_coerce($s) {
return 0 + $s;
}
function test_string_cast($s) {
return (integer)$s;
}
$iter = 10000000;
print "-- running each text $iter times.\n";
// string co-erce
$string_coerce = new Timer;
$string_coerce->Start();
print "String Coerce test\n";
for( $i = 0; $i < $iter ; $i++ ) {
test_string_coerce('123');
}
$string_coerce->Stop();
// string cast
$string_cast = new Timer;
$string_cast->Start();
print "String Cast test\n";
for( $i = 0; $i < $iter ; $i++ ) {
test_string_cast('123');
}
$string_cast->Stop();
// string co-erce fail.
$string_coerce_fail = new Timer;
$string_coerce_fail->Start();
print "String Coerce fail test\n";
for( $i = 0; $i < $iter ; $i++ ) {
test_string_coerce('hello');
}
$string_coerce_fail->Stop();
// string cast fail
$string_cast_fail = new Timer;
$string_cast_fail->Start();
print "String Cast fail test\n";
for( $i = 0; $i < $iter ; $i++ ) {
test_string_cast('hello');
}
$string_cast_fail->Stop();
// -----------------
print "\n";
print "string coerce: ".$string_coerce->Elapsed()."\n";
print "string cast: ".$string_cast->Elapsed()."\n";
print "string fail coerce: ".$string_coerce_fail->Elapsed()."\n";
print "string fail cast: ".$string_cast_fail->Elapsed()."\n";
class Timer {
var $ticking = null;
var $started_at = false;
var $elapsed = 0;
function Timer() {
$this->ticking = null;
}
function Start() {
$this->ticking = true;
$this->started_at = microtime(TRUE);
}
function Stop() {
if( $this->ticking )
$this->elapsed = microtime(TRUE) - $this->started_at;
$this->ticking = false;
}
function Elapsed() {
switch( $this->ticking ) {
case true: return "Still Running";
case false: return $this->elapsed;
case null: return "Not Started";
}
}
}
Você pode simplesmente converter cadeia longa em inteiro usando FLOAT
$float = (float)$num;
Ou se você quiser inteiro não flutuante val, em seguida, ir com
$float = (int)$num;
Por ex.
(int) "1212.3" = 1212
(float) "1212.3" = 1212.3
$int = settype("100", "integer"); //convert the numeric string to int
extrato inteiro de qualquer string
$ em 'tel.123-12-33';
preg_match_all('!\d+!', $in, $matches);
$out = (int)implode('', $matches[0]);
// $ out = '1231233';
Mais ad-hoc resultados de benchmark:
$ time php -r 'for ($x = 0;$x < 999999999; $x++){$i = (integer) "-11";}'
real 2m10.397s
user 2m10.220s
sys 0m0.025s
$ time php -r 'for ($x = 0;$x < 999999999; $x++){$i += "-11";}'
real 2m1.724s
user 2m1.635s
sys 0m0.009s
$ time php -r 'for ($x = 0;$x < 999999999; $x++){$i = + "-11";}'
real 1m21.000s
user 1m20.964s
sys 0m0.007s
Ran um ponto de referência, e verifica-se a maneira mais rápida de obter um real inteiro (usando todos os métodos disponíveis) é
$foo = (int)+"12.345";
Apenas usando
$foo = +"12.345";
retorna um float.