我使用名为emcee的Python模块来对分发进行采样。我需要将一个名为events
的数组(37,100)(我分别命名为Ntrig和Nsamp)传递给下面的函数。
def mp(SNR2, *events):
events = np.asarray(events).reshape((Ntrig,Nsamp))
bessel = special.iv(0,np.sqrt(x*SNR2(event)))
exp = np.exp(-0.5*(x+SNR2(event)))
I = integrate.quad(lambda x: exp*bessel,0,SNRth**2)[0]
return np.asarray([np.array[I for event in events[i]] for i in range(len(events))]).reshape(events.shape)
我一直收到错误:
ValueError: total size of new array must be unchanged
据我了解,*events
会将events
数组拆分为37 * 100个单独的参数。我不应该重新塑造数组的下一行将它重新放回37 x 100阵列?
P.S。在你问为什么我甚至打扰将events
分解成单独的参数之前 - 模块需要这个才能工作,它不能采用数组。
完整追踪错误:
ValueError Traceback (most recent call last)
<ipython-input-17-c8e815326a69> in <module>()
----> 1 mp(SNR2,events)
<ipython-input-16-9f73f234c628> in mp(SNR2, *events)
5 def mp(SNR2, *events):
6 events = np.asarray(events).reshape((Ntrig,Nsamp))
----> 7 return np.asarray([np.array([integrate.quad(lambda x: np.exp(-0.5*(x+SNR2(event)))*special.iv(0,np.sqrt(x*SNR2(event))),0,SNRth**2)[0] for event in events[i]]) for i in range(len(events))]).reshape(events.shape)
8 # return integrate.quad(lambda x: 0.5*np.exp(-0.5*(x+SNR2(event)))*special.iv(0,np.sqrt(x*SNR2(event))),0,SNRth**2)[0]
9 def pp(SNR2, *events):
/opt/local/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/site-packages/scipy/integrate/quadpack.pyc in quad(func, a, b, args, full_output, epsabs, epsrel, limit, points, weight, wvar, wopts, maxp1, limlst)
279 args = (args,)
280 if (weight is None):
--> 281 retval = _quad(func,a,b,args,full_output,epsabs,epsrel,limit,points)
282 else:
283 retval = _quad_weight(func,a,b,args,full_output,epsabs,epsrel,limlst,limit,maxp1,weight,wvar,wopts)
/opt/local/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/site-packages/scipy/integrate/quadpack.pyc in _quad(func, a, b, args, full_output, epsabs, epsrel, limit, points)
343 if points is None:
344 if infbounds == 0:
--> 345 return _quadpack._qagse(func,a,b,args,full_output,epsabs,epsrel,limit)
346 else:
347 return _quadpack._qagie(func,bound,infbounds,args,full_output,epsabs,epsrel,limit)
<ipython-input-16-9f73f234c628> in <lambda>(x)
5 def mp(SNR2, *events):
6 events = np.asarray(events).reshape((Ntrig,Nsamp))
----> 7 return np.asarray([np.array([integrate.quad(lambda x: np.exp(-0.5*(x+SNR2(event)))*special.iv(0,np.sqrt(x*SNR2(event))),0,SNRth**2)[0] for event in events[i]]) for i in range(len(events))]).reshape(events.shape)
8 # return integrate.quad(lambda x: 0.5*np.exp(-0.5*(x+SNR2(event)))*special.iv(0,np.sqrt(x*SNR2(event))),0,SNRth**2)[0]
9 def pp(SNR2, *events):
<ipython-input-16-9f73f234c628> in SNR2(*events)
1 def SNR2(*events):
----> 2 events = np.asarray(events).reshape((Ntrig,Nsamp))
3 C = 5*np.pi**(-1.33333)*events**(1.66667)/(96*d**2)
4 return C*integrate.quad(lambda f: f**(-2.3333)/S(f), 20, 1500, limit=1000)[0]
5 def mp(SNR2, *events):
ValueError: total size of new array must be unchanged
答案 0 :(得分:3)
据我所知,
events
会将事件数组分解为37 * 100个单独的参数。
事实并非如此。如果您使用
致电mp
mp(SNR2, events)
然后在mp
内,events
将是1元素元组(arr,)
,其中arr
是(37,100)形数组。
如果您使用
致电mp
mp(SNR2, *events)
然后在mp
内,events
将是一个37元素的元组,其中37个元素是(37,100)形数组的37行。
如果您使用
致电mp
mp(SNR2, *events.flat)
然后在mp
内,events
将是37 * 100元素的元组。
请注意,追溯的最后一节说:
<ipython-input-16-9f73f234c628> in SNR2(*events)
1 def SNR2(*events):
----> 2 events = np.asarray(events).reshape((Ntrig,Nsamp))
3 C = 5*np.pi**(-1.33333)*events**(1.66667)/(96*d**2)
4 return C*integrate.quad(lambda f: f**(-2.3333)/S(f), 20, 1500, limit=1000)[0]
5 def mp(SNR2, *events):
ValueError: total size of new array must be unchanged
因此当Python在SNR2
函数中时会引发错误。
由于使用SNR2
在mp
中调用了SNR2(event)
,而event
是一个(37,100)形数组,event
中的SNR2
变量是包含原始数组的1元素元组。这不是你想要的。
修复代码的最简单方法是定义
def SNR2(events):
# no longer needed
# events = np.asarray(events).reshape((Ntrig,Nsamp))
只是按照预期传递事件。
但是,如果您无法更改SNR2
的签名,则必须使用
SNR2(*event.flat)
在mp
函数内。
参考:这是*
解包运算符的an excellent explanation,以及在定义函数和调用函数时如何使用语法。