Converting from m/s to km/h using F# Units of Measure

后端 未结 2 1548
Happy的楠姐
Happy的楠姐 2021-02-15 12:24

I\'m in the process of learning F# - and is currently looking into Units of Measure. I have a simple calculation returning meters per second, and I want to introduce a function

相关标签:
2条回答
  • 2021-02-15 12:42

    First, your msToKmph is totally incorrect. Although it returns a correct return value, what it is actually doing, is it just drops the original <m/s> value by converting to a plain, measureless float and then multiplies the measureless value to a 3.6<km/h>.

    To better express the relations between UoM's, consider this:

    let kmToM = 1000.0<m/km>  // relation between kilometers and meters
    let hrToSec = 3600.0<s/h> // relation between seconds and hours
    let msToKmph(speed : float<m/s>) =
        speed / kmToM * hrToSec
    

    Note, all "magic numbers" are encapsulated within UoM converters, hence your formulas remain clean, e.g. they simply operate values and constants, but the UoM are calculated by the compiler.

    Update: The philosophy of UoM conversion is that the conversion formulas should be something that has physical sense. The rule of thumb is whether your conversion value presents in reference books. In plain English, 3.6<km/h> from above is useless, but 1000.0<m/km> just says, "there is 1000 m in 1 km", which makes sense.

    You can even improve hrToSec like this:

    let hrToSec2 = 60.0<s/minute> * 60.0<minute/h>
    

    This will make every value a well-known value found in reference books.

    0 讨论(0)
  • 2021-02-15 12:54

    You're right that removing unit information is a bad thing. You should create a few constants with appropriate units for conversion.

    let mPerKm = 1000.0<m/km>
    let secondPerHour = 3600.0<s/h>
    
    // val msToKmph : float<m/s> -> float<km/h>
    let msToKmph(speed : float<m/s>) =
        speed / mPerKm * secondPerHour
    

    For km and m, a generic solution is to define a unit prefix k so it works for many UoMs which have kilo as a metric:

    [<Measure>] type k
    
    let kilo = 1000.0<1/k>
    let secondPerHour = 3600.0<s/h>
    
    // val msToKmph : float<m/s> -> float<k m/h>
    let msToKmph(speed : float<m/s>) =
        speed / kilo * secondPerHour
    
    0 讨论(0)
提交回复
热议问题