在PHP中将字符串转换为整数的最快方法

Fastest way to convert string to integer in PHP

提问人:nickf 提问时间:10/27/2008 最后编辑:billjamesdevnickf 更新时间:8/21/2015 访问量:375542

问:

使用 PHP,将这样的字符串转换为整数的最快方法是什么?"123"

为什么这种特定方法是最快的?如果它得到意外的输入,例如 或数组,会发生什么情况?"hello"

PHP 优化 铸造

评论

12赞 nickf 12/2/2008
好吧,如果它没有伤害(可读性),为什么不以最有效的方式做事呢?
19赞 Andy Lester 10/22/2009
如果它不影响速度,为什么不以最易读的方式做事呢?
4赞 nickf 10/22/2009
@Andy,请查看下面的基准测试。和之间的差异可以超过400%!(int)intval()
6赞 philipp 4/17/2012
速度很重要,因为速度对用户体验很重要。当您进行大量操作时,您希望快速进行操作!
13赞 totallyNotLizards 10/22/2012
在不踢死马医的情况下,我还想说,在这种情况下,速度与可读性的问题是无关紧要的,因为这个问题被标记为优化。在优化下标记的问题中想要速度的原因是不言自明的。

答:

375赞 nickf 10/27/2008 #1

我刚刚设置了一个快速的基准测试练习:

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%)

平均而言,调用 intval() 的速度要慢两倍半,如果你的输入已经是整数,则差异最大。

不过我很想知道为什么


更新:我再次运行测试,这次是强制执行的(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   |
|-----------------|------------|-----------|-----------|

补遗:我刚刚遇到了一个有点出乎意料的行为,您在选择以下方法之一时应该注意:

$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)

使用 PHP 5.3.1 进行测试

评论

9赞 staticsan 10/27/2008
这可能与 intval() 调用函数调用有关,而强制转换直接在解释器的表达式计算器中处理。这也可能是强制更快的原因。
10赞 Ozzy 4/2/2013
通过使用 php 鲜为人知的一元加运算符,可以进一步简化您的强制示例。$x + 0 -> +$x
0赞 John 1/8/2015
@nickf 反对 100% 的...什么?你说是最慢的,但你没有给出简单干净的答案,因为 OP 要求的最是什么。intval
1赞 ToolmakerSteve 8/21/2015
@John 由于他在第一个代码中只测试了两种情况,并且在每对中都给出了 % ,因此基本情况必须为 。但你有一个很好的观点,如果他这么明确地说会更清楚,特别是因为他后来添加了第三个案例!(int)intvalintval(int)
2赞 Artyom 8/22/2017
在较新的PHP版本中,这会导致任何变化吗?
15赞 staticsan 10/27/2008 #2

运行测试。

   string coerce:          7.42296099663
   string cast:            8.05654597282
   string fail coerce:     7.14159703255
   string fail cast:       7.87444186211

这是一个测试,每个场景运行了 10,000,000 次。:-)

Co-ercion 是0 + "123"

铸造是(integer)"123"

我认为 Co-ercion 要快一点。哦,尝试是PHP中的致命错误。您可能希望代码检查所提供值的类型。0 + array('123')

我的测试代码如下。


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

评论

0赞 bstoney 8/11/2016
我添加到这个测试中并使用 PHP 7 运行它。演员表略微领先,整体性能有了很大的提高: 字符串强制:1.9255340099335 字符串强制:1.5142338275909 字符串集类型:4.149735212326 字符串失败强制:1.2346560955048 字符串失败强制转换:1.3967711925507 字符串失败集类型:4.149735212326settype
35赞 Rexxars 10/27/2008 #3

我个人觉得选角是最漂亮的。

$iSomeVar = (int) $sSomeOtherVar;

如果发送了像“Hello”这样的字符串,它将被强制转换为整数 0。对于像“22 岁”这样的字符串,它将被强制转换为整数 22。它无法解析为数字的任何内容都将变为 0。

如果你真的需要速度,我想这里的其他建议是正确的,假设胁迫是最快的。

评论

7赞 nickf 10/27/2008
有趣的是,数组被强制转换为 1。去想想。
3赞 Super Cat 8/22/2015
@nickf 不是这样 - 它也可以转换为 0。它将布尔值 (true | false) 转换为整数 - 'false' = 0, 'true' = 1。如果数组 100% 为空,则为 false,如果数组包含任何数据,则为 true,即使它只是空字符串或 NULL 值。如果将空数组转换为整数,它将变为 0。(是的,我知道这么老了!
7赞 Elric Wamugu 4/3/2012 #4
$int = settype("100", "integer"); //convert the numeric string to int

评论

3赞 Mehran 10/29/2012
我相信一些参考或证明是有道理的!
0赞 Queue 3/7/2014
如果语句有效,$int实际上是一个布尔值,但它不会,因为 setType() 的第一个参数是通过引用传递的,因此必须是 var。
12赞 Nishchit 4/28/2014 #5

您可以简单地使用 FLOAT 将长字符串转换为整数

$float = (float)$num;

或者,如果您想要整数而不是浮动 val,那么请使用

$float = (int)$num;

例如。

(int)   "1212.3"   = 1212 
(float) "1212.3"   = 1212.3

评论

1赞 ToolmakerSteve 8/21/2015
哼?如果你想要一个 int,你为什么不使用 ?如果字符串包含一个整数,那将返回一个很像整数的值(即使它的内部类型可能是),但是如果规范是返回整数值,你为什么要这样做?假设传入的字符串是“1.3”?你不会得到一个整数。另外,为了将来阅读代码的任何人,你应该说出你的意思。如果你的意思是“它应该是一个整数”,那么就说 ,而不是 。(int)(float)float(int)(float)
4赞 Daniel 5/30/2014 #6

更多临时基准测试结果:

$ 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

评论

0赞 ToolmakerSteve 8/21/2015
最好在每个循环中编写 10 次正在测试的语句,这样时间就不会被循环开销所支配。例如 .直接使用文字值也是有风险的,除非你确定该语言不会在编译时完成一些工作。对于像PHP这样的动态语言来说可能没问题,但是如果测试其他语言的公式,我会提到以供将来参考。在测试循环之前设置一个变量,然后使用它更安全。因此,内部代码是 .{ $i = +"-11"; $i = +"-11"; $i= +"-11"; $i= +"-11"; ... }"-11"$x = "-11"$i =+$x
7赞 Developer 7/2/2014 #7

来自任何字符串的整数 excract

$in = '电话.123-12-33';

preg_match_all('!\d+!', $in, $matches);
$out =  (int)implode('', $matches[0]);

$out ='1231233';

评论

0赞 Kamnibula 6/10/2015
你是精英中的精英!我花了几个小时将一些 var 从 json 数据字符串转换为整数,只有您的方法有帮助!谢谢!
4赞 Andrew Plank 9/10/2014 #8

运行基准测试,结果发现获得真实整数的最快方法(使用所有可用方法)是

$foo = (int)+"12.345";

只是使用

$foo = +"12.345";

返回一个浮点数。

评论

2赞 ToolmakerSteve 8/21/2015
这比简单的 ?快了多少百分比?(int)"12.345"