-
Notifications
You must be signed in to change notification settings - Fork 28
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Conversions and user-defined units #123
Comments
I don't think that
library(units) # branch udunits
# udunits system database from /usr/share/xml/udunits
# Attaching package: ‘units’
# The following object is masked from ‘package:base’:
# %*%
freezing_C <- set_units(0, degree_C)
freezing_F <- set_units(32, degree_F)
boiling_C <- set_units(100, degree_C)
boiling_F <- set_units(212, degree_F)
set_units(freezing_C, degree_F)
# 32 degree_F
set_units(freezing_F, degree_C)
# 3.552714e-14 °C
set_units(boiling_F, degree_C)
# 100 °C
set_units(boiling_C, degree_F)
# 212 degree_F
# now self-constructed my_C, using udunits C API
# v C = 9/5 v + 32 F:
foo <- install_conversion_offset("degree_F", "__C", 32)
foo <- install_conversion_constant("__C", "my_C", 5/9) # interprets constant 1/c
set_units(freezing_F, my_C)
# -2.131628e-14 my_C
set_units(boiling_F, my_C)
# 100 my_C |
I completely agree. Summing up, |
If we can do everything we want with If |
That is an attractive idea; however, udunits2 represents quantities (AFAICT!) as numbers and the powers for each base unit, so for representing 1 mile/gallon it would create a new number, and retain -2 for m. I think that would scare users off! We still have the issue that 1 mg/kg is not retained as such: it now errors, before it would become 1e-6 1. It's the trade-off between usability and being principled. With the > units:::R_ut_format(units:::R_ut_parse("mile/gallon"))
[1] "425143.683171079 m⁻²" |
Could a new function like show_unit_as("m^-2", "mile/gallon") Then whenever units was going to print show_unit_as("m^-2", "mile/gallon", system="fuel efficiency")
show_unit_as("1", "mg/kg", system="drug dosing") It would add potentially significant overhead to printing, but it seems to be a straight-forward implementation of the goal to show the user what they expect. |
I don't know if this is a duplicate, but I didn't find anything exactly like it, and it is something we have discussed before. In any case, it is related to #89, #85, #84, and #67 I think.
While
udunits2
is terrific in that it gives us a lot of units and lets us convert between them, it doesn't always play nice with the unit arithmetic we have. In many ways, I think we could be better off by just extracting the information it has in its tables of conversions and go it alone inunits
. (Especially because I am still running into installation difficulties from time to time when testing packages on Travis, but it has been a little while so it might be resolved by now).The conversion rules I implemented two years ago could almost get us there. They assumed we had a conversion constant between convertible types, and that was that. We got those constants from
udunits2
, true, but nothing would really prevent us from implementing this simple logic from a database, andunits
would have one dependency less.What didn't work in my implementation was cases when we have affine functions for mapping one unit to another, e.g. C to F which works as v C = 9/5 v + 32 F.
I cannot think of any unit conversion that is not affine between some base units, but I stand ready to be corrected.
It isn't much of a problem to implement affine conversion in a very similar way to the linear one, but I got stomped by 0 C + 0 C = 0 C (obviously) but 32 F + 32 F = 64 F (obviously), although these quantities are the same. We agreed that it would be necessary to distinguish between "values" and "quantities", in the sense that both 0 C and 32 F refer to the freezing point of water, but 0 C, as a quantity, is zero change in temperature, while 32 F is not.
At that point, I had gotten sidetracked by another project and didn't get any further with it, and it has been hiding in my TODO list ever since.
Until now, that is. I have thought a little about it today. At first, I thought the problem was harder than it is, but now I think we should just stick with "linear" conversions in arithmetic and affine conversions otherwise.
If you add two temperatures, you must be considering them as quantities. It is not a meaningful operation to add them otherwise, as the C and F example clearly shows. And for quantities, we can always use the linear scaling.
As a mockup implementation of units, consider
and conversion functions
I just have a mockup of C and F conversion, so it is both simpler and more complex than what we actually have implemented right now...
Explicit conversion, using
convert
, will be the affine version. If you explicitly convert from C to F, I will assume this is what you want.This will work as we would expect:
For arithmetic, though, we scale linearly:
With this, we get arithmetic that matches what we would expect, I think.
This semantics of conversion could be implemented in the current framework with little overhead. When I simplify expressions or test for compatibility, I already use a linear scaling. I think
udunit2
is only involved with explicit conversion and with getting scaling constants (but I might misremember here).If linear scaling is what we already have, then we are most of the way to handle all conversions in
units
. We just need to get the affine transformations included as well. If, for all compatible units, we know the scale and the offset, then we have all that we need. In arithmetic, we only use the scale, in explicit conversions we use the offset as well. For units that are already linearly scalable, the offset will just be zero.We don't need to build this table right away. As a first step, we can still use
udunits2
. We get the offset by converting zero and the scale we can get from the offset and converting one.Does this sound reasonble, and is it something you would like me to have a look at?
Cheers
The text was updated successfully, but these errors were encountered: