oop - PHP 7.4 带浮点数/双精度数的类型化属性
问题描述
如果这是一个蹩脚的问题,我提前道歉,这是我的第一个问题。我正在为一个大学项目构建一个小框架,我想尽可能地强制执行类型,很高兴看到 PHP 7.4 对属性有严格的类型,
但即使声明 strict_types 也没有正确执行。
顺便说一句,我知道人们说在 PHP 中双精度和浮点数之间没有区别,但是对于类型化属性,PHP 不将双精度识别为数据类型。
请参阅下面的简单代码测试:
class FloatTest
{
private float $float;
private int $int;
function __construct()
{
}
public function setFloat(float $float):void
{
$this->float = $float;
}
public function getFloat()
{
return $this->float;
}
public function setInt(int $int):void
{
$this->int = $int;
}
public function getInt():int
{
return $this->int;
}
}
$ft = new FloatTest();
$ft->setFloat(8);//No error is thrown, converts to float but no decimals
$ft->getFloat();// Returns number 8 as float, but there is no enforcing of decimal point?
var_dump(is_float(8));//returns false
//Argument 1 passed to FloatTest::setInt() must be of the type int, float given
$ft->setInt(8.2);//Works as expected, great!
class DoubleTest
{
private double $double;
function __construct()
{
# code...
}
public function setDouble(double $double):void
{
$this->double = $double;
}
public function getDouble():double
{
return $this->double;
}
}
$dt = new DoubleTest();
//Argument 1 passed to DoubleTest::setDouble() must be an instance of double, int given:
$dt->setDouble(8);
$double = $dt->getDouble();
var_dump(is_double(8)); returns false
基于这个非常简单的测试,我有几点我觉得很奇怪:
为什么 PHP 正确执行 int 类型而不是 float 类型?为什么当我使用 is_float() 检查时它返回 false 但函数接受整数?
为什么整数类型可以完美执行,而不是浮点数?
即使 PHP 有一个有效的 double 数据类型,为什么它假定 double 是一个实例?double 绝对是 PHP 中的原始数据类型,因为 is_double() 函数完美运行,参见上面抛出的异常。
基本上,在 PHP 中强制执行十进制数字的最佳、最干净的工作是什么?
解决方案
为什么 PHP 正确执行 int 类型而不是 float 类型?为什么当我使用 is_float() 检查时它返回 false 但函数接受整数?
因为浮点范围可以包含整数而不会丢失任何数据。换句话说,浮点数就像整数的超集,所以你不能说我只想要浮点数我不想要整数,即使在强类型语言中,即使你的代码处于strict_types
模式。
为什么整数类型可以完美执行,而不是浮点数?
PHP 类型声明带有强制模式作为默认模式,因此可以更改声明的类型,并且只有当 PHP 可以将错误类型的值强制转换为预期值时,才会抛出 TypeError。
在您的情况下,您将8
(整数)传递给需要浮点数的方法,这完全没问题,PHP 不会抱怨它,因为强制8
浮点数不会改变任何东西/丢失任何东西。
那么做什么strict_types
呢?
它将 PHP 的行为从强制变为严格。这意味着当涉及的操作可能导致数据丢失时,PHP 将不会容忍。
以您为例,当我们设置declare(strict_types=1)
以下行时会出现问题
$ft->setInt(8.2);//Works as expected, great!
因为我们试图将float
数字传递给int
参数,这意味着数据丢失(8.2 变为 8)PHP 将抛出异常
致命错误:未捕获的类型错误:传递给 FloatTest::setInt() 的参数 1 必须是 int 类型,给定浮点数
即使 PHP 有一个有效的 double 数据类型,为什么它假定 double 是一个实例?double 绝对是 PHP 中的原始数据类型,因为 is_double() 函数可以完美运行
PHP没有double
数据类型,is_double
只是一个别名is_float()
基本上,在 PHP 中强制执行十进制数字的最佳、最干净的工作是什么?
你所做的是最好和最干净的工作:)
推荐阅读
- python - 唯一用户的时间序列
- excel - Excel IF/THEN/VLOOKUP 嵌套公式
- jupyter-notebook - jupyter ipython notebook 无法打开正常运行的 qtconsole
- postgresql - postgres regexp_matches 奇怪的行为
- python - 在 python 中通过身份验证从视频 url 获取帧
- powerbi - Power BI如何编辑数据源连接SQL语句
- reactjs - 将函数/状态变量传递给 React 组件(箭头符号)
- emacs - 当前行高亮(hl-line)不尊重现有高亮
- html - 我应该使用
- python - Dask IndexError:列表索引超出范围