为什么Tensorflow无法解决此类问题?

时间:2018-09-14 02:15:32

标签: tensorflow 3d numerical-methods

data.py用于生成样本数据。 test.py是用于参数求解的主要代码。

3D空间中的对象移动可以用4x4矩阵描述,而这种矩阵可以用6个参数表示。 我有一组对象位置数据,以及对象转换后的另一组最终位置数据。我想将tensorflow用于这6个参数的数值计算。当我运行以下代码(test.py)时,它显示“模型训练:无”错误。有人可以帮我调试吗?这类矩阵计算问题是张量流的限制吗?

data.py

# coding: utf-8

# In[1]:


import numpy as np
import tensorflow
import pandas as pd


# In[2]:


def sample_points(low=0,high=100,N=10000):
    '''how to stack 1darray to 2darray'''
    x = np.random.randint(low,high,N)
    y = x*2
    z = np.random.randint(low,high,N)
    L = np.full((1,N),1,dtype=np.int)
    return np.array([x,y,z,[1 for i in range(N)]])


# In[3]:


xyzL_h=sample_points()
print(xyzL_h)


# In[4]:


def save_points(xyzL,filename='xyz.txt',homogenous=False):
    '''
    save 
    xyz  to ?.csv file, or xyz1 to ?.csv file
    '''
    xyzL_v = np.transpose(xyzL,(1,0))
    #print(xyzL_v.shape)
    df = pd.DataFrame(xyzL_v)
    df.columns=['x','y','z','1']
    if not homogenous:
        df.iloc[:,0:3].to_csv(filename,index=False,header=False)
    else:
        df.to_csv(filename,index=False,header=False)


# In[5]:


save_points(xyzL_h,filename='raw_xyz.txt')


# In[6]:


import matplotlib.pyplot as plt
get_ipython().magic('matplotlib inline')


# In[7]:


def show_data(xyzL_h):
    plt.plot(xyzL_h[0],xyzL_h[1],'r-')
    plt.plot(xyzL_h[0],xyzL_h[2],'b+')


# In[8]:


show_data(xyzL_h)


# In[9]:


class Screw():
    '''
    based on screw thoery of screw motion and transformation matrix 
    '''
    def __init__(self,lib):
        self._lib = lib
        self._p = None
        self._v = None
        self._w = None
        self._seta = 0
        self._m = None

    def dot(self,x,y):
        if self._lib.__name__ is 'numpy':
            return self._lib.dot(x,y)
        if self._lib.__name__ is 'tensorflow':
            #print("rank of y: ",y.shape,len(y.shape))
            if len(y.shape)==1:
                y = tensorflow.reshape(y,(-1,1))
            return self._lib.matmul(x,y)

    def cross(self,x,y):
        '''
        cross product of two vertors
        '''
        x_n = self._lib.reshape(x,(1,3))
        y_n = self._lib.reshape(y,(1,3))
        return self._lib.cross(x_n,y_n)

    def transform(self,cloud):
        '''
        cloud with shape of [4,None]
        '''
        return self.dot(self._m,cloud)

    def I(self,n):
        '''
        unit matrix
        '''
        if self._lib.__name__ == 'numpy':
            return self._lib.identity(n)
        elif self._lib.__name__ == 'tensorflow':
            return self._lib.matrix_diag([1.0 for i in range(n)])

    def w_from_rad(self,angle):
        '''
        define vector by sphere coordinate
        first of angle(alpha): is radian angle between direction vector and Z axis
        second of angle(beta): is radian angle between X-O-Y projection of vector and X axis
        '''
        lib = self._lib
        alpha,beta = angle[0],angle[1]
        k = lib.cos(alpha)
        i = lib.sin(alpha)*lib.cos(beta)
        j = lib.sin(alpha)*lib.sin(beta)
        if lib.__name__ == 'numpy':
            ijk = lib.array([i,j,k])
        elif lib.__name__ == 'tensorflow':
            ijk = lib.stack([i,j,k])
        self._w = ijk
        return ijk

    def wx(self,w):
        '''                
        #skew symmetric matrix of twist component[0:3] w 
        '''
        if self._lib.__name__ is 'numpy':
            m = self._lib.zeros((3,3))
            m[0,1]=-w[2]
            m[0,2]=w[1]
            m[1,0]=w[2]
            m[1,2]=-w[0]
            m[2,0]=-w[1]
            m[2,1]=w[0]
        elif self._lib.__name__ is 'tensorflow':
            m = self._lib.stack([[0,-w[2],w[1]],[w[2],0,-w[0]],[-w[1],w[0],0]])
           # m = self._lib.reshape(m,(3,3))
        return m

    def v(self,w,p):
        '''
        # twist component[3:6] v
        '''
        _v = self.dot(self.wx(w),p)
        self._v = -_v
        return -_v


    def tm(self,w,p,seta):
        '''
        transformation matrix out of screw theory
        this matrix is used for right multiplication operation of point cloud
        '''
        lib = self._lib
        if lib.__name__ is 'numpy':
            w = lib.array(w)
            p = lib.array(p)
        if lib.__name__ is 'tensorflow':
            w = lib.to_float(lib.Variable(w))
            p = lib.to_float(lib.Variable(p))            
        _v = self.v(w,p)
        _wx = self.wx(w)
        self._w = w
        self._v = _v
        self._p = p
        m33 = self.I(3)+_wx*lib.sin(seta)+self.dot(_wx,_wx)*(1-lib.cos(seta))
        mm = self.I(3)-m33
        #+w*self.dot(w,_v)*seta #'w*np.dot(w,_v)*seta' should be 0, when no translation movement 
        vv = self._lib.reshape(self.cross(w,_v),(-1,1))

        t = self.dot(mm,vv)[:,0]
        if lib.__name__ is 'numpy':
            m44 = self.I(4)
            m44[0:3,0:3]=m33
            m44[0:3,3]=t
            self._m = m44#lib.transpose(m44,(1,0))
        elif lib.__name__ is 'tensorflow':
            #print('t:',t)
            m34 = self._lib.stack([m33[:,0],m33[:,1],m33[:,2],t],axis=1)
            m44 = self._lib.stack([m34[0],m34[1],m34[2],[0.0,0.0,0.0,1.0]])
            self._m = m44#lib.transpose(m44,perm=[1,0])
        return self._m


# In[10]:


screw = Screw(np)
q,seta=[0.0,100.0,0.0],np.pi/2
w =screw.w_from_rad([np.pi/4,np.pi/4])
res = screw.tm(w,q,seta)


# In[11]:


print(res)


# In[12]:


class CloudProcess():
    def __init__(self,cloud,screw):
        self._cloud = cloud
        self._matrix = screw
    def transform(self):
        new_cloud = self._matrix.transform(self._cloud)
        return new_cloud


# In[13]:


op = CloudProcess(xyzL_h,screw)
xyzL_h_new = op.transform()
save_points(xyzL_h_new,filename='new_xyz.txt')


# In[14]:


screw = Screw(tensorflow)
clouds = tensorflow.placeholder(dtype=tensorflow.float32,shape=[4,None],name='XYZL')
p = tensorflow.Variable([0,100,0],name='PointOfAxis',dtype=tensorflow.float32)
angle = tensorflow.Variable([np.pi/4,np.pi/4],name='DirectionAngle',dtype=tensorflow.float32)
seta = tensorflow.constant(np.pi/2,dtype=tensorflow.float32)
w = screw.w_from_rad(angle)
m = screw.tm(w,p,seta)
op = CloudProcess(clouds,screw)
xyzL_new = op.transform()


# In[15]:


init = tensorflow.global_variables_initializer()
with tensorflow.Session() as sess:
    sess.run(init)
    res=sess.run(xyzL_new,feed_dict={clouds:xyzL_h})
    save_points(res,filename='new_xyz_tf.txt')


# In[ ]:

test.py

# coding: utf-8

# In[1]:


import tensorflow
import pandas as pd
import numpy as np


# In[2]:


def load_pointXYZ(filename):
    df = pd.read_csv(filename,header=None)
    df['1']=1
    df.columns = ['X','Y','Z','1']
    return df


# In[3]:


df1 = load_pointXYZ('raw_xyz.txt')
df2 = load_pointXYZ('new_xyz_tf.txt')


# In[4]:


X = df1.as_matrix().T
Y = df2.as_matrix().T
print(X)
print(Y)


# In[5]:


class Screw():
    '''
    based on screw thoery of screw motion and transformation matrix 
    '''
    def __init__(self,lib):
        self._lib = lib
        self._p = None
        self._v = None
        self._w = None
        self._seta = 0
        self._m = None

    def dot(self,x,y):
        if self._lib.__name__ is 'numpy':
            return self._lib.dot(x,y)
        if self._lib.__name__ is 'tensorflow':
            #print("rank of y: ",y.shape,len(y.shape))
            if len(y.shape)==1:
                y = tensorflow.reshape(y,(-1,1))
            return self._lib.matmul(x,y)

    def cross(self,x,y):
        '''
        cross product of two vertors
        '''
        x_n = self._lib.reshape(x,(1,3))
        y_n = self._lib.reshape(y,(1,3))
        return self._lib.cross(x_n,y_n)

    def transform(self,cloud):
        '''
        cloud with shape of [4,None]
        '''
        return self.dot(self._m,cloud)

    def I(self,n):
        '''
        unit matrix
        '''
        if self._lib.__name__ == 'numpy':
            return self._lib.identity(n)
        elif self._lib.__name__ == 'tensorflow':
            return self._lib.matrix_diag([1.0 for i in range(n)])

    def w_from_rad(self,angle):
        '''
        define vector by sphere coordinate
        first of angle(alpha): is radian angle between direction vector and Z axis
        second of angle(beta): is radian angle between X-O-Y projection of vector and X axis
        '''
        lib = self._lib
        alpha,beta = angle[0],angle[1]
        k = lib.cos(alpha)
        i = lib.sin(alpha)*lib.cos(beta)
        j = lib.sin(alpha)*lib.sin(beta)
        if lib.__name__ == 'numpy':
            ijk = lib.array([i,j,k])
        elif lib.__name__ == 'tensorflow':
            ijk = lib.stack([i,j,k])
        self._w = ijk
        return ijk

    def wx(self,w):
        '''                
        #skew symmetric matrix of twist component[0:3] w 
        '''
        if self._lib.__name__ is 'numpy':
            m = self._lib.zeros((3,3))
            m[0,1]=-w[2]
            m[0,2]=w[1]
            m[1,0]=w[2]
            m[1,2]=-w[0]
            m[2,0]=-w[1]
            m[2,1]=w[0]
        elif self._lib.__name__ is 'tensorflow':
            m = self._lib.stack([[0,-w[2],w[1]],[w[2],0,-w[0]],[-w[1],w[0],0]])
           # m = self._lib.reshape(m,(3,3))
        return m

    def v(self,w,p):
        '''
        # twist component[3:6] v
        '''
        _v = self.dot(self.wx(w),p)
        self._v = -_v
        return -_v


    def tm(self,w,p,seta):
        '''
        transformation matrix out of screw theory
        this matrix is used for right multiplication operation of point cloud
        '''
        lib = self._lib
        if lib.__name__ is 'numpy':
            w = lib.array(w)
            p = lib.array(p)
        if lib.__name__ is 'tensorflow':
            w = lib.to_float(lib.Variable(w))
            p = lib.to_float(lib.Variable(p))            
        _v = self.v(w,p)
        _wx = self.wx(w)
        self._w = w
        self._v = _v
        self._p = p
        m33 = self.I(3)+_wx*lib.sin(seta)+self.dot(_wx,_wx)*(1-lib.cos(seta))
        mm = self.I(3)-m33
        #+w*self.dot(w,_v)*seta #'w*np.dot(w,_v)*seta' should be 0, when no translation movement 
        vv = self._lib.reshape(self.cross(w,_v),(-1,1))

        t = self.dot(mm,vv)[:,0]
        if lib.__name__ is 'numpy':
            m44 = self.I(4)
            m44[0:3,0:3]=m33
            m44[0:3,3]=t
            self._m = m44#lib.transpose(m44,(1,0))
        elif lib.__name__ is 'tensorflow':
            m34 = self._lib.stack([m33[:,0],m33[:,1],m33[:,2],t],axis=1)
            m44 = self._lib.stack([m34[0],m34[1],m34[2],[0.0,0.0,0.0,1.0]])
            self._m = m44#lib.transpose(m44,perm=[1,0])
        return self._m


# In[6]:


class CloudProcess():
    def __init__(self,cloud,screw):
        self._cloud = cloud
        self._matrix = screw
    def transform(self):
        new_cloud = self._matrix.transform(self._cloud)
        return new_cloud


# In[7]:


screw = Screw(tensorflow)
cld_1 = tensorflow.placeholder(dtype=tensorflow.float32,shape=[4,None],name='XYZL')
cld_2 = tensorflow.placeholder(dtype=tensorflow.float32,shape=[4,None],name='XYZL')
p = tensorflow.Variable([0,50,2],name='PointOfAxis',dtype=tensorflow.float32)
angle = tensorflow.Variable([0.3,0.2],name='DirectionAngle',dtype=tensorflow.float32)
seta = tensorflow.constant(np.pi/2,dtype=tensorflow.float32)
w = screw.w_from_rad(angle)
m = screw.tm(w,p,seta)
op = CloudProcess(cld_1,screw)
xyzL_new = op.transform()
loss = tensorflow.reduce_sum(tensorflow.abs(xyzL_new-cld_2))
optimizer = tensorflow.train.GradientDescentOptimizer(learning_rate=0.1)
model = optimizer.minimize(loss)

init = tensorflow.global_variables_initializer()
with tensorflow.Session() as sess:
    sess.run(init)
    for i in range(10):
        res = sess.run(model,feed_dict={cld_1:X,cld_2:Y})
        print('model tranning:',res)
    res = sess.run(loss,feed_dict={cld_1:X,cld_2:Y})
    print('loss:',res)
    print(angle.eval())
    print(p.eval() )

0 个答案:

没有答案