3v4l.org

run code in 300+ PHP versions simultaneously
<?php function do_split($name) { return preg_split('/([A-Z]{2,}|([A-Z][^A-Z]*))/', $name, null, PREG_SPLIT_DELIM_CAPTURE | PREG_SPLIT_NO_EMPTY); } function convert_case($name) { return implode( '_', array_map( 'strtolower', do_split($name) ) ); } function convert_case2($in) { $len = strlen($in); if ($len < 2) { return strtolower($in); } // no state $upper = function ($c) { return 'A' <= $c && $c <= 'Z'; }; $lower = function ($c) { return 'a' <= $c && $c <= 'z'; }; $alpha = function ($c) use ($upper, $lower) { return $upper($c) || $lower($c); }; $num = function ($c) { return '0' <= $c && $c <= '9'; }; $alpha_num = function ($c) use ($alpha, $num) { return $alpha($c) || $num($c); }; // stream state $pos = 0; $char = $in[0]; $peek = function() use ($in, &$pos, $len) { if ($pos+1 >= $len) { return null; } return $in[$pos+1]; }; $next = function() use ($peek, &$pos, &$char) { $char = $peek(); $pos++; return $char !== null; }; // lex state $out = array(); $upper_word = null; $lower_word = null; $cap_word = null; $num_word = null; $find_word = function () use (&$char, $alpha_num, $lower, $upper, $next, $peek, &$lower_word, &$upper_word, &$cap_word, &$num_word) { while (!$alpha_num($char)) { if (!$next()) { return null; } } if ($lower($char)) { return $lower_word; } if ($upper($char)) { if ($upper($peek())) { // uppercase word return $upper_word; } return $cap_word; // capitalized word } return $num_word; }; $upper_word = function() use (&$out, &$char, $next, $peek, $upper, $lower, $find_word) { $word = $char; while ($next() && $upper($char)) { if ($lower($peek())) { // start of capitalized word (e.g. FOOBar at B) break; } $word .= $char; } $out[] = $word; return $find_word; }; $lower_word = function() use (&$out, &$char, $next, $lower, $find_word) { $word = $char; while ($next() && $lower($char)) { $word .= $char; } $out[] = $word; return $find_word; }; $cap_word = $lower_word; $state = $find_word; while ($state !== null) { $state = $state(); } return strtolower(implode('_', $out)); } convert_case2('FooBARbar0123'); $samples = array( 'FooBar', 'FooBAR', 'FOOBar', 'foo_bar', 'FOO_bar', 'FOO_Bar', 'foo_BAR', 'foo_Bar', 'Foo_bar', ); print_r(array_map(function ($name) { return array( 'name' => $name, 'converted' => convert_case2($name), // 'converted' => convert_case($name), ); }, $samples));
Output for 5.3.0 - 5.3.29, 5.4.0 - 5.4.45, 5.5.0 - 5.5.38, 5.6.0 - 5.6.28, 7.0.0 - 7.0.20, 7.1.0 - 7.1.33, 7.2.0 - 7.2.33, 7.3.0 - 7.3.33, 7.4.0 - 7.4.33, 8.0.0 - 8.0.30, 8.1.0 - 8.1.27, 8.2.0 - 8.2.18, 8.3.0 - 8.3.4, 8.3.6
Array ( [0] => Array ( [name] => FooBar [converted] => foo_bar ) [1] => Array ( [name] => FooBAR [converted] => foo_bar ) [2] => Array ( [name] => FOOBar [converted] => foo_bar ) [3] => Array ( [name] => foo_bar [converted] => foo_bar ) [4] => Array ( [name] => FOO_bar [converted] => foo_bar ) [5] => Array ( [name] => FOO_Bar [converted] => foo_bar ) [6] => Array ( [name] => foo_BAR [converted] => foo_bar ) [7] => Array ( [name] => foo_Bar [converted] => foo_bar ) [8] => Array ( [name] => Foo_bar [converted] => foo_bar ) )
Output for 8.3.5
Warning: PHP Startup: Unable to load dynamic library 'sodium.so' (tried: /usr/lib/php/8.3.5/modules/sodium.so (libsodium.so.23: cannot open shared object file: No such file or directory), /usr/lib/php/8.3.5/modules/sodium.so.so (/usr/lib/php/8.3.5/modules/sodium.so.so: cannot open shared object file: No such file or directory)) in Unknown on line 0 Array ( [0] => Array ( [name] => FooBar [converted] => foo_bar ) [1] => Array ( [name] => FooBAR [converted] => foo_bar ) [2] => Array ( [name] => FOOBar [converted] => foo_bar ) [3] => Array ( [name] => foo_bar [converted] => foo_bar ) [4] => Array ( [name] => FOO_bar [converted] => foo_bar ) [5] => Array ( [name] => FOO_Bar [converted] => foo_bar ) [6] => Array ( [name] => foo_BAR [converted] => foo_bar ) [7] => Array ( [name] => foo_Bar [converted] => foo_bar ) [8] => Array ( [name] => Foo_bar [converted] => foo_bar ) )

preferences:
216.36 ms | 404 KiB | 342 Q