我在3D空间中有一组x,y,z点和另一个名为charge
的变量,它表示在特定x,y,z坐标中沉积的电荷量。我想对此数据进行加权(通过检测器中沉积的电荷量加权,这恰好是较高的权重,以获得更多的电荷),以使其通过给定点即顶点。
现在,当我对2D进行此操作时,我尝试了各种方法(将顶点带到原点,并对所有其他点进行相同的转换,并迫使拟合通过原点,从而使顶点确实很高重量),但没有一个比Jaime在这里给出的答案要好:How to do a polynomial fit with fixed points
它使用Lagrange乘法器的方法,我在本科的Advanced Multi变量课程中对此很熟悉,但是没有太多其他东西,而且看起来该代码的转换不会像添加az那样容易坐标。 (请注意,即使代码没有考虑到所存费用的数量,它仍然给了我最好的结果)。我想知道是否有相同算法的版本,但使用3D。我还通过Gmail与该答案的作者联系,但没有收到他的回音。
以下是有关我的数据以及我要在2D模式中尝试做的事情的更多信息:How to weigh the points in a scatter plot for a fit?
这是我的代码,用于执行以下操作:强制顶点位于原点,然后拟合数据设置fit_intercept=False
。我目前正在使用此方法处理2D数据,因为我不确定是否存在用于Lagrange乘法器的3D版本,但是例如可以使用线性回归方法在3D中进行此操作,例如:Fitting a line in 3D:
import numpy as np
import sklearn.linear_model
def plot_best_fit(image_array, vertexX, vertexY):
weights = np.array(image_array)
x = np.where(weights>0)[1]
y = np.where(weights>0)[0]
size = len(image_array) * len(image_array[0])
y = np.zeros((len(image_array), len(image_array[0])))
for i in range(len(np.where(weights>0)[0])):
y[np.where(weights>0)[0][i]][np.where(weights>0)[1][i]] = np.where(weights>0)[0][i]
y = y.reshape(size)
x = np.array(range(len(image_array)) * len(image_array[0]))
weights = weights.reshape((size))
for i in range(len(x)):
x[i] -= vertexX
y[i] -= vertexY
model = sklearn.linear_model.LinearRegression(fit_intercept=False)
model.fit(x.reshape((-1, 1)),y,sample_weight=weights)
line_x = np.linspace(0, 512, 100).reshape((-1,1))
pred = model.predict(line_x)
m, b = np.polyfit(np.linspace(0, 512, 100), np.array(pred), 1)
angle = math.atan(m) * 180/math.pi
return line_x, pred, angle, b, m
image_array
是一个numpy数组,vertexX
和vertexY
分别是顶点的x和y坐标。这是我的数据:https://uploadfiles.io/bbhxo。我无法创建玩具数据,因为没有简单的方法可以复制此数据,它是通过Geant4模拟中微子与氩原子核相互作用产生的。我不想摆脱数据的复杂性。而且这个特定事件恰好是我的代码无法正常工作的事件,我不确定是否可以专门生成数据,因此我的代码无法正常工作。
答案 0 :(得分:3)
这是使用基本优化的手工制作解决方案。很简单。只需使用基本http://simple.com/file.php
来测量点到要拟合的线的距离,并最小化加权距离。代码如下:
optimize.leastsq
返回
# -*- coding: utf-8 -*
from matplotlib import pyplot as plt
from mpl_toolkits.mplot3d import Axes3D
import matplotlib.cm as cm
from scipy import optimize
import numpy as np
def rnd( a ):
return a * ( 1 - 2 * np.random.random() )
def affine_line( s, theta, phi, x0, y0, z0 ):
a = np.sin( theta) * np.cos( phi )
b = np.sin( theta) * np.sin( phi )
c = np.cos( theta )
return np.array( [ s * a + x0, s * b + y0, s * c + z0 ] )
def point_to_line_distance( x , y, z , theta, phi, x0, y0, z0 ):
xx = x - x0
yy = y - y0
zz = z - z0
a = np.sin( theta) * np.cos( phi )
b = np.sin( theta) * np.sin( phi )
c = np.cos( theta )
r = np.array( [ xx, yy, zz ] )
t = np.array( [ a, b, c ] )
return np.linalg.norm( r - np.dot( r, t) * t )
def residuals( parameters, fixpoint, data, weights=None ):
theta, phi = parameters
x0, y0, z0 = fixpoint
if weights is None:
w = np.ones( len( data ) )
else:
w = np.array( weights )
diff = np.array( [ point_to_line_distance( x , y, z , theta, phi , *fixpoint ) for x, y, z in data ] )
diff = diff * w
return diff
### some test data
fixpoint = [ 1, 2 , -.3 ]
trueline = np.array( [ affine_line( s, .7, 1.7, *fixpoint ) for s in np.linspace( -1, 2, 50 ) ] )
rndData = np.array( [ np.array( [ a + rnd( .6), b + rnd( .35 ), c + rnd( .45 ) ] ) for a, b, c in trueline ] )
zData = [ 20 * point_to_line_distance( x , y, z , .7, 1.7, *fixpoint ) for x, y, z in rndData ]
### unweighted
bestFitValuesUW, ier= optimize.leastsq(residuals, [ 0, 0],args=( fixpoint, rndData ) )
print bestFitValuesUW
uwLine = np.array( [ affine_line( s, bestFitValuesUW[0], bestFitValuesUW[1], *fixpoint ) for s in np.linspace( -2, 2, 50 ) ] )
### weighted ( chose inverse distance as weight....would be charge in OP's case )
bestFitValuesW, ier= optimize.leastsq(residuals, [ 0, 0],args=( fixpoint, rndData, [ 1./s for s in zData ] ) )
print bestFitValuesW
wLine = np.array( [ affine_line( s, bestFitValuesW[0], bestFitValuesW[1], *fixpoint ) for s in np.linspace( -2, 2, 50 ) ] )
### plotting
fig = plt.figure()
ax = fig.add_subplot( 1, 1, 1, projection='3d' )
ax.plot( *np.transpose(trueline ) )
ax.scatter( *fixpoint, color='k' )
ax.scatter( rndData[::,0], rndData[::,1], rndData[::,2] , c=zData, cmap=cm.jet )
ax.plot( *np.transpose( uwLine ) )
ax.plot( *np.transpose( wLine ) )
ax.set_xlim( [ 0, 2.5 ] )
ax.set_ylim( [ 1, 3.5 ] )
ax.set_zlim( [ -1.25, 1.25 ] )
plt.show()
固定点显示为黑色。原始线为蓝色。未加权和加权拟合分别为橙色和绿色。数据根据到行的距离着色。