"""
Uncertainty propagation class, and log() and exp() functions.
Based on scalars or numpy vectors, this class allows you to store and
manipulate values+uncertainties, with propagation of gaussian error for
addition, subtraction, multiplication, division, power, exp() and log().
Storage properties are determined by the numbers used to set the value
and uncertainty. Be sure to use floating point uncertainty vectors
for inplace operations since numpy does not do automatic type conversion.
Normal operations can use mixed integer and floating point. In place
operations (a *= b, etc.) create at most one extra copy for each operation.
c = a*b by contrast uses four intermediate vectors, so shouldn't be used
for huge arrays.
"""
from __future__ import division
import numpy
import err1d
from formatnum import format_uncertainty
__all__ = ['Uncertainty']
# TODO: rename to Measurement and add support for units?
# TODO: C implementation of *,/,**?
[docs]class Uncertainty(object):
# Make standard deviation available
def _getdx(self): return numpy.sqrt(self.variance)
def _setdx(self,dx):
# Direct operation
# variance = dx**2
# Indirect operation to avoid temporaries
self.variance[:] = dx
self.variance **= 2
dx = property(_getdx,_setdx,doc="standard deviation")
# Constructor
def __init__(self, x, variance=None):
self.x, self.variance = x, variance
# Numpy array slicing operations
def __len__(self):
return len(self.x)
def __getitem__(self,key):
return Uncertainty(self.x[key],self.variance[key])
def __setitem__(self,key,value):
self.x[key] = value.x
self.variance[key] = value.variance
def __delitem__(self, key):
del self.x[key]
del self.variance[key]
#def __iter__(self): pass # Not sure we need iter
# Normal operations: may be of mixed type
def __add__(self, other):
if isinstance(other,Uncertainty):
return Uncertainty(*err1d.add(self.x,self.variance,other.x,other.variance))
else:
return Uncertainty(self.x+other, self.variance+0) # Force copy
def __sub__(self, other):
if isinstance(other,Uncertainty):
return Uncertainty(*err1d.sub(self.x,self.variance,other.x,other.variance))
else:
return Uncertainty(self.x-other, self.variance+0) # Force copy
def __mul__(self, other):
if isinstance(other,Uncertainty):
return Uncertainty(*err1d.mul(self.x,self.variance,other.x,other.variance))
else:
return Uncertainty(self.x*other, self.variance*other**2)
def __truediv__(self, other):
if isinstance(other,Uncertainty):
return Uncertainty(*err1d.div(self.x,self.variance,other.x,other.variance))
else:
return Uncertainty(self.x/other, self.variance/other**2)
def __pow__(self, other):
if isinstance(other,Uncertainty):
# Haven't calcuated variance in (a+/-da) ** (b+/-db)
return NotImplemented
else:
return Uncertainty(*err1d.pow(self.x,self.variance,other))
# Reverse operations
def __radd__(self, other):
return Uncertainty(self.x+other, self.variance+0) # Force copy
def __rsub__(self, other):
return Uncertainty(other-self.x, self.variance+0)
def __rmul__(self, other):
return Uncertainty(self.x*other, self.variance*other**2)
def __rtruediv__(self, other):
x,variance = err1d.pow(self.x,self.variance,-1)
return Uncertainty(x*other,variance*other**2)
def __rpow__(self, other): return NotImplemented
# In-place operations: may be of mixed type
def __iadd__(self, other):
if isinstance(other,Uncertainty):
self.x,self.variance \
= err1d.add_inplace(self.x,self.variance,other.x,other.variance)
else:
self.x+=other
return self
def __isub__(self, other):
if isinstance(other,Uncertainty):
self.x,self.variance \
= err1d.sub_inplace(self.x,self.variance,other.x,other.variance)
else:
self.x-=other
return self
def __imul__(self, other):
if isinstance(other,Uncertainty):
self.x, self.variance \
= err1d.mul_inplace(self.x,self.variance,other.x,other.variance)
else:
self.x *= other
self.variance *= other**2
return self
def __itruediv__(self, other):
if isinstance(other,Uncertainty):
self.x,self.variance \
= err1d.div_inplace(self.x,self.variance,other.x,other.variance)
else:
self.x /= other
self.variance /= other**2
return self
def __ipow__(self, other):
if isinstance(other,Uncertainty):
# Haven't calcuated variance in (a+/-da) ** (b+/-db)
return NotImplemented
else:
self.x,self.variance = err1d.pow_inplace(self.x, self.variance, other)
return self
# Use true division instead of integer division
def __div__(self, other): return self.__truediv__(other)
def __rdiv__(self, other): return self.__rtruediv__(other)
def __idiv__(self, other): return self.__itruediv__(other)
# Unary ops
def __neg__(self):
return Uncertainty(-self.x,self.variance)
def __pos__(self):
return self
def __abs__(self):
return Uncertainty(numpy.abs(self.x),self.variance)
def __str__(self):
#return str(self.x)+" +/- "+str(numpy.sqrt(self.variance))
if numpy.isscalar(self.x):
return format_uncertainty(self.x,numpy.sqrt(self.variance))
else:
return [format_uncertainty(v,dv)
for v,dv in zip(self.x,numpy.sqrt(self.variance))]
def __repr__(self):
return "Uncertainty(%s,%s)"%(str(self.x),str(self.variance))
# Not implemented
def __floordiv__(self, other): return NotImplemented
def __mod__(self, other): return NotImplemented
def __divmod__(self, other): return NotImplemented
def __mod__(self, other): return NotImplemented
def __lshift__(self, other): return NotImplemented
def __rshift__(self, other): return NotImplemented
def __and__(self, other): return NotImplemented
def __xor__(self, other): return NotImplemented
def __or__(self, other): return NotImplemented
def __rfloordiv__(self, other): return NotImplemented
def __rmod__(self, other): return NotImplemented
def __rdivmod__(self, other): return NotImplemented
def __rmod__(self, other): return NotImplemented
def __rlshift__(self, other): return NotImplemented
def __rrshift__(self, other): return NotImplemented
def __rand__(self, other): return NotImplemented
def __rxor__(self, other): return NotImplemented
def __ror__(self, other): return NotImplemented
def __ifloordiv__(self, other): return NotImplemented
def __imod__(self, other): return NotImplemented
def __idivmod__(self, other): return NotImplemented
def __imod__(self, other): return NotImplemented
def __ilshift__(self, other): return NotImplemented
def __irshift__(self, other): return NotImplemented
def __iand__(self, other): return NotImplemented
def __ixor__(self, other): return NotImplemented
def __ior__(self, other): return NotImplemented
def __invert__(self): return NotImplmented # For ~x
def __complex__(self): return NotImplmented
def __int__(self): return NotImplmented
def __long__(self): return NotImplmented
def __float__(self): return NotImplmented
def __oct__(self): return NotImplmented
def __hex__(self): return NotImplmented
def __index__(self): return NotImplmented
def __coerce__(self): return NotImplmented
[docs] def log(self):
return Uncertainty(*err1d.log(self.x,self.variance))
[docs] def exp(self):
return Uncertainty(*err1d.exp(self.x,self.variance))
def log(val): return self.log()
def exp(val): return self.exp()
def test():
a = Uncertainty(5,3)
b = Uncertainty(4,2)
# Scalar operations
z = a+4
assert z.x == 5+4 and z.variance == 3
z = a-4
assert z.x == 5-4 and z.variance == 3
z = a*4
assert z.x == 5*4 and z.variance == 3*4**2
z = a/4
assert z.x == 5./4 and z.variance == 3./4**2
# Reverse scalar operations
z = 4+a
assert z.x == 4+5 and z.variance == 3
z = 4-a
assert z.x == 4-5 and z.variance == 3
z = 4*a
assert z.x == 4*5 and z.variance == 3*4**2
z = 4/a
assert z.x == 4./5 and abs(z.variance - 3./5**4 * 4**2) < 1e-15
# Power operations
z = a**2
assert z.x == 5**2 and z.variance == 4*3*5**2
z = a**1
assert z.x == 5**1 and z.variance == 3
z = a**0
assert z.x == 5**0 and z.variance == 0
z = a**-1
assert z.x == 5**-1 and abs(z.variance - 3./5**4) < 1e-15
# Binary operations
z = a+b
assert z.x == 5+4 and z.variance == 3+2
z = a-b
assert z.x == 5-4 and z.variance == 3+2
z = a*b
assert z.x == 5*4 and z.variance == (5**2*2 + 4**2*3)
z = a/b
assert z.x == 5./4 and abs(z.variance - (3./5**2 + 2./4**2)*(5./4)**2) < 1e-15
# ===== Inplace operations =====
# Scalar operations
y = a+0; y += 4
z = a+4
assert y.x == z.x and abs(y.variance-z.variance) < 1e-15
y = a+0; y -= 4
z = a-4
assert y.x == z.x and abs(y.variance-z.variance) < 1e-15
y = a+0; y *= 4
z = a*4
assert y.x == z.x and abs(y.variance-z.variance) < 1e-15
y = a+0; y /= 4
z = a/4
assert y.x == z.x and abs(y.variance-z.variance) < 1e-15
# Power operations
y = a+0; y **= 4
z = a**4
assert y.x == z.x and abs(y.variance-z.variance) < 1e-15
# Binary operations
y = a+0; y += b
z = a+b
assert y.x == z.x and abs(y.variance-z.variance) < 1e-15
y = a+0; y -= b
z = a-b
assert y.x == z.x and abs(y.variance-z.variance) < 1e-15
y = a+0; y *= b
z = a*b
assert y.x == z.x and abs(y.variance-z.variance) < 1e-15
y = a+0; y /= b
z = a/b
assert y.x == z.x and abs(y.variance-z.variance) < 1e-15
# =============== vector operations ================
# Slicing
z = Uncertainty(numpy.array([1,2,3,4,5]),numpy.array([2,1,2,3,2]))
assert z[2].x == 3 and z[2].variance == 2
assert (z[2:4].x == [3,4]).all()
assert (z[2:4].variance == [2,3]).all()
z[2:4] = Uncertainty(numpy.array([8,7]),numpy.array([4,5]))
assert z[2].x == 8 and z[2].variance == 4
A = Uncertainty(numpy.array([a.x]*2),numpy.array([a.variance]*2))
B = Uncertainty(numpy.array([b.x]*2),numpy.array([b.variance]*2))
# TODO complete tests of copy and inplace operations for vectors and slices.
# Binary operations
z = A+B
assert (z.x == 5+4).all() and (z.variance == 3+2).all()
z = A-B
assert (z.x == 5-4).all() and (z.variance == 3+2).all()
z = A*B
assert (z.x == 5*4).all() and (z.variance == (5**2*2 + 4**2*3)).all()
z = A/B
assert (z.x == 5./4).all()
assert (abs(z.variance - (3./5**2 + 2./4**2)*(5./4)**2) < 1e-15).all()
# printing; note that sqrt(3) ~ 1.7
assert str(Uncertainty(5,3)) == "5.0(17)"
assert str(Uncertainty(15,3)) == "15.0(17)"
assert str(Uncertainty(151.23356,0.324185**2)) == "151.23(32)"
if __name__ == "__main__": test()