123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340 |
- import numpy
- import scipy.optimize
- import functools
- import fitModel
- class initialValueGenerator:
-
- def __init__(self):
- self.map={'gaus':initialValueGenerator.drawGauss,
- 'loggaus':initialValueGenerator.drawLogGauss,
- 'poisson':initialValueGenerator.drawPoisson,
- 'flat':initialValueGenerator.drawFlat,
- 'const':initialValueGenerator.drawConst}
- def add(self,gtype='gaus',parameters=[0,1],bounds=[-numpy.inf,numpy.inf]):
- v={'type':gtype,'parameters':parameters,'bounds':bounds}
- try:
- self.vals.append(v)
- except AttributeError:
- self.vals=[v]
- def getN(self):
- return len(self.vals)
-
- def drawGauss(p):
- return numpy.random.normal(p[0],p[1])
- def drawLogGauss(p):
- return numpy.power(10,initialValueGenerator.drawGauss(p))
- def drawPoisson(p):
- return numpy.random.poisson(p[0])
-
- def drawFlat(p):
- return p[0]+numpy.random.random()*(p[1]-p[0])
- def drawConst(p):
- return p[0]
- def draw(self,typ,p,lb,ub):
- f=self.map[typ]
- while True:
- x=f(p)
- if x<lb:
- continue
- if x>ub:
- continue
- return x
-
- def generate(self):
- n=len(self.vals)
- par=numpy.zeros(n)
- lb=numpy.zeros(n)
- ub=numpy.zeros(n)
- for i in range(n):
- x=self.vals[i]
- p=x['parameters']
- lb[i]=x['bounds'][0]
- ub[i]=x['bounds'][1]
- par[i]=self.draw(x['type'],p,lb[i],ub[i])
- return par, (lb,ub)
- def getBoundsScalar(self):
- gbounds=[]
- for x in self.vals:
- gbounds.append(x['bounds'])
- return gbounds
- def generateIVF():
- ig=initialValueGenerator()
- ig.add('poisson',[200],[0,numpy.inf])
- ig.add('loggaus',[1,0.5],[0,numpy.inf])
- ig.add('loggaus',[-1,1],[0,numpy.inf])
- ig.add('const',[6,1],[0,numpy.inf])
- ig.add('poisson',[200],[0,numpy.inf])
- ig.add('loggaus',[-2,1],[0,0.1])
- return ig
- def generateIVFFinite():
- ig=initialValueGenerator()
-
- ig.add('poisson',[200],[0,10000])
-
- ig.add('loggaus',[1,0.5],[0,40])
-
- ig.add('loggaus',[-1,1],[0,5])
-
- ig.add('const',[6,1],[6,50])
-
- ig.add('poisson',[200],[0,10000])
-
- ig.add('loggaus',[-2,1],[0,0.1])
- return ig
- def generateCModel():
-
- ig=initialValueGenerator()
- ig.add('loggaus',[-3,1],[0,numpy.inf])
- ig.add('flat',[0,1],[0,1])
- ig.add('loggaus',[-7,2],[0,numpy.inf])
- ig.add('gaus',[10,5],[0,numpy.inf])
- return ig
- def generateCModelFinite():
-
- ig=initialValueGenerator()
-
- ig.add('loggaus',[-3,1],[0,1])
-
- ig.add('flat',[0,1],[0,1])
-
- ig.add('loggaus',[-7,2],[0,1])
-
- ig.add('gaus',[10,2],[5,50])
- return ig
- def generateGauss(fit,bounds,n=1):
- sig=scipy.linalg.sqrtm(fit.cov)
- r=fit.mu.shape[0]
- out=numpy.ones((r,n))
- i=0
- while True:
- s=numpy.matmul(sig,numpy.random.normal(size=r))
- s+=fit.mu
- if in_bounds(s,bounds[0],bounds[1]):
- out[:,i]=s
- i+=1
- if i==n:
- break
- return out
-
-
- def in_bounds(x0,lb,ub):
- r=x0.shape[0]
- for i in range(r):
- if x0[i]<lb[i]:
- return False
- if x0[i]>ub[i]:
- return False
- return True
- def getFit(samples,threshold=numpy.inf,verbose=0):
- class fit:pass
- chi2=samples[0,:]
-
-
- fit.mu=numpy.mean(samples[1:,chi2<threshold],axis=1)
- fit.cov=numpy.cov(samples[1:,chi2<threshold])
- if verbose>0:
- print(fit.mu)
- print(fit.cov)
- return fit
- def fitIVF(t, centers,nfit=10,nbatch=20):
-
-
-
- m=numpy.unravel_index(numpy.argmax(centers),centers.shape)[0]
-
- ivf=centers[m]
-
- w=numpy.ones(t.shape)
- fun=functools.partial(fitModel.fDiff,fitModel.fIVF,t,ivf,w)
- jac=functools.partial(fitModel.jacIVF,t)
-
-
- ig=generateIVF()
- n=ig.getN()
- samples=numpy.zeros((n+1,nfit))
- for j in range(nfit):
- fMin=1e30
- for i in range(nbatch):
- par,bounds=ig.generate()
- result=scipy.optimize.least_squares(fun=fun,x0=par,bounds=bounds,jac=jac)
-
-
- xm=[x for x in result.x]
- xm[1]=numpy.max([result.x[1],1/result.x[2]])
- xm[2]=numpy.max([1/result.x[1],result.x[2]])
- if result.cost<fMin:
- fMin=result.cost
- x1=xm
- samples[1:,j]=x1
- samples[0,j]=fMin
-
- return m,samples
- def fitIVFGlobal(t, centers,nfit=10, qLambda=0):
-
- m=numpy.unravel_index(numpy.argmax(centers),centers.shape)[0]
-
- ivf=centers[m]
-
- w=numpy.ones(t.shape)
-
-
- funScalar=functools.partial(fitModel.fDiffScalar,fitModel.fIVF,t,ivf,w)
-
- funScalarRegularized=functools.partial(fitModel.fDiffScalarRegularized,funScalar,fitModel.fIVFRegA,qLambda)
-
- jac=functools.partial(fitModel.jacIVF,t)
-
- jacScalar=functools.partial(fitModel.jacScalar,fitModel.fIVF,t,ivf,w,jac)
-
- jacScalarRegularized=functools.partial(fitModel.jacScalarRegularized,jacScalar,fitModel.jacIVFRegA,qLambda)
- ig=generateIVFFinite()
- boundsScalar=ig.getBoundsScalar()
-
- n=ig.getN()
- samples=numpy.zeros((n+1,nfit))
-
- minSetup=dict(method='L-BFGS-B',jac=jacScalar)
- if qLambda>0:
- minSetup['jac']=jacScalarRegularized
- for j in range(nfit):
- if qLambda>0:
- result=scipy.optimize.dual_annealing(func=funScalarRegularized,bounds=boundsScalar,minimizer_kwargs=minSetup)
- else:
- result=scipy.optimize.dual_annealing(func=funScalar,bounds=boundsScalar,minimizer_kwargs=minSetup)
- xm=[qx for qx in result.x]
- xm[1]=numpy.max([result.x[1],1/result.x[2]])
- xm[2]=numpy.max([1/result.x[1],result.x[2]])
- if result.x[2]!=xm[2]:
- pass
-
- samples[1:,j]=xm
- samples[0,j]=result.fun
-
- return m,samples
- def fitCompartment(ivfFit, t, qf ,nfit=10,nbatch=20, useJac=False):
-
- igIVF=generateIVF()
- pars,boundsIVF=igIVF.generate()
- ivfSamples=generateGauss(ivfFit,boundsIVF,nfit)
- ig=generateCModel()
- nC=ig.getN()
- boundsScalar=ig.getBoundsScalar()
- samples=numpy.zeros((1+nC+nIVF,nfit))
- for j in range(nfit):
-
- ivfPar=ivfSamples[:,j]
- w=numpy.ones(t.shape)
- fc1=functools.partial(fitModel.fCompartment,ivfPar)
- fun=functools.partial(fitModel.fDiff,fc1,t,qf,w)
- jac=functools.partial(fitModel.jacDep,ivfPar,t)
-
- funScalar=functools.partial(fitData.fDiffScalar,fc1,t,qf,w)
- jacScalar=functools.partial(fitData.jacScalar,fc1,t,qf,w,jac)
-
- fMin=1e30
- for i in range(nbatch):
-
- par,bounds=ig.generate()
-
- if useJac:
- result=scipy.optimize.least_squares(fun=fun,x0=par,bounds=bounds,jac=jac)
-
-
-
- else:
- result=scipy.optimize.least_squares(fun=fun,x0=par,bounds=bounds)
- if result.cost<fMin:
- fMin=result.cost
- x1=result.x
-
- samples[0,j]=fMin
- samples[1:nC+1,j]=x1
- samples[(1+nC):,j]=ivfPar
-
-
- return samples
- def fitCompartmentGlobal(ivfFit, t, qf ,nfit=10,nbatch=20, useJac=False):
-
-
-
-
- igIVF=generateIVF()
- xs,boundsIVF=igIVF.generate()
- ivfSamples=generateGauss(ivfFit,boundsIVF,nfit)
-
- nIVF=igIVF.getN()
- ig=generateCModelFinite()
- nC=ig.getN()
- samples=numpy.zeros((1+nC+nIVF,nfit))
-
- boundsScalar=ig.getBoundsScalar()
-
- scale=1
- fi=qf.sum()
- if fi<0.1:
- scale=10/fi
- print('scale={}'.format(scale))
-
- qf*=scale
- for j in range(nfit):
- ivfPar=ivfSamples[:,j]
- w=numpy.ones(t.shape)
- fc1=functools.partial(fitModel.fCompartment,ivfPar)
- funScalar=functools.partial(fitModel.fDiffScalar,fc1,t,qf,w)
- jac=functools.partial(fitModel.jacDep,ivfPar,t)
- jacScalar=functools.partial(fitModel.jacScalar,fc1,t,qf,w,jac)
- minSetup=dict(method='L-BFGS-B',jac=jacScalar)
-
- result=scipy.optimize.dual_annealing(func=funScalar,bounds=boundsScalar,minimizer_kwargs=minSetup)
- print(f'[{j}/{nfit}]')
-
- samples[0,j]=result.fun/scale
- qx=result.x
- qx[0]/=scale
- qx[1]/=scale
- print(qx)
- samples[1:nC+1,j]=qx
- samples[(1+nC):,j]=ivfPar
-
-
- return samples
|