启发式对PHP中任何“种类”的技术度量进行排序

时间:2019-03-19 17:35:47

标签: php measurement

我有不同的列表,它们具有相同的尺寸,但是有点像

“ 1 m,200 mm,1 ft”

或者也许

“ 1°C,273 K”,等等。

现在我想按绝对顺序对它们进行排序

“ 200 mm,1 ft,1 m”和“ 273 K,1°C”

我想知道这是否已经解决了,因为我不想重新发明轮子。恐怕,这可能是某种“购买PHP扩展”的问题,但我已经找到了一些有用的软件包:

https://github.com/PhpUnitsOfMeasure/php-units-of-measure可以在度量单位之间进行各种对话。

我已经创建了用于区分单位和编号的代码。

所以我在想,要把部队“蛮力”到其中的某个维度:

https://github.com/PhpUnitsOfMeasure/php-units-of-measure/tree/master/source/PhysicalQuantity

接下来,我可以选择第一个维度并将所有内容转换为第一个“主要” SI单位并进行排序。

对吗?

2 个答案:

答案 0 :(得分:2)

通常,您需要做的是将这些单位转换为某种通用度量,但仅出于排序目的。

使用usort()和自定义的回调函数。在您的回调中,为了进行比较而进行转换。

尽管返回结果时请务必保留原始单位,否则四舍五入的误差会逐渐蔓延。

答案 1 :(得分:0)

这是我根据建议提出的解决方案

public function testCompareLength()
{
    $this->assertLessThan(0, $this->objectDe->compareFunction('100 mm', '1 m'));
}

public function testCompareTemperature()
{
    $this->assertLessThan(0, $this->objectDe->compareFunction('1 K', '0 °C'));
    $this->assertGreaterThan(0, $this->objectDe->compareFunction('0 °C', '1 K'));
    $this->assertEquals(0, $this->objectDe->compareFunction('-273 °C', '0 K'));
}

/**
 * @param $numberString
 *
 * @return array
 */
public function parseNumber($numberString): array
{
    $values = preg_split('/(?<=[0-9.,])(?=[^0-9,.]+)/i', $numberString);

    $float = $values[0];
    $unit = $values[1] ?? '';

    $decPos = strpos($float, '.');
    if ($decPos === false) {
        $precision = 0;
    } else {
        $precision = strlen($float) - $decPos - 1;
    }

    return ['float' => $float, 'unit' => $unit, 'precision' => $precision];
}


private function heuristicMeasureFactory($measure)
{
    $prioritizedDimensions = [
        Temperature::class,
        Length::class,
    ];

    $unit = trim($measure['unit']);

    foreach ($prioritizedDimensions as $class) {
        foreach ($class::getUnitDefinitions() as $definition) {
            if ($definition->getName() == $unit) {
                return new $class($measure['float'], $unit);
            }
        }
    }

    // now process aliases
    foreach ($prioritizedDimensions as $class) {
        foreach ($class::getUnitDefinitions() as $definition) {
            foreach ($definition->aliases as $alias) {
                if ($alias == $unit) {
                    return new $class($measure['float'], $unit);
                }
            }
        }
    }

    return null; // NaN
}

/**
 * Sort apples and oranges -- kind of. Not.
 *
 * Compares two strings which represend a measurement of the same physical dimension
 */
public function compareFunction($a, $b)
{
    $definitions = Temperature::getUnitDefinitions();

    $aParsed = $this->parseNumber($a);
    $aVal = $this->heuristicMeasureFactory($aParsed);

    $bParsed = $this->parseNumber($b);
    $bVal = $this->heuristicMeasureFactory($bParsed);

    if ($aVal == null || $bVal == null) {
        return strnatcmp($aVal, $bVal); // fallback to string comparision
    }

    return bccomp($aVal->subtract($bVal)->toNativeUnit(), 0, 36);
}