ChatGPT解决这个技术问题 Extra ChatGPT

从字符串中提取单个(无符号)整数

我想从包含数字和字母的字符串中提取数字,例如:

"In My Cart : 11 items"

我想提取数字 11


B
Bruno Viera Guerra

如果您只想过滤除数字之外的所有内容,最简单的方法是使用 filter_var

$str = 'In My Cart : 11 items';
$int = (int) filter_var($str, FILTER_SANITIZE_NUMBER_INT);

数字前的破折号将被视为减号。
不仅在数字之前。任何 +/- 都不会被截断。如果您有 xxx-yyy-24,您将获得 --24。您可以使用 str_repleace(array('+','-'), '', $result) 删除这两个标志。
@imclickingmaniac 你可以简单地做 max(0,filter_var($str, FILTER_SANITIZE_NUMBER_INT)) 而不是使用数组和字符串替换。
使用 max() 也会返回字符串中的减号,希望 strreplace() 是这里的选项。感谢您的提示。
abs((int) filter_var($str, FILTER_SANITIZE_NUMBER_INT));
G
Gaurav
$str = 'In My Cart : 11 12 items';
preg_match_all('!\d+!', $str, $matches);
print_r($matches);

如何在一个简单的变量中而不是在 Array 中获取我的号码?
$var = implode(' ', $matches[0]);
@Bizboss 以类似的方式尝试$newstr = preg_replace('!\d+!', '', $str);。它将从您的字符串中删除所有数字。示例:$str = 'In My Cart : 11 12 items'; 将输出 In My Cart : items
这不是一种资源友好的方式。使用 preg_replace("/[^0-9]/","",$string);filter_var($str, FILTER_SANITIZE_NUMBER_INT);
@eapo filter_var($str, FILTER_SANITIZE_NUMBER_INT); 提取的数字,plusminus 符号。正则表达式只提取数字!
s
simhumileco
preg_replace('/[^0-9]/', '', $string);

这应该做得更好!...


我喜欢这个,因为它返回一个字符串,而不是像 preg_match_all 示例那样的数组,我喜欢它是因为我只想要数字,而不是像 filter_var 示例那样的加号或破折号。
小心使用这种方法:类似 'In My Car_Price : 500.00 将打印 : 50000
@ErickBest 考虑到问题是 extract the numbers from a string,这是一件好事——小数点不是数字!
简单一点:preg_replace('/\D/', '', $string);
只需使用 preg_replace('/[^0-9.]/','',$string) 而不是您想将 . 保留在您的号码中。
b
bob jomes

使用 preg_replace

$str = '(111) 111-1111';
$str = preg_replace('/\D/', '', $str);
echo $str;

输出:1111111111


它工作得很好。它将 +387 (61) 634-783 字符串直接转换为 38761634783。您不必像 Gaurav 的回答那样处理数组。
绝对是最佳答案。 sandbox.onlinephpfunctions.com/code/…
我建议使用 + 量词来匹配更长的匹配项/更少的替换项。
这个答案似乎在回答一个不同的问题。 OP 仅从字符串中分离出一个整数值——购物车中的多个项目。此页面的理解范围似乎与 OP 实际要求的范围相去甚远。这是另一个问题的正确答案。
Y
Yash

对于浮点数,

preg_match_all('!\d+\.?\d+!', $string ,$match);

感谢您指出错误。 @mickmackusa


它实际上是错误的:) 应该是 '!\d*.?\d+!'
我更喜欢使用 ? 作为文字点的量词。我不希望浮点值包含多个小数点符号。具有讽刺意味的是,Yash 的注释模式是错误。点前面需要一个斜线以使其成为文字。
OP 的示例文本表明没有浮点值的潜力——在购物车中包含部分商品是没有意义的。示例文本中也只有一个整数值,因此使用 preg_match_all() 是多余的。这充其量是对不同问题的正确答案。
@mickmackusa 你是绝对正确的!我最终针对不同的用例提出了这个问题,并了解了如何实现它。提交的答案可能是我当时使用的解决方案。
C
Community

我不拥有这方面的功劳,但我只需要分享它。此正则表达式将从字符串中获取数字,包括小数点/位以及逗号:

/((?:[0-9]+,)*[0-9]+(?:\.[0-9]+)?)/

从这里引用:
php - regex - how to extract a number with decimal (dot and comma) from a string (e.g. 1,120.01)?


这不是在验证字符串中是否存在三个整数组之间的逗号。 ...复制粘贴别人的模式而不理解/测试它的问题。
当用户将其他 Stack Overflow 答案复制粘贴到页面上时,Stack Overflow 不会受益。如果另一个 Stack Overflow 页面包含此页面的解决建议,那么正确的行为是使用超链接作为关闭此页面作为副本的理由。最终,这个答案根本不适合 OP 的问题。 OP 不可能隔离浮点值,我们不知道是否会有数千个分隔符。
D
Dmitri Gudkov

您可以使用 preg_match

$s = "In My Cart : 11 items";
preg_match("|\d+|", $s, $m);
var_dump($m);

echo '<p>'.$m[0].'</p>';
H
Hariadi

使用 preg_replace

$str = 'In My Cart : 11 12 items';
$str = preg_replace('/\D/', '', $str);
echo $str;

我建议使用 + 量词来匹配更长的匹配项/更少的替换项。
为什么 OP 要形成字符串值 1112?可以从中获得什么可能的好处?
R
Rakesh
$value = '25%';

或者

$value = '25.025$';

或者

$value = 'I am numeric 25';
$onlyNumeric = filter_var($value, FILTER_SANITIZE_NUMBER_FLOAT, FILTER_FLAG_ALLOW_FRACTION);

这将只返回数值


为什么 OP 可能想要容纳浮点值? OP 的用户如何在他们的购物车中拥有一小部分商品?这似乎是对另一个问题的正确答案。
R
Ross Allen

您可以使用以下功能:

function extract_numbers($string)
{
   preg_match_all('/([\d]+)/', $string, $match);

   return $match[0];
}

在字符类中写入 \d 没有任何好处。由于您正在访问全字符串匹配,因此将捕获组写入模式是没有意义的。
OP 只有一个整数来隔离——preg_match_all() 对于所提出的问题来说是一个不合适的工具。这(充其量)是对不同问题的有效答案。
e
eapo

顶级资源友好型解决方案

<?php
    var $string = "In My Cart : 11 items";
?>

1. 最快:filter_var — 使用指定过滤器过滤变量

<?php
    filter_var($string, FILTER_SANITIZE_NUMBER_INT); // string(2) "11"
?>

2. 几乎最快:str_replace——用替换字符串替换所有出现的搜索字符串

<?php
    str_replace(array('In My Cart : ',' item', 's'),"", $string); // string(2) "11"
?>

3. 足够快:preg_replace——执行正则表达式搜索和替换

<?php
    preg_replace("/[^0-9]/","",$string); // string(2) "11"
?>

然而

str_replace 的简单性导致速度,但即使是有限的用例

preg_replace 比 str_replace 或 filter_var 更通用

相反,可以使用函数来指定使用 preg_replace_callback 替换的内容

使用 preg_replace_callback 可以在一次调用中进行多次替换

filter_var 卫生选项受限


B
Bradley Kovacs
preg_match_all('!\d+!', $some_string, $matches);
$string_of_numbers = implode(' ', $matches[0]);

在这种特定情况下, implode 中的第一个参数表示“用一个空格分隔 match[0] 中的每个元素”。 Implode 不会在第一个数字之前或最后一个数字之后放置空格(或任何您的第一个参数)。

还要注意的是 $matches[0] 是存储找到的匹配数组(匹配此正则表达式)的位置。

有关数组中其他索引的进一步说明,请参阅:http://php.net/manual/en/function.preg-match-all.php


对于 OP 的示例输入,preg_match_all() 是不必要的额外工作,它会创建一个具有不必要深度的 $matches 数组。这看起来像是完全忘记了OP在问题中提出的问题。
D
Dave

试试这个,使用 preg_replace

$string = "Hello! 123 test this? 456. done? 100%";
$int = intval(preg_replace('/[^0-9]+/', '', $string), 10);
echo $int;

DEMO


[^0-9] 更简单地写为 \D
为什么 OP 要将所有这些值粉碎成一个值? 123456100 有什么可能的用途?
u
user889030

我们可以从中提取 int 像

$string = 'In My Car_Price : 50660.00';

echo intval(preg_replace('/[^0-9.]/','',$string));  # without number format   output: 50660
echo number_format(intval(preg_replace('/[^0-9.]/','',$string)));  # with number format  output :50,660

演示:http://sandbox.onlinephpfunctions.com/code/82d58b5983e85a0022a99882c7d0de90825aa398


为什么要容纳小数位?你读过OP的问题吗? OP 的用户如何在他们的购物车中拥有一小部分商品?
j
jewelhuq

按照此步骤将字符串转换为数字

$value = '$0025.123';
$onlyNumeric = filter_var($value, FILTER_SANITIZE_NUMBER_FLOAT, FILTER_FLAG_ALLOW_FRACTION);
settype($onlyNumeric,"float");

$result=($onlyNumeric+100);
echo $result;

另一种方法:

$res = preg_replace("/[^0-9.]/", "", "$15645623.095605659");

你读过这个问题吗? OP 的用户如何在他们的购物车中拥有一小部分商品?或者您是否只是阅读了其他答案而忘记查看 OP 实际问了什么?
m
mickmackusa

由于您的字符串中只有 1 个数值要隔离,因此我赞同并亲自将 filter_var()FILTER_SANITIZE_NUMBER_INT 一起使用。

echo filter_var($string, FILTER_SANITIZE_NUMBER_INT);

由于只有 1 个数值并且整数之前的唯一字符是字母、冒号或空格,因此有效的一种更奇葩的技术是使用 ltrim() 和字符掩码,然后将剩余的字符串转换为整数。

Demo

$string = "In My Cart : 11 items";
echo (int)ltrim($string, 'A..z: ');
// 11

如果由于某种原因有多个整数值并且您想获取第一个整数值,那么正则表达式将是一种直接的技术。

Demo

echo preg_match('/\d+/', $string, $m) ? $m[0] : '';

如果您需要将数字字符串显式转换为整数(或浮点数),sscanf() 会非常方便。如果整数值可能/未知出现在字符串的开头,则在扫描之前将非数字字符添加到输入字符串。以下技术匹配前导非数字(并在 % 之后使用 * 忽略它们),然后匹配第一个出现的数字序列并将返回的子字符串转换为整数。

Demo

var_dump(sscanf(' ' . $string, '%*[^0-9]%d')[0]);

要采用这种技术来提取浮点值,只需将 d 更改为 f。有关 sscanf() 的(当前未记录的)分配抑制功能的更多信息,请参阅 this post


M
Mehrdad

其他方式(甚至Unicode字符串):

$res = array();
$str = 'test 1234 555 2.7 string ..... 2.2 3.3';
$str = preg_replace("/[^0-9\.]/", " ", $str);
$str = trim(preg_replace('/\s+/u', ' ', $str));
$arr = explode(' ', $str);
for ($i = 0; $i < count($arr); $i++) {
    if (is_numeric($arr[$i])) {
        $res[] = $arr[$i];
    }
}
print_r($res); //Array ( [0] => 1234 [1] => 555 [2] => 2.7 [3] => 2.2 [4] => 3.3 ) 

F
Francesco Terenzani

sscanf 的替代解决方案:

$str = "In My Cart : 11 items";
list($count) = sscanf($str, 'In My Cart : %s items');

对于这种特定情况(字符串的非数字部分可以预见是静态的),这是一种非常明智的技术。研究人员可能需要了解 %d 也可用于将数字子字符串转换为整数类型数据。 3v4l.org/lLTXv @salathe 提到了 this technique back in 2011,但被 OP 说服删除答案,理由是完整的字符串没有可靠的静态格式。 (文本的可变性应该由 OP 澄清。)
k
kasimir

根据您的用例,这也可能是一个选项:

$str = 'In My Cart : 11 items';
$num = '';

for ($i = 0; $i < strlen($str); $i++) {

    if (is_numeric($str[$i])) {
        $num .= $str[$i];
    }
}

echo $num; // 11

虽然我同意正则表达式或 filter_var() 在所述情况下会更有用。


M
Mehrdad

对于 utf8 字符串:

function unicodeStrDigits($str) {
    $arr = array();
    $sub = '';
    for ($i = 0; $i < strlen($str); $i++) { 
        if (is_numeric($str[$i])) {
            $sub .= $str[$i];
            continue;
        } else {
            if ($sub) {
                array_push($arr, $sub);
                $sub = '';
            }
        }
    }

    if ($sub) {
        array_push($arr, $sub); 
    }

    return $arr;
}

这绝对不是 utf8 准备好,因为您没有使用 mb_ 函数并且您正在使用 [offset] 语法来访问字符串中的每个字节。这个无法解释的答案是“no bueno”。
S
Sajjad Hossain Sagor

如果您不知道数字是哪种格式? int 或 float,然后使用:

$string = '$125.22';

$string2 = '$125';

preg_match_all('/(\d+.?\d+)/',$string,$matches); // $matches[1] = 125.22

preg_match_all('/(\d+.?\d+)/',$string2,$matches); // $matches[1] = 125

捕获组是不必要的,因为您正在捕获全字符串匹配 - 有效地将匹配数组的大小加倍而没有任何好处。如果输入字符串仅以 $ 开头,则只需使用 ltrim($string, '$')
K
Khaldoon Masud

此函数还将处理浮点数

$str = "Doughnuts, 4; doughnuts holes, 0.08; glue, 3.4";
$str = preg_replace('/[^0-9\.]/','-', $str);
$str = preg_replace('/(\-+)(\.\.+)/','-', $str);
$str = trim($str, '-');
$arr = explode('-', $str);

在这个过程中引入连字符是完全没有必要的。这个答案也没有解释它是如何工作的,以及为什么你觉得它是一个好主意。
C
ChrisF

该脚本首先创建一个文件,将数字写入一行,如果获得数字以外的字符,则更改为下一行。最后,它再次将数字整理到一个列表中。

string1 = "hello my name 12 is after 198765436281094and14 and 124de"
f= open("created_file.txt","w+")
for a in string1:
    if a in ['1','2','3','4','5','6','7','8','9','0']:
        f.write(a)
    else:
        f.write("\n" +a+ "\n")
f.close()


#desired_numbers=[x for x in open("created_file.txt")]

#print(desired_numbers)

k=open("created_file.txt","r")
desired_numbers=[]
for x in k:
    l=x.rstrip()
    print(len(l))
    if len(l)==15:
        desired_numbers.append(l)


#desired_numbers=[x for x in k if len(x)==16]
print(desired_numbers)

这个问题似乎是一个与 PHP 相关的问题。