3113 lines
72 KiB
C++
3113 lines
72 KiB
C++
//////////////////////////////////////////////////////////////////////
|
||
// Matrix.cpp
|
||
//
|
||
// 操作矩阵的类 CMatrix 的实现文件
|
||
////////////////////////////////////////////////////////////////////////
|
||
// Matrix.cpp
|
||
//
|
||
// 操作矩阵的类 CMatrix 的实现文件
|
||
//
|
||
// 编制, 2002/8
|
||
//////////////////////////////////////////////////////////////////////
|
||
|
||
#include "Matrix1.h"
|
||
#include <math.h>
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// Construction/Destruction
|
||
//////////////////////////////////////////////////////////////////////
|
||
|
||
|
||
/////////////////////////////////////////////////////////////////////
|
||
// Construction/Destruction
|
||
//////////////////////////////////////////////////////////////////////
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 基本构造函数
|
||
//////////////////////////////////////////////////////////////////////
|
||
CMatrix1::CMatrix1()
|
||
{
|
||
m_nNumColumns = 1;
|
||
m_nNumRows = 1;
|
||
m_pData = NULL;
|
||
BOOL bSuccess = Init(m_nNumRows, m_nNumColumns);
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 指定行列构造函数
|
||
//
|
||
// 参数:
|
||
// 1. int nRows - 指定的矩阵行数
|
||
// 2. int nCols - 指定的矩阵列数
|
||
//////////////////////////////////////////////////////////////////////
|
||
CMatrix1::CMatrix1(int nRows, int nCols)
|
||
{
|
||
m_nNumRows = nRows;
|
||
m_nNumColumns = nCols;
|
||
m_pData = NULL;
|
||
BOOL bSuccess = Init(m_nNumRows, m_nNumColumns);
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 指定值构造函数
|
||
//
|
||
// 参数:
|
||
// 1. int nRows - 指定的矩阵行数
|
||
// 2. int nCols - 指定的矩阵列数
|
||
// 3. float value[] - 一维数组,长度为nRows*nCols,存储矩阵各元素的值
|
||
//////////////////////////////////////////////////////////////////////
|
||
CMatrix1::CMatrix1(int nRows, int nCols, float value[])
|
||
{
|
||
m_nNumRows = nRows;
|
||
m_nNumColumns = nCols;
|
||
m_pData = NULL;
|
||
BOOL bSuccess = Init(m_nNumRows, m_nNumColumns);
|
||
// ASSERT(bSuccess);
|
||
|
||
SetData(value);
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 方阵构造函数
|
||
//
|
||
// 参数:
|
||
// 1. int nSize - 方阵行列数
|
||
//////////////////////////////////////////////////////////////////////
|
||
CMatrix1::CMatrix1(int nSize)
|
||
{
|
||
m_nNumRows = nSize;
|
||
m_nNumColumns = nSize;
|
||
m_pData = NULL;
|
||
BOOL bSuccess = Init(nSize, nSize);
|
||
// ASSERT (bSuccess);
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 方阵构造函数
|
||
//
|
||
// 参数:
|
||
// 1. int nSize - 方阵行列数
|
||
// 2. float value[] - 一维数组,长度为nRows*nRows,存储方阵各元素的值
|
||
//////////////////////////////////////////////////////////////////////
|
||
CMatrix1::CMatrix1(int nSize, float value[])
|
||
{
|
||
m_nNumRows = nSize;
|
||
m_nNumColumns = nSize;
|
||
m_pData = NULL;
|
||
BOOL bSuccess = Init(nSize, nSize);
|
||
// ASSERT (bSuccess);
|
||
|
||
SetData(value);
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 拷贝构造函数
|
||
//
|
||
// 参数:
|
||
// 1. const CMatrix& other - 源矩阵
|
||
//////////////////////////////////////////////////////////////////////
|
||
CMatrix1::CMatrix1(const CMatrix1& other)
|
||
{
|
||
m_nNumColumns = other.GetNumColumns();
|
||
m_nNumRows = other.GetNumRows();
|
||
m_pData = NULL;
|
||
BOOL bSuccess = Init(m_nNumRows, m_nNumColumns);
|
||
// ASSERT(bSuccess);
|
||
|
||
// copy the pointer
|
||
memcpy(m_pData, other.m_pData, sizeof(float)*m_nNumColumns*m_nNumRows);
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 析构函数
|
||
//////////////////////////////////////////////////////////////////////
|
||
CMatrix1::~CMatrix1()
|
||
{
|
||
if (m_pData)
|
||
{
|
||
delete[] m_pData;
|
||
m_pData = NULL;
|
||
}
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 初始化函数
|
||
//
|
||
// 参数:
|
||
// 1. int nRows - 指定的矩阵行数
|
||
// 2. int nCols - 指定的矩阵列数
|
||
//
|
||
// 返回值:BOOL 型,初始化是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::Init(int nRows, int nCols)
|
||
{
|
||
if (m_pData)
|
||
{
|
||
delete[] m_pData;
|
||
m_pData = NULL;
|
||
}
|
||
|
||
m_nNumRows = nRows;
|
||
m_nNumColumns = nCols;
|
||
int nSize = nCols*nRows;
|
||
if (nSize < 0)
|
||
return FALSE;
|
||
|
||
// 分配内存
|
||
m_pData = new float[nSize];
|
||
|
||
if (m_pData == NULL)
|
||
return FALSE; // 内存分配失败
|
||
// if (IsBadReadPtr(m_pData, sizeof(float) * nSize))
|
||
// return FALSE;
|
||
|
||
// 将各元素值置0
|
||
memset(m_pData, 0, sizeof(float) * nSize);
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 将方阵初始化为单位矩阵
|
||
//
|
||
// 参数:
|
||
// 1. int nSize - 方阵行列数
|
||
//
|
||
// 返回值:BOOL 型,初始化是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::MakeUnitMatrix(int nSize)
|
||
{
|
||
if (! Init(nSize, nSize))
|
||
return FALSE;
|
||
|
||
for (int i=0; i<nSize; ++i)
|
||
for (int j=0; j<nSize; ++j)
|
||
if (i == j)
|
||
SetElement(i, j, 1);
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 将字符串转化为矩阵的值
|
||
//
|
||
// 参数:
|
||
// 1. CString s - 数字和分隔符构成的字符串
|
||
// 2. const CString& sDelim - 数字之间的分隔符,默认为空格
|
||
// 3. BOOL bLineBreak - 行与行之间是否有回车换行符,默认为真(有换行符)
|
||
// 当该参数为FALSE时,所有元素值都在一行中输入,字符串的第一个
|
||
// 数值应为矩阵的行数,第二个数值应为矩阵的列数
|
||
//
|
||
// 返回值:BOOL 型,转换是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::FromString(CString s, const CString& sDelim /*= " "*/, BOOL bLineBreak /*= TRUE*/)
|
||
{
|
||
if (s=="")
|
||
return FALSE;
|
||
|
||
// 分行处理
|
||
if (bLineBreak)
|
||
{
|
||
CTokenizer1 tk(s, "\r\n");
|
||
|
||
CStringList ListRow;
|
||
CString sRow;
|
||
while (tk.Next(sRow))
|
||
{
|
||
sRow.TrimLeft();
|
||
sRow.TrimRight();
|
||
if (sRow=="")
|
||
break;
|
||
|
||
ListRow.append(sRow);
|
||
}
|
||
|
||
// 行数
|
||
m_nNumRows = ListRow.count();
|
||
|
||
sRow = ListRow.at(0);
|
||
CTokenizer1 tkRow(sRow, sDelim);
|
||
CString sElement;
|
||
// 列数
|
||
m_nNumColumns = 0;
|
||
while (tkRow.Next(sElement))
|
||
{
|
||
m_nNumColumns++;
|
||
}
|
||
|
||
// 初始化矩阵
|
||
if (! Init(m_nNumRows, m_nNumColumns))
|
||
return FALSE;
|
||
|
||
// 设置值
|
||
for (int i=0; i<m_nNumRows; i++)
|
||
{
|
||
sRow = ListRow.at(i);
|
||
int j = 0;
|
||
CTokenizer1 tkRow(sRow, sDelim);
|
||
while (tkRow.Next(sElement))
|
||
{
|
||
sElement.TrimLeft();
|
||
sElement.TrimRight();
|
||
float v = atof(sElement.GetString());
|
||
SetElement(i, j++, v);
|
||
}
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
// 不分行(单行)处理
|
||
|
||
CTokenizer1 tk(s, sDelim);
|
||
|
||
CString sElement;
|
||
|
||
// 行数
|
||
tk.Next(sElement);
|
||
sElement.TrimLeft();
|
||
sElement.TrimRight();
|
||
m_nNumRows = atoi(sElement.GetString());
|
||
|
||
// 列数
|
||
tk.Next(sElement);
|
||
sElement.TrimLeft();
|
||
sElement.TrimRight();
|
||
m_nNumColumns = atoi(sElement.GetString());
|
||
|
||
// 初始化矩阵
|
||
if (! Init(m_nNumRows, m_nNumColumns))
|
||
return FALSE;
|
||
|
||
// 设置值
|
||
int i = 0, j = 0;
|
||
while (tk.Next(sElement))
|
||
{
|
||
sElement.TrimLeft();
|
||
sElement.TrimRight();
|
||
float v = atof(sElement.GetString());
|
||
SetElement(i, j++, v);
|
||
|
||
if (j == m_nNumColumns)
|
||
{
|
||
j = 0;
|
||
i++;
|
||
if (i == m_nNumRows)
|
||
break;
|
||
}
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 将矩阵各元素的值转化为字符串
|
||
//
|
||
// 参数:
|
||
// 1. const CString& sDelim - 数字之间的分隔符,默认为空格
|
||
// 2 BOOL bLineBreak - 行与行之间是否有回车换行符,默认为真(有换行符)
|
||
//
|
||
// 返回值:CString 型,转换得到的字符串
|
||
//////////////////////////////////////////////////////////////////////
|
||
CString CMatrix1::ToString(const CString& sDelim /*= " "*/, BOOL bLineBreak /*= TRUE*/) const
|
||
{
|
||
CString s="";
|
||
|
||
for (int i=0; i<m_nNumRows; ++i)
|
||
{
|
||
for (int j=0; j<m_nNumColumns; ++j)
|
||
{
|
||
CString ss;
|
||
ss.Format("%f", GetElement(i, j));
|
||
s += ss;
|
||
|
||
if (bLineBreak)
|
||
{
|
||
if (j != m_nNumColumns-1)
|
||
s += sDelim;
|
||
}
|
||
else
|
||
{
|
||
if (i != m_nNumRows-1 || j != m_nNumColumns-1)
|
||
s += sDelim;
|
||
}
|
||
}
|
||
if (bLineBreak)
|
||
if (i != m_nNumRows-1)
|
||
s += "\r\n";
|
||
}
|
||
|
||
return s;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 将矩阵指定行中各元素的值转化为字符串
|
||
//
|
||
// 参数:
|
||
// 1. int nRow - 指定的矩阵行,nRow = 0表示第一行
|
||
// 2. const CString& sDelim - 数字之间的分隔符,默认为空格
|
||
//
|
||
// 返回值:CString 型,转换得到的字符串
|
||
//////////////////////////////////////////////////////////////////////
|
||
CString CMatrix1::RowToString(int nRow, const CString& sDelim /*= " "*/) const
|
||
{
|
||
CString s ="";
|
||
|
||
if (nRow >= m_nNumRows)
|
||
return s;
|
||
|
||
for (int j=0; j<m_nNumColumns; ++j)
|
||
{
|
||
CString ss;
|
||
ss.Format("%f", GetElement(nRow, j));
|
||
s += ss;
|
||
if (j != m_nNumColumns-1)
|
||
s += sDelim;
|
||
}
|
||
|
||
return s;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 将矩阵指定列中各元素的值转化为字符串
|
||
//
|
||
// 参数:
|
||
// 1. int nCol - 指定的矩阵行,nCol = 0表示第一列
|
||
// 2. const CString& sDelim - 数字之间的分隔符,默认为空格
|
||
//
|
||
// 返回值:CString 型,转换得到的字符串
|
||
//////////////////////////////////////////////////////////////////////
|
||
CString CMatrix1::ColToString(int nCol, const CString& sDelim /*= " "*/) const
|
||
{
|
||
CString s = "";
|
||
|
||
if (nCol >= m_nNumColumns)
|
||
return s;
|
||
|
||
for (int i=0; i<m_nNumRows; ++i)
|
||
{
|
||
CString ss;
|
||
ss.Format("%f", GetElement(i, nCol));
|
||
s += ss;
|
||
if (i != m_nNumRows-1)
|
||
s += sDelim;
|
||
}
|
||
|
||
return s;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 设置矩阵各元素的值
|
||
//
|
||
// 参数:
|
||
// 1. float value[] - 一维数组,长度为m_nNumColumns*m_nNumRows,存储
|
||
// 矩阵各元素的值
|
||
//
|
||
// 返回值:无
|
||
//////////////////////////////////////////////////////////////////////
|
||
void CMatrix1::SetData(float value[])
|
||
{
|
||
// empty the memory
|
||
memset(m_pData, 0, sizeof(float) * m_nNumColumns*m_nNumRows);
|
||
// copy data
|
||
memcpy(m_pData, value, sizeof(float)*m_nNumColumns*m_nNumRows);
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 设置指定元素的值
|
||
//
|
||
// 参数:
|
||
// 1. int nRows - 指定的矩阵行数
|
||
// 2. int nCols - 指定的矩阵列数
|
||
// 3. float value - 指定元素的值
|
||
//
|
||
// 返回值:BOOL 型,说明设置是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::SetElement(int nRow, int nCol, float value)
|
||
{
|
||
if (nCol < 0 || nCol >= m_nNumColumns || nRow < 0 || nRow >= m_nNumRows)
|
||
return FALSE; // array bounds error
|
||
if (m_pData == NULL)
|
||
return FALSE; // bad pointer error
|
||
|
||
m_pData[nCol + nRow * m_nNumColumns] = value;
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 设置指定元素的值
|
||
//
|
||
// 参数:
|
||
// 1. int nRows - 指定的矩阵行数
|
||
// 2. int nCols - 指定的矩阵列数
|
||
//
|
||
// 返回值:float 型,指定元素的值
|
||
//////////////////////////////////////////////////////////////////////
|
||
float CMatrix1::GetElement(int nRow, int nCol) const
|
||
{
|
||
// ASSERT(nCol >= 0 && nCol < m_nNumColumns && nRow >= 0 && nRow < m_nNumRows); // array bounds error
|
||
// ASSERT(m_pData);
|
||
if(!m_pData) return 0;// bad pointer error
|
||
return m_pData[nCol + nRow * m_nNumColumns] ;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 获取矩阵的列数
|
||
//
|
||
// 参数:无
|
||
//
|
||
// 返回值:int 型,矩阵的列数
|
||
//////////////////////////////////////////////////////////////////////
|
||
int CMatrix1::GetNumColumns() const
|
||
{
|
||
return m_nNumColumns;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 获取矩阵的行数
|
||
//
|
||
// 参数:无
|
||
//
|
||
// 返回值:int 型,矩阵的行数
|
||
//////////////////////////////////////////////////////////////////////
|
||
int CMatrix1::GetNumRows() const
|
||
{
|
||
return m_nNumRows;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 获取矩阵的数据
|
||
//
|
||
// 参数:无
|
||
//
|
||
// 返回值:float型指针,指向矩阵各元素的数据缓冲区
|
||
//////////////////////////////////////////////////////////////////////
|
||
float* CMatrix1::GetData() const
|
||
{
|
||
return m_pData;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 获取指定行的向量
|
||
//
|
||
// 参数:
|
||
// 1. int nRows - 指定的矩阵行数
|
||
// 2. float* pVector - 指向向量中各元素的缓冲区
|
||
//
|
||
// 返回值:int 型,向量中元素的个数,即矩阵的列数
|
||
//////////////////////////////////////////////////////////////////////
|
||
int CMatrix1::GetRowVector(int nRow, float* pVector) const
|
||
{
|
||
if (pVector == NULL)
|
||
delete pVector;
|
||
|
||
pVector = new float[m_nNumColumns];
|
||
// ASSERT(pVector != NULL);
|
||
|
||
for (int j=0; j<m_nNumColumns; ++j)
|
||
pVector[j] = GetElement(nRow, j);
|
||
|
||
return m_nNumColumns;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 获取指定列的向量
|
||
//
|
||
// 参数:
|
||
// 1. int nCols - 指定的矩阵列数
|
||
// 2. float* pVector - 指向向量中各元素的缓冲区
|
||
//
|
||
// 返回值:int 型,向量中元素的个数,即矩阵的行数
|
||
//////////////////////////////////////////////////////////////////////
|
||
int CMatrix1::GetColVector(int nCol, float* pVector) const
|
||
{
|
||
if (pVector == NULL)
|
||
delete pVector;
|
||
|
||
pVector = new float[m_nNumRows];
|
||
|
||
for (int i=0; i<m_nNumRows; ++i)
|
||
pVector[i] = GetElement(i, nCol);
|
||
|
||
return m_nNumRows;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 重载运算符=,给矩阵赋值
|
||
//
|
||
// 参数:
|
||
// 1. const CMatrix& other - 用于给矩阵赋值的源矩阵
|
||
//
|
||
// 返回值:CMatrix型的引用,所引用的矩阵与other相等
|
||
//////////////////////////////////////////////////////////////////////
|
||
CMatrix1& CMatrix1::operator=(const CMatrix1& other)
|
||
{
|
||
if (&other != this)
|
||
{
|
||
BOOL bSuccess = Init(other.GetNumRows(), other.GetNumColumns());
|
||
|
||
// copy the pointer
|
||
memcpy(m_pData, other.m_pData, sizeof(float)*m_nNumColumns*m_nNumRows);
|
||
}
|
||
|
||
// finally return a reference to ourselves
|
||
return *this ;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 重载运算符==,判断矩阵是否相等
|
||
//
|
||
// 参数:
|
||
// 1. const CMatrix& other - 用于比较的矩阵
|
||
//
|
||
// 返回值:BOOL 型,两个矩阵相等则为TRUE,否则为FALSE
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::operator==(const CMatrix1& other) const
|
||
{
|
||
// 首先检查行列数是否相等
|
||
if (m_nNumColumns != other.GetNumColumns() || m_nNumRows != other.GetNumRows())
|
||
return FALSE;
|
||
|
||
for (int i=0; i<m_nNumRows; ++i)
|
||
{
|
||
for (int j=0; j<m_nNumColumns; ++j)
|
||
{
|
||
if (GetElement(i, j) != other.GetElement(i, j))
|
||
return FALSE;
|
||
}
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 重载运算符!=,判断矩阵是否不相等
|
||
//
|
||
// 参数:
|
||
// 1. const CMatrix& other - 用于比较的矩阵
|
||
//
|
||
// 返回值:BOOL 型,两个不矩阵相等则为TRUE,否则为FALSE
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::operator!=(const CMatrix1& other) const
|
||
{
|
||
return !(*this == other);
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 重载运算符+,实现矩阵的加法
|
||
//
|
||
// 参数:
|
||
// 1. const CMatrix& other - 与指定矩阵相加的矩阵
|
||
//
|
||
// 返回值:CMatrix型,指定矩阵与other相加之和
|
||
//////////////////////////////////////////////////////////////////////
|
||
CMatrix1 CMatrix1::operator+(const CMatrix1& other) const
|
||
{
|
||
// 首先检查行列数是否相等
|
||
// ASSERT (m_nNumColumns == other.GetNumColumns() && m_nNumRows == other.GetNumRows());
|
||
|
||
// 构造结果矩阵
|
||
CMatrix1 result(*this) ; // 拷贝构造
|
||
// 矩阵加法
|
||
for (int i = 0 ; i < m_nNumRows ; ++i)
|
||
{
|
||
for (int j = 0 ; j < m_nNumColumns; ++j)
|
||
result.SetElement(i, j, result.GetElement(i, j) + other.GetElement(i, j)) ;
|
||
}
|
||
|
||
return result ;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 重载运算符-,实现矩阵的减法
|
||
//
|
||
// 参数:
|
||
// 1. const CMatrix& other - 与指定矩阵相减的矩阵
|
||
//
|
||
// 返回值:CMatrix型,指定矩阵与other相减之差
|
||
//////////////////////////////////////////////////////////////////////
|
||
CMatrix1 CMatrix1::operator-(const CMatrix1& other) const
|
||
{
|
||
// 首先检查行列数是否相等
|
||
// ASSERT (m_nNumColumns == other.GetNumColumns() && m_nNumRows == other.GetNumRows());
|
||
|
||
// 构造目标矩阵
|
||
CMatrix1 result(*this) ; // copy ourselves
|
||
// 进行减法操作
|
||
for (int i = 0 ; i < m_nNumRows ; ++i)
|
||
{
|
||
for (int j = 0 ; j < m_nNumColumns; ++j)
|
||
result.SetElement(i, j, result.GetElement(i, j) - other.GetElement(i, j)) ;
|
||
}
|
||
|
||
return result ;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 重载运算符*,实现矩阵的数乘
|
||
//
|
||
// 参数:
|
||
// 1. float value - 与指定矩阵相乘的实数
|
||
//
|
||
// 返回值:CMatrix型,指定矩阵与value相乘之积
|
||
//////////////////////////////////////////////////////////////////////
|
||
CMatrix1 CMatrix1::operator*(float value) const
|
||
{
|
||
// 构造目标矩阵
|
||
CMatrix1 result(*this) ; // copy ourselves
|
||
// 进行数乘
|
||
for (int i = 0 ; i < m_nNumRows ; ++i)
|
||
{
|
||
for (int j = 0 ; j < m_nNumColumns; ++j)
|
||
result.SetElement(i, j, result.GetElement(i, j) * value) ;
|
||
}
|
||
|
||
return result ;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 重载运算符*,实现矩阵的乘法
|
||
//
|
||
// 参数:
|
||
// 1. const CMatrix& other - 与指定矩阵相乘的矩阵
|
||
//
|
||
// 返回值:CMatrix型,指定矩阵与other相乘之积
|
||
//////////////////////////////////////////////////////////////////////
|
||
CMatrix1 CMatrix1::operator*(const CMatrix1& other) const
|
||
{
|
||
// 首先检查行列数是否符合要求
|
||
// ASSERT (m_nNumColumns == other.GetNumRows());
|
||
|
||
// construct the object we are going to return
|
||
CMatrix1 result(m_nNumRows, other.GetNumColumns()) ;
|
||
|
||
// 矩阵乘法,即
|
||
//
|
||
// [A][B][C] [G][H] [A*G + B*I + C*K][A*H + B*J + C*L]
|
||
// [D][E][F] * [I][J] = [D*G + E*I + F*K][D*H + E*J + F*L]
|
||
// [K][L]
|
||
//
|
||
float value ;
|
||
for (int i = 0 ; i < result.GetNumRows() ; ++i)
|
||
{
|
||
for (int j = 0 ; j < other.GetNumColumns() ; ++j)
|
||
{
|
||
value = 0.0 ;
|
||
for (int k = 0 ; k < m_nNumColumns ; ++k)
|
||
{
|
||
value += GetElement(i, k) * other.GetElement(k, j) ;
|
||
}
|
||
|
||
result.SetElement(i, j, value) ;
|
||
}
|
||
}
|
||
|
||
return result ;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 复矩阵的乘法
|
||
//
|
||
// 参数:
|
||
// 1. const CMatrix& AR - 左边复矩阵的实部矩阵
|
||
// 2. const CMatrix& AI - 左边复矩阵的虚部矩阵
|
||
// 3. const CMatrix& BR - 右边复矩阵的实部矩阵
|
||
// 4. const CMatrix& BI - 右边复矩阵的虚部矩阵
|
||
// 5. CMatrix& CR - 乘积复矩阵的实部矩阵
|
||
// 6. CMatrix& CI - 乘积复矩阵的虚部矩阵
|
||
//
|
||
// 返回值:BOOL型,复矩阵乘法是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::CMul(const CMatrix1& AR, const CMatrix1& AI, const CMatrix1& BR, const CMatrix1& BI, CMatrix1& CR, CMatrix1& CI) const
|
||
{
|
||
// 首先检查行列数是否符合要求
|
||
if (AR.GetNumColumns() != AI.GetNumColumns() ||
|
||
AR.GetNumRows() != AI.GetNumRows() ||
|
||
BR.GetNumColumns() != BI.GetNumColumns() ||
|
||
BR.GetNumRows() != BI.GetNumRows() ||
|
||
AR.GetNumColumns() != BR.GetNumRows())
|
||
return FALSE;
|
||
|
||
// 构造乘积矩阵实部矩阵和虚部矩阵
|
||
CMatrix1 mtxCR(AR.GetNumRows(), BR.GetNumColumns()), mtxCI(AR.GetNumRows(), BR.GetNumColumns());
|
||
// 复矩阵相乘
|
||
for (int i=0; i<AR.GetNumRows(); ++i)
|
||
{
|
||
for (int j=0; j<BR.GetNumColumns(); ++j)
|
||
{
|
||
float vr = 0;
|
||
float vi = 0;
|
||
for (int k =0; k<AR.GetNumColumns(); ++k)
|
||
{
|
||
float p = AR.GetElement(i, k) * BR.GetElement(k, j);
|
||
float q = AI.GetElement(i, k) * BI.GetElement(k, j);
|
||
float s = (AR.GetElement(i, k) + AI.GetElement(i, k)) * (BR.GetElement(k, j) + BI.GetElement(k, j));
|
||
vr += p - q;
|
||
vi += s - p - q;
|
||
}
|
||
mtxCR.SetElement(i, j, vr);
|
||
mtxCI.SetElement(i, j, vi);
|
||
}
|
||
}
|
||
|
||
CR = mtxCR;
|
||
CI = mtxCI;
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 矩阵的转置
|
||
//
|
||
// 参数:无
|
||
//
|
||
// 返回值:CMatrix型,指定矩阵转置矩阵
|
||
//////////////////////////////////////////////////////////////////////
|
||
CMatrix1 CMatrix1::Transpose() const
|
||
{
|
||
// 构造目标矩阵
|
||
CMatrix1 Trans(m_nNumColumns, m_nNumRows);
|
||
|
||
// 转置各元素
|
||
for (int i = 0 ; i < m_nNumRows ; ++i)
|
||
{
|
||
for (int j = 0 ; j < m_nNumColumns ; ++j)
|
||
Trans.SetElement(j, i, GetElement(i, j)) ;
|
||
}
|
||
|
||
return Trans;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 实矩阵求逆的全选主元高斯-约当法
|
||
//
|
||
// 参数:无
|
||
//
|
||
// 返回值:BOOL型,求逆是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::InvertGaussJordan()
|
||
{
|
||
int *pnRow, *pnCol,i,j,k,l,u,v;
|
||
float d = 0, p = 0;
|
||
|
||
// 分配内存
|
||
pnRow = new int[m_nNumColumns];
|
||
pnCol = new int[m_nNumColumns];
|
||
if (pnRow == NULL || pnCol == NULL)
|
||
return FALSE;
|
||
|
||
// 消元
|
||
for (k=0; k<=m_nNumColumns-1; k++)
|
||
{
|
||
d=0.0;
|
||
for (i=k; i<=m_nNumColumns-1; i++)
|
||
{
|
||
for (j=k; j<=m_nNumColumns-1; j++)
|
||
{
|
||
l=i*m_nNumColumns+j; p=fabs(m_pData[l]);
|
||
if (p>d)
|
||
{
|
||
d=p;
|
||
pnRow[k]=i;
|
||
pnCol[k]=j;
|
||
}
|
||
}
|
||
}
|
||
|
||
// 失败
|
||
if (d == 0.0)
|
||
{
|
||
delete[] pnRow;
|
||
delete[] pnCol;
|
||
return FALSE;
|
||
}
|
||
|
||
if (pnRow[k] != k)
|
||
{
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
{
|
||
u=k*m_nNumColumns+j;
|
||
v=pnRow[k]*m_nNumColumns+j;
|
||
p=m_pData[u];
|
||
m_pData[u]=m_pData[v];
|
||
m_pData[v]=p;
|
||
}
|
||
}
|
||
|
||
if (pnCol[k] != k)
|
||
{
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
u=i*m_nNumColumns+k;
|
||
v=i*m_nNumColumns+pnCol[k];
|
||
p=m_pData[u];
|
||
m_pData[u]=m_pData[v];
|
||
m_pData[v]=p;
|
||
}
|
||
}
|
||
|
||
l=k*m_nNumColumns+k;
|
||
m_pData[l]=1.0/m_pData[l];
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
{
|
||
if (j != k)
|
||
{
|
||
u=k*m_nNumColumns+j;
|
||
m_pData[u]=m_pData[u]*m_pData[l];
|
||
}
|
||
}
|
||
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
if (i!=k)
|
||
{
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
{
|
||
if (j!=k)
|
||
{
|
||
u=i*m_nNumColumns+j;
|
||
m_pData[u]=m_pData[u]-m_pData[i*m_nNumColumns+k]*m_pData[k*m_nNumColumns+j];
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
if (i!=k)
|
||
{
|
||
u=i*m_nNumColumns+k;
|
||
m_pData[u]=-m_pData[u]*m_pData[l];
|
||
}
|
||
}
|
||
}
|
||
|
||
// 调整恢复行列次序
|
||
for (k=m_nNumColumns-1; k>=0; k--)
|
||
{
|
||
if (pnCol[k]!=k)
|
||
{
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
{
|
||
u=k*m_nNumColumns+j;
|
||
v=pnCol[k]*m_nNumColumns+j;
|
||
p=m_pData[u];
|
||
m_pData[u]=m_pData[v];
|
||
m_pData[v]=p;
|
||
}
|
||
}
|
||
|
||
if (pnRow[k]!=k)
|
||
{
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
u=i*m_nNumColumns+k;
|
||
v=i*m_nNumColumns+pnRow[k];
|
||
p=m_pData[u];
|
||
m_pData[u]=m_pData[v];
|
||
m_pData[v]=p;
|
||
}
|
||
}
|
||
}
|
||
|
||
// 清理内存
|
||
delete[] pnRow;
|
||
delete[] pnCol;
|
||
|
||
// 成功返回
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 复矩阵求逆的全选主元高斯-约当法
|
||
//
|
||
// 参数:
|
||
// 1. CMatrix& mtxImag - 复矩阵的虚部矩阵,当前矩阵为复矩阵的实部
|
||
//
|
||
// 返回值:BOOL型,求逆是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::InvertGaussJordan(CMatrix1& mtxImag)
|
||
{
|
||
int *pnRow,*pnCol,i,j,k,l,u,v,w;
|
||
float p,q,s,t,d,b;
|
||
|
||
// 分配内存
|
||
pnRow = new int[m_nNumColumns];
|
||
pnCol = new int[m_nNumColumns];
|
||
if (pnRow == NULL || pnCol == NULL)
|
||
return FALSE;
|
||
|
||
// 消元
|
||
for (k=0; k<=m_nNumColumns-1; k++)
|
||
{
|
||
d=0.0;
|
||
for (i=k; i<=m_nNumColumns-1; i++)
|
||
{
|
||
for (j=k; j<=m_nNumColumns-1; j++)
|
||
{
|
||
u=i*m_nNumColumns+j;
|
||
p=m_pData[u]*m_pData[u]+mtxImag.m_pData[u]*mtxImag.m_pData[u];
|
||
if (p>d)
|
||
{
|
||
d=p;
|
||
pnRow[k]=i;
|
||
pnCol[k]=j;
|
||
}
|
||
}
|
||
}
|
||
|
||
// 失败
|
||
if (d == 0.0)
|
||
{
|
||
delete[] pnRow;
|
||
delete[] pnCol;
|
||
return(0);
|
||
}
|
||
|
||
if (pnRow[k]!=k)
|
||
{
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
{
|
||
u=k*m_nNumColumns+j;
|
||
v=pnRow[k]*m_nNumColumns+j;
|
||
t=m_pData[u];
|
||
m_pData[u]=m_pData[v];
|
||
m_pData[v]=t;
|
||
t=mtxImag.m_pData[u];
|
||
mtxImag.m_pData[u]=mtxImag.m_pData[v];
|
||
mtxImag.m_pData[v]=t;
|
||
}
|
||
}
|
||
|
||
if (pnCol[k]!=k)
|
||
{
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
u=i*m_nNumColumns+k;
|
||
v=i*m_nNumColumns+pnCol[k];
|
||
t=m_pData[u];
|
||
m_pData[u]=m_pData[v];
|
||
m_pData[v]=t;
|
||
t=mtxImag.m_pData[u];
|
||
mtxImag.m_pData[u]=mtxImag.m_pData[v];
|
||
mtxImag.m_pData[v]=t;
|
||
}
|
||
}
|
||
|
||
l=k*m_nNumColumns+k;
|
||
m_pData[l]=m_pData[l]/d; mtxImag.m_pData[l]=-mtxImag.m_pData[l]/d;
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
{
|
||
if (j!=k)
|
||
{
|
||
u=k*m_nNumColumns+j;
|
||
p=m_pData[u]*m_pData[l];
|
||
q=mtxImag.m_pData[u]*mtxImag.m_pData[l];
|
||
s=(m_pData[u]+mtxImag.m_pData[u])*(m_pData[l]+mtxImag.m_pData[l]);
|
||
m_pData[u]=p-q;
|
||
mtxImag.m_pData[u]=s-p-q;
|
||
}
|
||
}
|
||
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
if (i!=k)
|
||
{
|
||
v=i*m_nNumColumns+k;
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
{
|
||
if (j!=k)
|
||
{
|
||
u=k*m_nNumColumns+j;
|
||
w=i*m_nNumColumns+j;
|
||
p=m_pData[u]*m_pData[v];
|
||
q=mtxImag.m_pData[u]*mtxImag.m_pData[v];
|
||
s=(m_pData[u]+mtxImag.m_pData[u])*(m_pData[v]+mtxImag.m_pData[v]);
|
||
t=p-q;
|
||
b=s-p-q;
|
||
m_pData[w]=m_pData[w]-t;
|
||
mtxImag.m_pData[w]=mtxImag.m_pData[w]-b;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
if (i!=k)
|
||
{
|
||
u=i*m_nNumColumns+k;
|
||
p=m_pData[u]*m_pData[l];
|
||
q=mtxImag.m_pData[u]*mtxImag.m_pData[l];
|
||
s=(m_pData[u]+mtxImag.m_pData[u])*(m_pData[l]+mtxImag.m_pData[l]);
|
||
m_pData[u]=q-p;
|
||
mtxImag.m_pData[u]=p+q-s;
|
||
}
|
||
}
|
||
}
|
||
|
||
// 调整恢复行列次序
|
||
for (k=m_nNumColumns-1; k>=0; k--)
|
||
{
|
||
if (pnCol[k]!=k)
|
||
{
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
{
|
||
u=k*m_nNumColumns+j;
|
||
v=pnCol[k]*m_nNumColumns+j;
|
||
t=m_pData[u];
|
||
m_pData[u]=m_pData[v];
|
||
m_pData[v]=t;
|
||
t=mtxImag.m_pData[u];
|
||
mtxImag.m_pData[u]=mtxImag.m_pData[v];
|
||
mtxImag.m_pData[v]=t;
|
||
}
|
||
}
|
||
|
||
if (pnRow[k]!=k)
|
||
{
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
u=i*m_nNumColumns+k;
|
||
v=i*m_nNumColumns+pnRow[k];
|
||
t=m_pData[u];
|
||
m_pData[u]=m_pData[v];
|
||
m_pData[v]=t;
|
||
t=mtxImag.m_pData[u];
|
||
mtxImag.m_pData[u]=mtxImag.m_pData[v];
|
||
mtxImag.m_pData[v]=t;
|
||
}
|
||
}
|
||
}
|
||
|
||
// 清理内存
|
||
delete[] pnRow;
|
||
delete[] pnCol;
|
||
|
||
// 成功返回
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 对称正定矩阵的求逆
|
||
//
|
||
// 参数:无
|
||
//
|
||
// 返回值:BOOL型,求逆是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::InvertSsgj()
|
||
{
|
||
int i, j ,k, m;
|
||
float w, g, *pTmp;
|
||
|
||
// 临时内存
|
||
pTmp = new float[m_nNumColumns];
|
||
|
||
// 逐列处理
|
||
for (k=0; k<=m_nNumColumns-1; k++)
|
||
{
|
||
w=m_pData[0];
|
||
if (w == 0.0)
|
||
{
|
||
delete[] pTmp;
|
||
return FALSE;
|
||
}
|
||
|
||
m=m_nNumColumns-k-1;
|
||
for (i=1; i<=m_nNumColumns-1; i++)
|
||
{
|
||
g=m_pData[i*m_nNumColumns];
|
||
pTmp[i]=g/w;
|
||
if (i<=m)
|
||
pTmp[i]=-pTmp[i];
|
||
for (j=1; j<=i; j++)
|
||
m_pData[(i-1)*m_nNumColumns+j-1]=m_pData[i*m_nNumColumns+j]+g*pTmp[j];
|
||
}
|
||
|
||
m_pData[m_nNumColumns*m_nNumColumns-1]=1.0/w;
|
||
for (i=1; i<=m_nNumColumns-1; i++)
|
||
m_pData[(m_nNumColumns-1)*m_nNumColumns+i-1]=pTmp[i];
|
||
}
|
||
|
||
// 行列调整
|
||
for (i=0; i<=m_nNumColumns-2; i++)
|
||
for (j=i+1; j<=m_nNumColumns-1; j++)
|
||
m_pData[i*m_nNumColumns+j]=m_pData[j*m_nNumColumns+i];
|
||
|
||
// 临时内存清理
|
||
delete[] pTmp;
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 托伯利兹矩阵求逆的埃兰特方法
|
||
//
|
||
// 参数:无
|
||
//
|
||
// 返回值:BOOL型,求逆是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::InvertTrench()
|
||
{
|
||
int i,j,k;
|
||
float a,s,*t,*tt,*c,*r,*p;
|
||
|
||
// 上三角元素
|
||
t = new float[m_nNumColumns];
|
||
// 下三角元素
|
||
tt = new float[m_nNumColumns];
|
||
|
||
// 上、下三角元素赋值
|
||
for (i=0; i<m_nNumColumns; ++i)
|
||
{
|
||
t[i] = GetElement(0, i);
|
||
tt[i] = GetElement(i, 0);
|
||
}
|
||
|
||
// 临时缓冲区
|
||
c = new float[m_nNumColumns];
|
||
r = new float[m_nNumColumns];
|
||
p = new float[m_nNumColumns];
|
||
|
||
// 非Toeplitz矩阵,返回
|
||
if (t[0] == 0.0)
|
||
{
|
||
delete[] t;
|
||
delete[] tt;
|
||
delete[] c;
|
||
delete[] r;
|
||
delete[] p;
|
||
return FALSE;
|
||
}
|
||
|
||
a=t[0];
|
||
c[0]=tt[1]/t[0];
|
||
r[0]=t[1]/t[0];
|
||
|
||
for (k=0; k<=m_nNumColumns-3; k++)
|
||
{
|
||
s=0.0;
|
||
for (j=1; j<=k+1; j++)
|
||
s=s+c[k+1-j]*tt[j];
|
||
|
||
s=(s-tt[k+2])/a;
|
||
for (i=0; i<=k; i++)
|
||
p[i]=c[i]+s*r[k-i];
|
||
|
||
c[k+1]=-s;
|
||
s=0.0;
|
||
for (j=1; j<=k+1; j++)
|
||
s=s+r[k+1-j]*t[j];
|
||
|
||
s=(s-t[k+2])/a;
|
||
for (i=0; i<=k; i++)
|
||
{
|
||
r[i]=r[i]+s*c[k-i];
|
||
c[k-i]=p[k-i];
|
||
}
|
||
|
||
r[k+1]=-s;
|
||
a=0.0;
|
||
for (j=1; j<=k+2; j++)
|
||
a=a+t[j]*c[j-1];
|
||
|
||
a=t[0]-a;
|
||
|
||
// 求解失败
|
||
if (a == 0.0)
|
||
{
|
||
delete[] t;
|
||
delete[] tt;
|
||
delete[] c;
|
||
delete[] r;
|
||
delete[] p;
|
||
return FALSE;
|
||
}
|
||
}
|
||
|
||
m_pData[0]=1.0/a;
|
||
for (i=0; i<=m_nNumColumns-2; i++)
|
||
{
|
||
k=i+1;
|
||
j=(i+1)*m_nNumColumns;
|
||
m_pData[k]=-r[i]/a;
|
||
m_pData[j]=-c[i]/a;
|
||
}
|
||
|
||
for (i=0; i<=m_nNumColumns-2; i++)
|
||
{
|
||
for (j=0; j<=m_nNumColumns-2; j++)
|
||
{
|
||
k=(i+1)*m_nNumColumns+j+1;
|
||
m_pData[k]=m_pData[i*m_nNumColumns+j]-c[i]*m_pData[j+1];
|
||
m_pData[k]=m_pData[k]+c[m_nNumColumns-j-2]*m_pData[m_nNumColumns-i-1];
|
||
}
|
||
}
|
||
|
||
// 临时内存清理
|
||
delete[] t;
|
||
delete[] tt;
|
||
delete[] c;
|
||
delete[] r;
|
||
delete[] p;
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 求行列式值的全选主元高斯消去法
|
||
//
|
||
// 参数:无
|
||
//
|
||
// 返回值:float型,行列式的值
|
||
//////////////////////////////////////////////////////////////////////
|
||
float CMatrix1::DetGauss()
|
||
{
|
||
int i,j,k,is,js,l,u,v;
|
||
float f,det,q,d;
|
||
|
||
// 初值
|
||
f=1.0;
|
||
det=1.0;
|
||
|
||
// 消元
|
||
for (k=0; k<=m_nNumColumns-2; k++)
|
||
{
|
||
q=0.0;
|
||
for (i=k; i<=m_nNumColumns-1; i++)
|
||
{
|
||
for (j=k; j<=m_nNumColumns-1; j++)
|
||
{
|
||
l=i*m_nNumColumns+j;
|
||
d=fabs(m_pData[l]);
|
||
if (d>q)
|
||
{
|
||
q=d;
|
||
is=i;
|
||
js=j;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (q == 0.0)
|
||
{
|
||
det=0.0;
|
||
return(det);
|
||
}
|
||
|
||
if (is!=k)
|
||
{
|
||
f=-f;
|
||
for (j=k; j<=m_nNumColumns-1; j++)
|
||
{
|
||
u=k*m_nNumColumns+j;
|
||
v=is*m_nNumColumns+j;
|
||
d=m_pData[u];
|
||
m_pData[u]=m_pData[v];
|
||
m_pData[v]=d;
|
||
}
|
||
}
|
||
|
||
if (js!=k)
|
||
{
|
||
f=-f;
|
||
for (i=k; i<=m_nNumColumns-1; i++)
|
||
{
|
||
u=i*m_nNumColumns+js;
|
||
v=i*m_nNumColumns+k;
|
||
d=m_pData[u];
|
||
m_pData[u]=m_pData[v];
|
||
m_pData[v]=d;
|
||
}
|
||
}
|
||
|
||
l=k*m_nNumColumns+k;
|
||
det=det*m_pData[l];
|
||
for (i=k+1; i<=m_nNumColumns-1; i++)
|
||
{
|
||
d=m_pData[i*m_nNumColumns+k]/m_pData[l];
|
||
for (j=k+1; j<=m_nNumColumns-1; j++)
|
||
{
|
||
u=i*m_nNumColumns+j;
|
||
m_pData[u]=m_pData[u]-d*m_pData[k*m_nNumColumns+j];
|
||
}
|
||
}
|
||
}
|
||
|
||
// 求值
|
||
det=f*det*m_pData[m_nNumColumns*m_nNumColumns-1];
|
||
|
||
return(det);
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 求矩阵秩的全选主元高斯消去法
|
||
//
|
||
// 参数:无
|
||
//
|
||
// 返回值:int型,矩阵的秩
|
||
//////////////////////////////////////////////////////////////////////
|
||
int CMatrix1::RankGauss()
|
||
{
|
||
int i,j,k,nn,is,js,l,ll,u,v;
|
||
float q,d;
|
||
|
||
// 秩小于等于行列数
|
||
nn = m_nNumRows;
|
||
if (m_nNumRows >= m_nNumColumns)
|
||
nn = m_nNumColumns;
|
||
|
||
k=0;
|
||
|
||
// 消元求解
|
||
for (l=0; l<=nn-1; l++)
|
||
{
|
||
q=0.0;
|
||
for (i=l; i<=m_nNumRows-1; i++)
|
||
{
|
||
for (j=l; j<=m_nNumColumns-1; j++)
|
||
{
|
||
ll=i*m_nNumColumns+j;
|
||
d=fabs(m_pData[ll]);
|
||
if (d>q)
|
||
{
|
||
q=d;
|
||
is=i;
|
||
js=j;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (q == 0.0)
|
||
return(k);
|
||
|
||
k=k+1;
|
||
if (is!=l)
|
||
{
|
||
for (j=l; j<=m_nNumColumns-1; j++)
|
||
{
|
||
u=l*m_nNumColumns+j;
|
||
v=is*m_nNumColumns+j;
|
||
d=m_pData[u];
|
||
m_pData[u]=m_pData[v];
|
||
m_pData[v]=d;
|
||
}
|
||
}
|
||
if (js!=l)
|
||
{
|
||
for (i=l; i<=m_nNumRows-1; i++)
|
||
{
|
||
u=i*m_nNumColumns+js;
|
||
v=i*m_nNumColumns+l;
|
||
d=m_pData[u];
|
||
m_pData[u]=m_pData[v];
|
||
m_pData[v]=d;
|
||
}
|
||
}
|
||
|
||
ll=l*m_nNumColumns+l;
|
||
for (i=l+1; i<=m_nNumColumns-1; i++)
|
||
{
|
||
d=m_pData[i*m_nNumColumns+l]/m_pData[ll];
|
||
for (j=l+1; j<=m_nNumColumns-1; j++)
|
||
{
|
||
u=i*m_nNumColumns+j;
|
||
m_pData[u]=m_pData[u]-d*m_pData[l*m_nNumColumns+j];
|
||
}
|
||
}
|
||
}
|
||
|
||
return(k);
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 对称正定矩阵的乔里斯基分解与行列式的求值
|
||
//
|
||
// 参数:
|
||
// 1. float* dblDet - 返回行列式的值
|
||
//
|
||
// 返回值:BOOL型,求解是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::DetCholesky(float* dblDet)
|
||
{
|
||
int i,j,k,u,l;
|
||
float d;
|
||
|
||
// 不满足求解要求
|
||
if (m_pData[0] <= 0.0)
|
||
return FALSE;
|
||
|
||
// 乔里斯基分解
|
||
|
||
m_pData[0]=sqrt(m_pData[0]);
|
||
d=m_pData[0];
|
||
|
||
for (i=1; i<=m_nNumColumns-1; i++)
|
||
{
|
||
u=i*m_nNumColumns;
|
||
m_pData[u]=m_pData[u]/m_pData[0];
|
||
}
|
||
|
||
for (j=1; j<=m_nNumColumns-1; j++)
|
||
{
|
||
l=j*m_nNumColumns+j;
|
||
for (k=0; k<=j-1; k++)
|
||
{
|
||
u=j*m_nNumColumns+k;
|
||
m_pData[l]=m_pData[l]-m_pData[u]*m_pData[u];
|
||
}
|
||
|
||
if (m_pData[l] <= 0.0)
|
||
return FALSE;
|
||
|
||
m_pData[l]=sqrt(m_pData[l]);
|
||
d=d*m_pData[l];
|
||
|
||
for (i=j+1; i<=m_nNumColumns-1; i++)
|
||
{
|
||
u=i*m_nNumColumns+j;
|
||
for (k=0; k<=j-1; k++)
|
||
m_pData[u]=m_pData[u]-m_pData[i*m_nNumColumns+k]*m_pData[j*m_nNumColumns+k];
|
||
|
||
m_pData[u]=m_pData[u]/m_pData[l];
|
||
}
|
||
}
|
||
|
||
// 行列式求值
|
||
*dblDet=d*d;
|
||
|
||
// 下三角矩阵
|
||
for (i=0; i<=m_nNumColumns-2; i++)
|
||
for (j=i+1; j<=m_nNumColumns-1; j++)
|
||
m_pData[i*m_nNumColumns+j]=0.0;
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 矩阵的三角分解,分解成功后,原矩阵将成为Q矩阵
|
||
//
|
||
// 参数:
|
||
// 1. CMatrix1& mtxL - 返回分解后的L矩阵
|
||
// 2. CMatrix1& mtxU - 返回分解后的U矩阵
|
||
//
|
||
// 返回值:BOOL型,求解是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::SplitLU(CMatrix1& mtxL, CMatrix1& mtxU)
|
||
{
|
||
int i,j,k,w,v,ll;
|
||
|
||
// 初始化结果矩阵
|
||
if (! mtxL.Init(m_nNumColumns, m_nNumColumns) ||
|
||
! mtxU.Init(m_nNumColumns, m_nNumColumns))
|
||
return FALSE;
|
||
|
||
for (k=0; k<=m_nNumColumns-2; k++)
|
||
{
|
||
ll=k*m_nNumColumns+k;
|
||
if (m_pData[ll] == 0.0)
|
||
return FALSE;
|
||
|
||
for (i=k+1; i<=m_nNumColumns-1; i++)
|
||
{
|
||
w=i*m_nNumColumns+k;
|
||
m_pData[w]=m_pData[w]/m_pData[ll];
|
||
}
|
||
|
||
for (i=k+1; i<=m_nNumColumns-1; i++)
|
||
{
|
||
w=i*m_nNumColumns+k;
|
||
for (j=k+1; j<=m_nNumColumns-1; j++)
|
||
{
|
||
v=i*m_nNumColumns+j;
|
||
m_pData[v]=m_pData[v]-m_pData[w]*m_pData[k*m_nNumColumns+j];
|
||
}
|
||
}
|
||
}
|
||
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
for (j=0; j<i; j++)
|
||
{
|
||
w=i*m_nNumColumns+j;
|
||
mtxL.m_pData[w]=m_pData[w];
|
||
mtxU.m_pData[w]=0.0;
|
||
}
|
||
|
||
w=i*m_nNumColumns+i;
|
||
mtxL.m_pData[w]=1.0;
|
||
mtxU.m_pData[w]=m_pData[w];
|
||
|
||
for (j=i+1; j<=m_nNumColumns-1; j++)
|
||
{
|
||
w=i*m_nNumColumns+j;
|
||
mtxL.m_pData[w]=0.0;
|
||
mtxU.m_pData[w]=m_pData[w];
|
||
}
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 一般实矩阵的QR分解,分解成功后,原矩阵将成为R矩阵
|
||
//
|
||
// 参数:
|
||
// 1. CMatrix1& mtxQ - 返回分解后的Q矩阵
|
||
//
|
||
// 返回值:BOOL型,求解是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::SplitQR(CMatrix1& mtxQ)
|
||
{
|
||
int i,j,k,l,nn,p,jj;
|
||
float u,alpha,w,t;
|
||
|
||
if (m_nNumRows < m_nNumColumns)
|
||
return FALSE;
|
||
|
||
// 初始化Q矩阵
|
||
if (! mtxQ.Init(m_nNumRows, m_nNumRows))
|
||
return FALSE;
|
||
|
||
// 对角线元素单位化
|
||
for (i=0; i<=m_nNumRows-1; i++)
|
||
{
|
||
for (j=0; j<=m_nNumRows-1; j++)
|
||
{
|
||
l=i*m_nNumRows+j;
|
||
mtxQ.m_pData[l]=0.0;
|
||
if (i==j)
|
||
mtxQ.m_pData[l]=1.0;
|
||
}
|
||
}
|
||
|
||
// 开始分解
|
||
|
||
nn=m_nNumColumns;
|
||
if (m_nNumRows == m_nNumColumns)
|
||
nn=m_nNumRows-1;
|
||
|
||
for (k=0; k<=nn-1; k++)
|
||
{
|
||
u=0.0;
|
||
l=k*m_nNumColumns+k;
|
||
for (i=k; i<=m_nNumRows-1; i++)
|
||
{
|
||
w=fabs(m_pData[i*m_nNumColumns+k]);
|
||
if (w>u)
|
||
u=w;
|
||
}
|
||
|
||
alpha=0.0;
|
||
for (i=k; i<=m_nNumRows-1; i++)
|
||
{
|
||
t=m_pData[i*m_nNumColumns+k]/u;
|
||
alpha=alpha+t*t;
|
||
}
|
||
|
||
if (m_pData[l]>0.0)
|
||
u=-u;
|
||
|
||
alpha=u*sqrt(alpha);
|
||
if (alpha == 0.0)
|
||
return FALSE;
|
||
|
||
u=sqrt(2.0*alpha*(alpha-m_pData[l]));
|
||
if ((u+1.0)!=1.0)
|
||
{
|
||
m_pData[l]=(m_pData[l]-alpha)/u;
|
||
for (i=k+1; i<=m_nNumRows-1; i++)
|
||
{
|
||
p=i*m_nNumColumns+k;
|
||
m_pData[p]=m_pData[p]/u;
|
||
}
|
||
|
||
for (j=0; j<=m_nNumRows-1; j++)
|
||
{
|
||
t=0.0;
|
||
for (jj=k; jj<=m_nNumRows-1; jj++)
|
||
t=t+m_pData[jj*m_nNumColumns+k]*mtxQ.m_pData[jj*m_nNumRows+j];
|
||
|
||
for (i=k; i<=m_nNumRows-1; i++)
|
||
{
|
||
p=i*m_nNumRows+j;
|
||
mtxQ.m_pData[p]=mtxQ.m_pData[p]-2.0*t*m_pData[i*m_nNumColumns+k];
|
||
}
|
||
}
|
||
|
||
for (j=k+1; j<=m_nNumColumns-1; j++)
|
||
{
|
||
t=0.0;
|
||
|
||
for (jj=k; jj<=m_nNumRows-1; jj++)
|
||
t=t+m_pData[jj*m_nNumColumns+k]*m_pData[jj*m_nNumColumns+j];
|
||
|
||
for (i=k; i<=m_nNumRows-1; i++)
|
||
{
|
||
p=i*m_nNumColumns+j;
|
||
m_pData[p]=m_pData[p]-2.0*t*m_pData[i*m_nNumColumns+k];
|
||
}
|
||
}
|
||
|
||
m_pData[l]=alpha;
|
||
for (i=k+1; i<=m_nNumRows-1; i++)
|
||
m_pData[i*m_nNumColumns+k]=0.0;
|
||
}
|
||
}
|
||
|
||
// 调整元素
|
||
for (i=0; i<=m_nNumRows-2; i++)
|
||
{
|
||
for (j=i+1; j<=m_nNumRows-1;j++)
|
||
{
|
||
p=i*m_nNumRows+j;
|
||
l=j*m_nNumRows+i;
|
||
t=mtxQ.m_pData[p];
|
||
mtxQ.m_pData[p]=mtxQ.m_pData[l];
|
||
mtxQ.m_pData[l]=t;
|
||
}
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 一般实矩阵的奇异值分解,分解成功后,原矩阵对角线元素就是矩阵的奇异值
|
||
//
|
||
// 参数:
|
||
// 1. CMatrix1& mtxU - 返回分解后的U矩阵
|
||
// 2. CMatrix1& mtxV - 返回分解后的V矩阵
|
||
// 3. float eps - 计算精度,默认值为0.000001
|
||
//
|
||
// 返回值:BOOL型,求解是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::SplitUV(CMatrix1& mtxU, CMatrix1& mtxV, float eps /*= 0.000001*/)
|
||
{
|
||
int i,j,k,l,it,ll,kk,ix,iy,mm,nn,iz,m1,ks;
|
||
float d,dd,t,sm,sm1,em1,sk,ek,b,c,shh,fg[2],cs[2];
|
||
float *s,*e,*w;
|
||
|
||
int m = m_nNumRows;
|
||
int n = m_nNumColumns;
|
||
|
||
// 初始化U, V矩阵
|
||
if (! mtxU.Init(m, m) || ! mtxV.Init(n, n))
|
||
return FALSE;
|
||
|
||
// 临时缓冲区
|
||
int ka = max(m, n) + 1;
|
||
s = new float[ka];
|
||
e = new float[ka];
|
||
w = new float[ka];
|
||
|
||
// 指定迭代次数为60
|
||
it=60;
|
||
k=n;
|
||
|
||
if (m-1<n)
|
||
k=m-1;
|
||
|
||
l=m;
|
||
if (n-2<m)
|
||
l=n-2;
|
||
if (l<0)
|
||
l=0;
|
||
|
||
// 循环迭代计算
|
||
ll=k;
|
||
if (l>k)
|
||
ll=l;
|
||
if (ll>=1)
|
||
{
|
||
for (kk=1; kk<=ll; kk++)
|
||
{
|
||
if (kk<=k)
|
||
{
|
||
d=0.0;
|
||
for (i=kk; i<=m; i++)
|
||
{
|
||
ix=(i-1)*n+kk-1;
|
||
d=d+m_pData[ix]*m_pData[ix];
|
||
}
|
||
|
||
s[kk-1]=sqrt(d);
|
||
if (s[kk-1]!=0.0)
|
||
{
|
||
ix=(kk-1)*n+kk-1;
|
||
if (m_pData[ix]!=0.0)
|
||
{
|
||
s[kk-1]=fabs(s[kk-1]);
|
||
if (m_pData[ix]<0.0)
|
||
s[kk-1]=-s[kk-1];
|
||
}
|
||
|
||
for (i=kk; i<=m; i++)
|
||
{
|
||
iy=(i-1)*n+kk-1;
|
||
m_pData[iy]=m_pData[iy]/s[kk-1];
|
||
}
|
||
|
||
m_pData[ix]=1.0+m_pData[ix];
|
||
}
|
||
|
||
s[kk-1]=-s[kk-1];
|
||
}
|
||
|
||
if (n>=kk+1)
|
||
{
|
||
for (j=kk+1; j<=n; j++)
|
||
{
|
||
if ((kk<=k)&&(s[kk-1]!=0.0))
|
||
{
|
||
d=0.0;
|
||
for (i=kk; i<=m; i++)
|
||
{
|
||
ix=(i-1)*n+kk-1;
|
||
iy=(i-1)*n+j-1;
|
||
d=d+m_pData[ix]*m_pData[iy];
|
||
}
|
||
|
||
d=-d/m_pData[(kk-1)*n+kk-1];
|
||
for (i=kk; i<=m; i++)
|
||
{
|
||
ix=(i-1)*n+j-1;
|
||
iy=(i-1)*n+kk-1;
|
||
m_pData[ix]=m_pData[ix]+d*m_pData[iy];
|
||
}
|
||
}
|
||
|
||
e[j-1]=m_pData[(kk-1)*n+j-1];
|
||
}
|
||
}
|
||
|
||
if (kk<=k)
|
||
{
|
||
for (i=kk; i<=m; i++)
|
||
{
|
||
ix=(i-1)*m+kk-1;
|
||
iy=(i-1)*n+kk-1;
|
||
mtxU.m_pData[ix]=m_pData[iy];
|
||
}
|
||
}
|
||
|
||
if (kk<=l)
|
||
{
|
||
d=0.0;
|
||
for (i=kk+1; i<=n; i++)
|
||
d=d+e[i-1]*e[i-1];
|
||
|
||
e[kk-1]=sqrt(d);
|
||
if (e[kk-1]!=0.0)
|
||
{
|
||
if (e[kk]!=0.0)
|
||
{
|
||
e[kk-1]=fabs(e[kk-1]);
|
||
if (e[kk]<0.0)
|
||
e[kk-1]=-e[kk-1];
|
||
}
|
||
|
||
for (i=kk+1; i<=n; i++)
|
||
e[i-1]=e[i-1]/e[kk-1];
|
||
|
||
e[kk]=1.0+e[kk];
|
||
}
|
||
|
||
e[kk-1]=-e[kk-1];
|
||
if ((kk+1<=m)&&(e[kk-1]!=0.0))
|
||
{
|
||
for (i=kk+1; i<=m; i++)
|
||
w[i-1]=0.0;
|
||
|
||
for (j=kk+1; j<=n; j++)
|
||
for (i=kk+1; i<=m; i++)
|
||
w[i-1]=w[i-1]+e[j-1]*m_pData[(i-1)*n+j-1];
|
||
|
||
for (j=kk+1; j<=n; j++)
|
||
{
|
||
for (i=kk+1; i<=m; i++)
|
||
{
|
||
ix=(i-1)*n+j-1;
|
||
m_pData[ix]=m_pData[ix]-w[i-1]*e[j-1]/e[kk];
|
||
}
|
||
}
|
||
}
|
||
|
||
for (i=kk+1; i<=n; i++)
|
||
mtxV.m_pData[(i-1)*n+kk-1]=e[i-1];
|
||
}
|
||
}
|
||
}
|
||
|
||
mm=n;
|
||
if (m+1<n)
|
||
mm=m+1;
|
||
if (k<n)
|
||
s[k]=m_pData[k*n+k];
|
||
if (m<mm)
|
||
s[mm-1]=0.0;
|
||
if (l+1<mm)
|
||
e[l]=m_pData[l*n+mm-1];
|
||
|
||
e[mm-1]=0.0;
|
||
nn=m;
|
||
if (m>n)
|
||
nn=n;
|
||
if (nn>=k+1)
|
||
{
|
||
for (j=k+1; j<=nn; j++)
|
||
{
|
||
for (i=1; i<=m; i++)
|
||
mtxU.m_pData[(i-1)*m+j-1]=0.0;
|
||
mtxU.m_pData[(j-1)*m+j-1]=1.0;
|
||
}
|
||
}
|
||
|
||
if (k>=1)
|
||
{
|
||
for (ll=1; ll<=k; ll++)
|
||
{
|
||
kk=k-ll+1;
|
||
iz=(kk-1)*m+kk-1;
|
||
if (s[kk-1]!=0.0)
|
||
{
|
||
if (nn>=kk+1)
|
||
{
|
||
for (j=kk+1; j<=nn; j++)
|
||
{
|
||
d=0.0;
|
||
for (i=kk; i<=m; i++)
|
||
{
|
||
ix=(i-1)*m+kk-1;
|
||
iy=(i-1)*m+j-1;
|
||
d=d+mtxU.m_pData[ix]*mtxU.m_pData[iy]/mtxU.m_pData[iz];
|
||
}
|
||
|
||
d=-d;
|
||
for (i=kk; i<=m; i++)
|
||
{
|
||
ix=(i-1)*m+j-1;
|
||
iy=(i-1)*m+kk-1;
|
||
mtxU.m_pData[ix]=mtxU.m_pData[ix]+d*mtxU.m_pData[iy];
|
||
}
|
||
}
|
||
}
|
||
|
||
for (i=kk; i<=m; i++)
|
||
{
|
||
ix=(i-1)*m+kk-1;
|
||
mtxU.m_pData[ix]=-mtxU.m_pData[ix];
|
||
}
|
||
|
||
mtxU.m_pData[iz]=1.0+mtxU.m_pData[iz];
|
||
if (kk-1>=1)
|
||
{
|
||
for (i=1; i<=kk-1; i++)
|
||
mtxU.m_pData[(i-1)*m+kk-1]=0.0;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
for (i=1; i<=m; i++)
|
||
mtxU.m_pData[(i-1)*m+kk-1]=0.0;
|
||
mtxU.m_pData[(kk-1)*m+kk-1]=1.0;
|
||
}
|
||
}
|
||
}
|
||
|
||
for (ll=1; ll<=n; ll++)
|
||
{
|
||
kk=n-ll+1;
|
||
iz=kk*n+kk-1;
|
||
|
||
if ((kk<=l)&&(e[kk-1]!=0.0))
|
||
{
|
||
for (j=kk+1; j<=n; j++)
|
||
{
|
||
d=0.0;
|
||
for (i=kk+1; i<=n; i++)
|
||
{
|
||
ix=(i-1)*n+kk-1;
|
||
iy=(i-1)*n+j-1;
|
||
d=d+mtxV.m_pData[ix]*mtxV.m_pData[iy]/mtxV.m_pData[iz];
|
||
}
|
||
|
||
d=-d;
|
||
for (i=kk+1; i<=n; i++)
|
||
{
|
||
ix=(i-1)*n+j-1;
|
||
iy=(i-1)*n+kk-1;
|
||
mtxV.m_pData[ix]=mtxV.m_pData[ix]+d*mtxV.m_pData[iy];
|
||
}
|
||
}
|
||
}
|
||
|
||
for (i=1; i<=n; i++)
|
||
mtxV.m_pData[(i-1)*n+kk-1]=0.0;
|
||
|
||
mtxV.m_pData[iz-n]=1.0;
|
||
}
|
||
|
||
for (i=1; i<=m; i++)
|
||
for (j=1; j<=n; j++)
|
||
m_pData[(i-1)*n+j-1]=0.0;
|
||
|
||
m1=mm;
|
||
it=60;
|
||
while (TRUE)
|
||
{
|
||
if (mm==0)
|
||
{
|
||
ppp(m_pData,e,s,mtxV.m_pData,m,n);
|
||
return TRUE;
|
||
}
|
||
if (it==0)
|
||
{
|
||
ppp(m_pData,e,s,mtxV.m_pData,m,n);
|
||
return FALSE;
|
||
}
|
||
|
||
kk=mm-1;
|
||
while ((kk!=0)&&(fabs(e[kk-1])!=0.0))
|
||
{
|
||
d=fabs(s[kk-1])+fabs(s[kk]);
|
||
dd=fabs(e[kk-1]);
|
||
if (dd>eps*d)
|
||
kk=kk-1;
|
||
else
|
||
e[kk-1]=0.0;
|
||
}
|
||
|
||
if (kk==mm-1)
|
||
{
|
||
kk=kk+1;
|
||
if (s[kk-1]<0.0)
|
||
{
|
||
s[kk-1]=-s[kk-1];
|
||
for (i=1; i<=n; i++)
|
||
{
|
||
ix=(i-1)*n+kk-1;
|
||
mtxV.m_pData[ix]=-mtxV.m_pData[ix];}
|
||
}
|
||
|
||
while ((kk!=m1)&&(s[kk-1]<s[kk]))
|
||
{
|
||
d=s[kk-1];
|
||
s[kk-1]=s[kk];
|
||
s[kk]=d;
|
||
if (kk<n)
|
||
{
|
||
for (i=1; i<=n; i++)
|
||
{
|
||
ix=(i-1)*n+kk-1;
|
||
iy=(i-1)*n+kk;
|
||
d=mtxV.m_pData[ix];
|
||
mtxV.m_pData[ix]=mtxV.m_pData[iy];
|
||
mtxV.m_pData[iy]=d;
|
||
}
|
||
}
|
||
|
||
if (kk<m)
|
||
{
|
||
for (i=1; i<=m; i++)
|
||
{
|
||
ix=(i-1)*m+kk-1;
|
||
iy=(i-1)*m+kk;
|
||
d=mtxU.m_pData[ix];
|
||
mtxU.m_pData[ix]=mtxU.m_pData[iy];
|
||
mtxU.m_pData[iy]=d;
|
||
}
|
||
}
|
||
|
||
kk=kk+1;
|
||
}
|
||
|
||
it=60;
|
||
mm=mm-1;
|
||
}
|
||
else
|
||
{
|
||
ks=mm;
|
||
while ((ks>kk)&&(fabs(s[ks-1])!=0.0))
|
||
{
|
||
d=0.0;
|
||
if (ks!=mm)
|
||
d=d+fabs(e[ks-1]);
|
||
if (ks!=kk+1)
|
||
d=d+fabs(e[ks-2]);
|
||
|
||
dd=fabs(s[ks-1]);
|
||
if (dd>eps*d)
|
||
ks=ks-1;
|
||
else
|
||
s[ks-1]=0.0;
|
||
}
|
||
|
||
if (ks==kk)
|
||
{
|
||
kk=kk+1;
|
||
d=fabs(s[mm-1]);
|
||
t=fabs(s[mm-2]);
|
||
if (t>d)
|
||
d=t;
|
||
|
||
t=fabs(e[mm-2]);
|
||
if (t>d)
|
||
d=t;
|
||
|
||
t=fabs(s[kk-1]);
|
||
if (t>d)
|
||
d=t;
|
||
|
||
t=fabs(e[kk-1]);
|
||
if (t>d)
|
||
d=t;
|
||
|
||
sm=s[mm-1]/d;
|
||
sm1=s[mm-2]/d;
|
||
em1=e[mm-2]/d;
|
||
sk=s[kk-1]/d;
|
||
ek=e[kk-1]/d;
|
||
b=((sm1+sm)*(sm1-sm)+em1*em1)/2.0;
|
||
c=sm*em1;
|
||
c=c*c;
|
||
shh=0.0;
|
||
|
||
if ((b!=0.0)||(c!=0.0))
|
||
{
|
||
shh=sqrt(b*b+c);
|
||
if (b<0.0)
|
||
shh=-shh;
|
||
|
||
shh=c/(b+shh);
|
||
}
|
||
|
||
fg[0]=(sk+sm)*(sk-sm)-shh;
|
||
fg[1]=sk*ek;
|
||
for (i=kk; i<=mm-1; i++)
|
||
{
|
||
sss(fg,cs);
|
||
if (i!=kk)
|
||
e[i-2]=fg[0];
|
||
|
||
fg[0]=cs[0]*s[i-1]+cs[1]*e[i-1];
|
||
e[i-1]=cs[0]*e[i-1]-cs[1]*s[i-1];
|
||
fg[1]=cs[1]*s[i];
|
||
s[i]=cs[0]*s[i];
|
||
|
||
if ((cs[0]!=1.0)||(cs[1]!=0.0))
|
||
{
|
||
for (j=1; j<=n; j++)
|
||
{
|
||
ix=(j-1)*n+i-1;
|
||
iy=(j-1)*n+i;
|
||
d=cs[0]*mtxV.m_pData[ix]+cs[1]*mtxV.m_pData[iy];
|
||
mtxV.m_pData[iy]=-cs[1]*mtxV.m_pData[ix]+cs[0]*mtxV.m_pData[iy];
|
||
mtxV.m_pData[ix]=d;
|
||
}
|
||
}
|
||
|
||
sss(fg,cs);
|
||
s[i-1]=fg[0];
|
||
fg[0]=cs[0]*e[i-1]+cs[1]*s[i];
|
||
s[i]=-cs[1]*e[i-1]+cs[0]*s[i];
|
||
fg[1]=cs[1]*e[i];
|
||
e[i]=cs[0]*e[i];
|
||
|
||
if (i<m)
|
||
{
|
||
if ((cs[0]!=1.0)||(cs[1]!=0.0))
|
||
{
|
||
for (j=1; j<=m; j++)
|
||
{
|
||
ix=(j-1)*m+i-1;
|
||
iy=(j-1)*m+i;
|
||
d=cs[0]*mtxU.m_pData[ix]+cs[1]*mtxU.m_pData[iy];
|
||
mtxU.m_pData[iy]=-cs[1]*mtxU.m_pData[ix]+cs[0]*mtxU.m_pData[iy];
|
||
mtxU.m_pData[ix]=d;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
e[mm-2]=fg[0];
|
||
it=it-1;
|
||
}
|
||
else
|
||
{
|
||
if (ks==mm)
|
||
{
|
||
kk=kk+1;
|
||
fg[1]=e[mm-2];
|
||
e[mm-2]=0.0;
|
||
for (ll=kk; ll<=mm-1; ll++)
|
||
{
|
||
i=mm+kk-ll-1;
|
||
fg[0]=s[i-1];
|
||
sss(fg,cs);
|
||
s[i-1]=fg[0];
|
||
if (i!=kk)
|
||
{
|
||
fg[1]=-cs[1]*e[i-2];
|
||
e[i-2]=cs[0]*e[i-2];
|
||
}
|
||
|
||
if ((cs[0]!=1.0)||(cs[1]!=0.0))
|
||
{
|
||
for (j=1; j<=n; j++)
|
||
{
|
||
ix=(j-1)*n+i-1;
|
||
iy=(j-1)*n+mm-1;
|
||
d=cs[0]*mtxV.m_pData[ix]+cs[1]*mtxV.m_pData[iy];
|
||
mtxV.m_pData[iy]=-cs[1]*mtxV.m_pData[ix]+cs[0]*mtxV.m_pData[iy];
|
||
mtxV.m_pData[ix]=d;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
kk=ks+1;
|
||
fg[1]=e[kk-2];
|
||
e[kk-2]=0.0;
|
||
for (i=kk; i<=mm; i++)
|
||
{
|
||
fg[0]=s[i-1];
|
||
sss(fg,cs);
|
||
s[i-1]=fg[0];
|
||
fg[1]=-cs[1]*e[i-1];
|
||
e[i-1]=cs[0]*e[i-1];
|
||
if ((cs[0]!=1.0)||(cs[1]!=0.0))
|
||
{
|
||
for (j=1; j<=m; j++)
|
||
{
|
||
ix=(j-1)*m+i-1;
|
||
iy=(j-1)*m+kk-2;
|
||
d=cs[0]*mtxU.m_pData[ix]+cs[1]*mtxU.m_pData[iy];
|
||
mtxU.m_pData[iy]=-cs[1]*mtxU.m_pData[ix]+cs[0]*mtxU.m_pData[iy];
|
||
mtxU.m_pData[ix]=d;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 内部函数,由SplitUV函数调用
|
||
//////////////////////////////////////////////////////////////////////
|
||
void CMatrix1::ppp(float a[], float e[], float s[], float v[], int m, int n)
|
||
{
|
||
int i,j,p,q;
|
||
float d;
|
||
|
||
if (m>=n)
|
||
i=n;
|
||
else
|
||
i=m;
|
||
|
||
for (j=1; j<=i-1; j++)
|
||
{
|
||
a[(j-1)*n+j-1]=s[j-1];
|
||
a[(j-1)*n+j]=e[j-1];
|
||
}
|
||
|
||
a[(i-1)*n+i-1]=s[i-1];
|
||
if (m<n)
|
||
a[(i-1)*n+i]=e[i-1];
|
||
|
||
for (i=1; i<=n-1; i++)
|
||
{
|
||
for (j=i+1; j<=n; j++)
|
||
{
|
||
p=(i-1)*n+j-1;
|
||
q=(j-1)*n+i-1;
|
||
d=v[p];
|
||
v[p]=v[q];
|
||
v[q]=d;
|
||
}
|
||
}
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 内部函数,由SplitUV函数调用
|
||
//////////////////////////////////////////////////////////////////////
|
||
void CMatrix1::sss(float fg[2], float cs[2])
|
||
{
|
||
float r,d;
|
||
|
||
if ((fabs(fg[0])+fabs(fg[1]))==0.0)
|
||
{
|
||
cs[0]=1.0;
|
||
cs[1]=0.0;
|
||
d=0.0;
|
||
}
|
||
else
|
||
{
|
||
d=sqrt(fg[0]*fg[0]+fg[1]*fg[1]);
|
||
if (fabs(fg[0])>fabs(fg[1]))
|
||
{
|
||
d=fabs(d);
|
||
if (fg[0]<0.0)
|
||
d=-d;
|
||
}
|
||
if (fabs(fg[1])>=fabs(fg[0]))
|
||
{
|
||
d=fabs(d);
|
||
if (fg[1]<0.0)
|
||
d=-d;
|
||
}
|
||
|
||
cs[0]=fg[0]/d;
|
||
cs[1]=fg[1]/d;
|
||
}
|
||
|
||
r=1.0;
|
||
if (fabs(fg[0])>fabs(fg[1]))
|
||
r=cs[1];
|
||
else if (cs[0]!=0.0)
|
||
r=1.0/cs[0];
|
||
|
||
fg[0]=d;
|
||
fg[1]=r;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 求广义逆的奇异值分解法,分解成功后,原矩阵对角线元素就是矩阵的奇异值
|
||
//
|
||
// 参数:
|
||
// 1. CMatrix1& mtxAP - 返回原矩阵的广义逆矩阵
|
||
// 2. CMatrix1& mtxU - 返回分解后的U矩阵
|
||
// 3. CMatrix1& mtxV - 返回分解后的V矩阵
|
||
// 4. float eps - 计算精度,默认值为0.000001
|
||
//
|
||
// 返回值:BOOL型,求解是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::GInvertUV(CMatrix1& mtxAP, CMatrix1& mtxU, CMatrix1& mtxV, float eps /*= 0.000001*/)
|
||
{
|
||
int i,j,k,l,t,p,q,f;
|
||
|
||
// 调用奇异值分解
|
||
if (! SplitUV(mtxU, mtxV, eps))
|
||
return FALSE;
|
||
|
||
int m = m_nNumRows;
|
||
int n = m_nNumColumns;
|
||
|
||
// 初始化广义逆矩阵
|
||
if (! mtxAP.Init(n, m))
|
||
return FALSE;
|
||
|
||
// 计算广义逆矩阵
|
||
|
||
j=n;
|
||
if (m<n)
|
||
j=m;
|
||
j=j-1;
|
||
k=0;
|
||
while ((k<=j)&&(m_pData[k*n+k]!=0.0))
|
||
k=k+1;
|
||
|
||
k=k-1;
|
||
for (i=0; i<=n-1; i++)
|
||
{
|
||
for (j=0; j<=m-1; j++)
|
||
{
|
||
t=i*m+j;
|
||
mtxAP.m_pData[t]=0.0;
|
||
for (l=0; l<=k; l++)
|
||
{
|
||
f=l*n+i;
|
||
p=j*m+l;
|
||
q=l*n+l;
|
||
mtxAP.m_pData[t]=mtxAP.m_pData[t]+mtxV.m_pData[f]*mtxU.m_pData[p]/m_pData[q];
|
||
}
|
||
}
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 约化对称矩阵为对称三对角阵的豪斯荷尔德变换法
|
||
//
|
||
// 参数:
|
||
// 1. CMatrix1& mtxQ - 返回豪斯荷尔德变换的乘积矩阵Q
|
||
// 2. CMatrix1& mtxT - 返回求得的对称三对角阵
|
||
// 3. float dblB[] - 一维数组,长度为矩阵的阶数,返回对称三对角阵的主对角线元素
|
||
// 4. float dblC[] - 一维数组,长度为矩阵的阶数,前n-1个元素返回对称三对角阵的次对角线元素
|
||
//
|
||
// 返回值:BOOL型,求解是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::MakeSymTri(CMatrix1& mtxQ, CMatrix1& mtxT, float dblB[], float dblC[])
|
||
{
|
||
int i,j,k,u;
|
||
float h,f,g,h2;
|
||
|
||
// 初始化矩阵Q和T
|
||
if (! mtxQ.Init(m_nNumColumns, m_nNumColumns) ||
|
||
! mtxT.Init(m_nNumColumns, m_nNumColumns))
|
||
return FALSE;
|
||
|
||
if (dblB == NULL || dblC == NULL)
|
||
return FALSE;
|
||
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
{
|
||
u=i*m_nNumColumns+j;
|
||
mtxQ.m_pData[u]=m_pData[u];
|
||
}
|
||
}
|
||
|
||
for (i=m_nNumColumns-1; i>=1; i--)
|
||
{
|
||
h=0.0;
|
||
if (i>1)
|
||
{
|
||
for (k=0; k<=i-1; k++)
|
||
{
|
||
u=i*m_nNumColumns+k;
|
||
h=h+mtxQ.m_pData[u]*mtxQ.m_pData[u];
|
||
}
|
||
}
|
||
|
||
if (h == 0.0)
|
||
{
|
||
dblC[i]=0.0;
|
||
if (i==1)
|
||
dblC[i]=mtxQ.m_pData[i*m_nNumColumns+i-1];
|
||
dblB[i]=0.0;
|
||
}
|
||
else
|
||
{
|
||
dblC[i]=sqrt(h);
|
||
u=i*m_nNumColumns+i-1;
|
||
if (mtxQ.m_pData[u]>0.0)
|
||
dblC[i]=-dblC[i];
|
||
|
||
h=h-mtxQ.m_pData[u]*dblC[i];
|
||
mtxQ.m_pData[u]=mtxQ.m_pData[u]-dblC[i];
|
||
f=0.0;
|
||
for (j=0; j<=i-1; j++)
|
||
{
|
||
mtxQ.m_pData[j*m_nNumColumns+i]=mtxQ.m_pData[i*m_nNumColumns+j]/h;
|
||
g=0.0;
|
||
for (k=0; k<=j; k++)
|
||
g=g+mtxQ.m_pData[j*m_nNumColumns+k]*mtxQ.m_pData[i*m_nNumColumns+k];
|
||
|
||
if (j+1<=i-1)
|
||
for (k=j+1; k<=i-1; k++)
|
||
g=g+mtxQ.m_pData[k*m_nNumColumns+j]*mtxQ.m_pData[i*m_nNumColumns+k];
|
||
|
||
dblC[j]=g/h;
|
||
f=f+g*mtxQ.m_pData[j*m_nNumColumns+i];
|
||
}
|
||
|
||
h2=f/(h+h);
|
||
for (j=0; j<=i-1; j++)
|
||
{
|
||
f=mtxQ.m_pData[i*m_nNumColumns+j];
|
||
g=dblC[j]-h2*f;
|
||
dblC[j]=g;
|
||
for (k=0; k<=j; k++)
|
||
{
|
||
u=j*m_nNumColumns+k;
|
||
mtxQ.m_pData[u]=mtxQ.m_pData[u]-f*dblC[k]-g*mtxQ.m_pData[i*m_nNumColumns+k];
|
||
}
|
||
}
|
||
|
||
dblB[i]=h;
|
||
}
|
||
}
|
||
|
||
for (i=0; i<=m_nNumColumns-2; i++)
|
||
dblC[i]=dblC[i+1];
|
||
|
||
dblC[m_nNumColumns-1]=0.0;
|
||
dblB[0]=0.0;
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
if ((dblB[i]!=0.0)&&(i-1>=0))
|
||
{
|
||
for (j=0; j<=i-1; j++)
|
||
{
|
||
g=0.0;
|
||
for (k=0; k<=i-1; k++)
|
||
g=g+mtxQ.m_pData[i*m_nNumColumns+k]*mtxQ.m_pData[k*m_nNumColumns+j];
|
||
|
||
for (k=0; k<=i-1; k++)
|
||
{
|
||
u=k*m_nNumColumns+j;
|
||
mtxQ.m_pData[u]=mtxQ.m_pData[u]-g*mtxQ.m_pData[k*m_nNumColumns+i];
|
||
}
|
||
}
|
||
}
|
||
|
||
u=i*m_nNumColumns+i;
|
||
dblB[i]=mtxQ.m_pData[u]; mtxQ.m_pData[u]=1.0;
|
||
if (i-1>=0)
|
||
{
|
||
for (j=0; j<=i-1; j++)
|
||
{
|
||
mtxQ.m_pData[i*m_nNumColumns+j]=0.0;
|
||
mtxQ.m_pData[j*m_nNumColumns+i]=0.0;
|
||
}
|
||
}
|
||
}
|
||
|
||
// 构造对称三对角矩阵
|
||
for (i=0; i<m_nNumColumns; ++i)
|
||
{
|
||
for (j=0; j<m_nNumColumns; ++j)
|
||
{
|
||
mtxT.SetElement(i, j, 0);
|
||
k = i - j;
|
||
if (k == 0)
|
||
mtxT.SetElement(i, j, dblB[j]);
|
||
else if (k == 1)
|
||
mtxT.SetElement(i, j, dblC[j]);
|
||
else if (k == -1)
|
||
mtxT.SetElement(i, j, dblC[i]);
|
||
}
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 实对称三对角阵的全部特征值与特征向量的计算
|
||
//
|
||
// 参数:
|
||
// 1. float dblB[] - 一维数组,长度为矩阵的阶数,传入对称三对角阵的主对角线元素;
|
||
// 返回时存放全部特征值。
|
||
// 2. float dblC[] - 一维数组,长度为矩阵的阶数,前n-1个元素传入对称三对角阵的次对角线元素
|
||
// 3. CMatrix1& mtxQ - 如果传入单位矩阵,则返回实对称三对角阵的特征值向量矩阵;
|
||
// 如果传入MakeSymTri函数求得的矩阵A的豪斯荷尔德变换的乘积矩阵Q,则返回矩阵A的
|
||
// 特征值向量矩阵。其中第i列为与数组dblB中第j个特征值对应的特征向量。
|
||
// 4. int nMaxIt - 迭代次数,默认值为60
|
||
// 5. float eps - 计算精度,默认值为0.000001
|
||
//
|
||
// 返回值:BOOL型,求解是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::SymTriEigenv(float dblB[], float dblC[], CMatrix1& mtxQ, int nMaxIt /*= 60*/, float eps /*= 0.000001*/)
|
||
{
|
||
int i,j,k,m,it,u,v;
|
||
float d,f,h,g,p,r,e,s;
|
||
|
||
// 初值
|
||
int n = mtxQ.GetNumColumns();
|
||
dblC[n-1]=0.0;
|
||
d=0.0;
|
||
f=0.0;
|
||
|
||
// 迭代计算
|
||
|
||
for (j=0; j<=n-1; j++)
|
||
{
|
||
it=0;
|
||
h=eps*(fabs(dblB[j])+fabs(dblC[j]));
|
||
if (h>d)
|
||
d=h;
|
||
|
||
m=j;
|
||
while ((m<=n-1)&&(fabs(dblC[m])>d))
|
||
m=m+1;
|
||
|
||
if (m!=j)
|
||
{
|
||
do
|
||
{
|
||
if (it==nMaxIt)
|
||
return FALSE;
|
||
|
||
it=it+1;
|
||
g=dblB[j];
|
||
p=(dblB[j+1]-g)/(2.0*dblC[j]);
|
||
r=sqrt(p*p+1.0);
|
||
if (p>=0.0)
|
||
dblB[j]=dblC[j]/(p+r);
|
||
else
|
||
dblB[j]=dblC[j]/(p-r);
|
||
|
||
h=g-dblB[j];
|
||
for (i=j+1; i<=n-1; i++)
|
||
dblB[i]=dblB[i]-h;
|
||
|
||
f=f+h;
|
||
p=dblB[m];
|
||
e=1.0;
|
||
s=0.0;
|
||
for (i=m-1; i>=j; i--)
|
||
{
|
||
g=e*dblC[i];
|
||
h=e*p;
|
||
if (fabs(p)>=fabs(dblC[i]))
|
||
{
|
||
e=dblC[i]/p;
|
||
r=sqrt(e*e+1.0);
|
||
dblC[i+1]=s*p*r;
|
||
s=e/r;
|
||
e=1.0/r;
|
||
}
|
||
else
|
||
{
|
||
e=p/dblC[i];
|
||
r=sqrt(e*e+1.0);
|
||
dblC[i+1]=s*dblC[i]*r;
|
||
s=1.0/r;
|
||
e=e/r;
|
||
}
|
||
|
||
p=e*dblB[i]-s*g;
|
||
dblB[i+1]=h+s*(e*g+s*dblB[i]);
|
||
for (k=0; k<=n-1; k++)
|
||
{
|
||
u=k*n+i+1;
|
||
v=u-1;
|
||
h=mtxQ.m_pData[u];
|
||
mtxQ.m_pData[u]=s*mtxQ.m_pData[v]+e*h;
|
||
mtxQ.m_pData[v]=e*mtxQ.m_pData[v]-s*h;
|
||
}
|
||
}
|
||
|
||
dblC[j]=s*p;
|
||
dblB[j]=e*p;
|
||
|
||
} while (fabs(dblC[j])>d);
|
||
}
|
||
|
||
dblB[j]=dblB[j]+f;
|
||
}
|
||
|
||
for (i=0; i<=n-1; i++)
|
||
{
|
||
k=i;
|
||
p=dblB[i];
|
||
if (i+1<=n-1)
|
||
{
|
||
j=i+1;
|
||
while ((j<=n-1)&&(dblB[j]<=p))
|
||
{
|
||
k=j;
|
||
p=dblB[j];
|
||
j=j+1;
|
||
}
|
||
}
|
||
|
||
if (k!=i)
|
||
{
|
||
dblB[k]=dblB[i];
|
||
dblB[i]=p;
|
||
for (j=0; j<=n-1; j++)
|
||
{
|
||
u=j*n+i;
|
||
v=j*n+k;
|
||
p=mtxQ.m_pData[u];
|
||
mtxQ.m_pData[u]=mtxQ.m_pData[v];
|
||
mtxQ.m_pData[v]=p;
|
||
}
|
||
}
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 约化一般实矩阵为赫申伯格矩阵的初等相似变换法
|
||
//
|
||
// 参数:无
|
||
//
|
||
// 返回值:无
|
||
//////////////////////////////////////////////////////////////////////
|
||
void CMatrix1::MakeHberg()
|
||
{
|
||
int i,j,k,u,v;
|
||
float d,t;
|
||
|
||
for (k=1; k<=m_nNumColumns-2; k++)
|
||
{
|
||
d=0.0;
|
||
for (j=k; j<=m_nNumColumns-1; j++)
|
||
{
|
||
u=j*m_nNumColumns+k-1;
|
||
t=m_pData[u];
|
||
if (fabs(t)>fabs(d))
|
||
{
|
||
d=t;
|
||
i=j;
|
||
}
|
||
}
|
||
|
||
if (d != 0.0)
|
||
{
|
||
if (i!=k)
|
||
{
|
||
for (j=k-1; j<=m_nNumColumns-1; j++)
|
||
{
|
||
u=i*m_nNumColumns+j;
|
||
v=k*m_nNumColumns+j;
|
||
t=m_pData[u];
|
||
m_pData[u]=m_pData[v];
|
||
m_pData[v]=t;
|
||
}
|
||
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
{
|
||
u=j*m_nNumColumns+i;
|
||
v=j*m_nNumColumns+k;
|
||
t=m_pData[u];
|
||
m_pData[u]=m_pData[v];
|
||
m_pData[v]=t;
|
||
}
|
||
}
|
||
|
||
for (i=k+1; i<=m_nNumColumns-1; i++)
|
||
{
|
||
u=i*m_nNumColumns+k-1;
|
||
t=m_pData[u]/d;
|
||
m_pData[u]=0.0;
|
||
for (j=k; j<=m_nNumColumns-1; j++)
|
||
{
|
||
v=i*m_nNumColumns+j;
|
||
m_pData[v]=m_pData[v]-t*m_pData[k*m_nNumColumns+j];
|
||
}
|
||
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
{
|
||
v=j*m_nNumColumns+k;
|
||
m_pData[v]=m_pData[v]+t*m_pData[j*m_nNumColumns+i];
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 求赫申伯格矩阵全部特征值的QR方法
|
||
//
|
||
// 参数:
|
||
// 1. float dblU[] - 一维数组,长度为矩阵的阶数,返回时存放特征值的实部
|
||
// 2. float dblV[] - 一维数组,长度为矩阵的阶数,返回时存放特征值的虚部
|
||
// 3. int nMaxIt - 迭代次数,默认值为60
|
||
// 4. float eps - 计算精度,默认值为0.000001
|
||
//
|
||
// 返回值:BOOL型,求解是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::HBergEigenv(float dblU[], float dblV[], int nMaxIt /*= 60*/, float eps /*= 0.000001*/)
|
||
{
|
||
int m,it,i,j,k,l,ii,jj,kk,ll;
|
||
float b,c,w,g,xy,p,q,r,x,s,e,f,z,y;
|
||
|
||
int n = m_nNumColumns;
|
||
|
||
it=0;
|
||
m=n;
|
||
while (m!=0)
|
||
{
|
||
l=m-1;
|
||
while ((l>0)&&(fabs(m_pData[l*n+l-1]) >
|
||
eps*(fabs(m_pData[(l-1)*n+l-1])+fabs(m_pData[l*n+l]))))
|
||
l=l-1;
|
||
|
||
ii=(m-1)*n+m-1;
|
||
jj=(m-1)*n+m-2;
|
||
kk=(m-2)*n+m-1;
|
||
ll=(m-2)*n+m-2;
|
||
if (l==m-1)
|
||
{
|
||
dblU[m-1]=m_pData[(m-1)*n+m-1];
|
||
dblV[m-1]=0.0;
|
||
m=m-1;
|
||
it=0;
|
||
}
|
||
else if (l==m-2)
|
||
{
|
||
b=-(m_pData[ii]+m_pData[ll]);
|
||
c=m_pData[ii]*m_pData[ll]-m_pData[jj]*m_pData[kk];
|
||
w=b*b-4.0*c;
|
||
y=sqrt(fabs(w));
|
||
if (w>0.0)
|
||
{
|
||
xy=1.0;
|
||
if (b<0.0)
|
||
xy=-1.0;
|
||
dblU[m-1]=(-b-xy*y)/2.0;
|
||
dblU[m-2]=c/dblU[m-1];
|
||
dblV[m-1]=0.0; dblV[m-2]=0.0;
|
||
}
|
||
else
|
||
{
|
||
dblU[m-1]=-b/2.0;
|
||
dblU[m-2]=dblU[m-1];
|
||
dblV[m-1]=y/2.0;
|
||
dblV[m-2]=-dblV[m-1];
|
||
}
|
||
|
||
m=m-2;
|
||
it=0;
|
||
}
|
||
else
|
||
{
|
||
if (it>=nMaxIt)
|
||
return FALSE;
|
||
|
||
it=it+1;
|
||
for (j=l+2; j<=m-1; j++)
|
||
m_pData[j*n+j-2]=0.0;
|
||
for (j=l+3; j<=m-1; j++)
|
||
m_pData[j*n+j-3]=0.0;
|
||
for (k=l; k<=m-2; k++)
|
||
{
|
||
if (k!=l)
|
||
{
|
||
p=m_pData[k*n+k-1];
|
||
q=m_pData[(k+1)*n+k-1];
|
||
r=0.0;
|
||
if (k!=m-2)
|
||
r=m_pData[(k+2)*n+k-1];
|
||
}
|
||
else
|
||
{
|
||
x=m_pData[ii]+m_pData[ll];
|
||
y=m_pData[ll]*m_pData[ii]-m_pData[kk]*m_pData[jj];
|
||
ii=l*n+l;
|
||
jj=l*n+l+1;
|
||
kk=(l+1)*n+l;
|
||
ll=(l+1)*n+l+1;
|
||
p=m_pData[ii]*(m_pData[ii]-x)+m_pData[jj]*m_pData[kk]+y;
|
||
q=m_pData[kk]*(m_pData[ii]+m_pData[ll]-x);
|
||
r=m_pData[kk]*m_pData[(l+2)*n+l+1];
|
||
}
|
||
|
||
if ((fabs(p)+fabs(q)+fabs(r))!=0.0)
|
||
{
|
||
xy=1.0;
|
||
if (p<0.0)
|
||
xy=-1.0;
|
||
s=xy*sqrt(p*p+q*q+r*r);
|
||
if (k!=l)
|
||
m_pData[k*n+k-1]=-s;
|
||
e=-q/s;
|
||
f=-r/s;
|
||
x=-p/s;
|
||
y=-x-f*r/(p+s);
|
||
g=e*r/(p+s);
|
||
z=-x-e*q/(p+s);
|
||
for (j=k; j<=m-1; j++)
|
||
{
|
||
ii=k*n+j;
|
||
jj=(k+1)*n+j;
|
||
p=x*m_pData[ii]+e*m_pData[jj];
|
||
q=e*m_pData[ii]+y*m_pData[jj];
|
||
r=f*m_pData[ii]+g*m_pData[jj];
|
||
if (k!=m-2)
|
||
{
|
||
kk=(k+2)*n+j;
|
||
p=p+f*m_pData[kk];
|
||
q=q+g*m_pData[kk];
|
||
r=r+z*m_pData[kk];
|
||
m_pData[kk]=r;
|
||
}
|
||
|
||
m_pData[jj]=q; m_pData[ii]=p;
|
||
}
|
||
|
||
j=k+3;
|
||
if (j>=m-1)
|
||
j=m-1;
|
||
|
||
for (i=l; i<=j; i++)
|
||
{
|
||
ii=i*n+k;
|
||
jj=i*n+k+1;
|
||
p=x*m_pData[ii]+e*m_pData[jj];
|
||
q=e*m_pData[ii]+y*m_pData[jj];
|
||
r=f*m_pData[ii]+g*m_pData[jj];
|
||
if (k!=m-2)
|
||
{
|
||
kk=i*n+k+2;
|
||
p=p+f*m_pData[kk];
|
||
q=q+g*m_pData[kk];
|
||
r=r+z*m_pData[kk];
|
||
m_pData[kk]=r;
|
||
}
|
||
|
||
m_pData[jj]=q;
|
||
m_pData[ii]=p;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 求实对称矩阵特征值与特征向量的雅可比法
|
||
//
|
||
// 参数:
|
||
// 1. float dblEigenValue[] - 一维数组,长度为矩阵的阶数,返回时存放特征值
|
||
// 2. CMatrix1& mtxEigenVector - 返回时存放特征向量矩阵,其中第i列为与
|
||
// 数组dblEigenValue中第j个特征值对应的特征向量
|
||
// 3. int nMaxIt - 迭代次数,默认值为60
|
||
// 4. float eps - 计算精度,默认值为0.000001
|
||
//
|
||
// 返回值:BOOL型,求解是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::JacobiEigenv(float dblEigenValue[], CMatrix1& mtxEigenVector, int nMaxIt /*= 60*/, float eps /*= 0.000001*/)
|
||
{
|
||
int i,j,p,q,u,w,t,s,l;
|
||
float fm,cn,sn,omega,x,y,d;
|
||
|
||
if (! mtxEigenVector.Init(m_nNumColumns, m_nNumColumns))
|
||
return FALSE;
|
||
|
||
l=1;
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
mtxEigenVector.m_pData[i*m_nNumColumns+i]=1.0;
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
if (i!=j)
|
||
mtxEigenVector.m_pData[i*m_nNumColumns+j]=0.0;
|
||
}
|
||
|
||
while (TRUE)
|
||
{
|
||
fm=0.0;
|
||
for (i=1; i<=m_nNumColumns-1; i++)
|
||
{
|
||
for (j=0; j<=i-1; j++)
|
||
{
|
||
d=fabs(m_pData[i*m_nNumColumns+j]);
|
||
if ((i!=j)&&(d>fm))
|
||
{
|
||
fm=d;
|
||
p=i;
|
||
q=j;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (fm<eps)
|
||
{
|
||
for (i=0; i<m_nNumColumns; ++i)
|
||
dblEigenValue[i] = GetElement(i,i);
|
||
return TRUE;
|
||
}
|
||
|
||
if (l>nMaxIt)
|
||
return FALSE;
|
||
|
||
l=l+1;
|
||
u=p*m_nNumColumns+q;
|
||
w=p*m_nNumColumns+p;
|
||
t=q*m_nNumColumns+p;
|
||
s=q*m_nNumColumns+q;
|
||
x=-m_pData[u];
|
||
y=(m_pData[s]-m_pData[w])/2.0;
|
||
omega=x/sqrt(x*x+y*y);
|
||
|
||
if (y<0.0)
|
||
omega=-omega;
|
||
|
||
sn=1.0+sqrt(1.0-omega*omega);
|
||
sn=omega/sqrt(2.0*sn);
|
||
cn=sqrt(1.0-sn*sn);
|
||
fm=m_pData[w];
|
||
m_pData[w]=fm*cn*cn+m_pData[s]*sn*sn+m_pData[u]*omega;
|
||
m_pData[s]=fm*sn*sn+m_pData[s]*cn*cn-m_pData[u]*omega;
|
||
m_pData[u]=0.0;
|
||
m_pData[t]=0.0;
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
{
|
||
if ((j!=p)&&(j!=q))
|
||
{
|
||
u=p*m_nNumColumns+j; w=q*m_nNumColumns+j;
|
||
fm=m_pData[u];
|
||
m_pData[u]=fm*cn+m_pData[w]*sn;
|
||
m_pData[w]=-fm*sn+m_pData[w]*cn;
|
||
}
|
||
}
|
||
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
if ((i!=p)&&(i!=q))
|
||
{
|
||
u=i*m_nNumColumns+p;
|
||
w=i*m_nNumColumns+q;
|
||
fm=m_pData[u];
|
||
m_pData[u]=fm*cn+m_pData[w]*sn;
|
||
m_pData[w]=-fm*sn+m_pData[w]*cn;
|
||
}
|
||
}
|
||
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
u=i*m_nNumColumns+p;
|
||
w=i*m_nNumColumns+q;
|
||
fm=mtxEigenVector.m_pData[u];
|
||
mtxEigenVector.m_pData[u]=fm*cn+mtxEigenVector.m_pData[w]*sn;
|
||
mtxEigenVector.m_pData[w]=-fm*sn+mtxEigenVector.m_pData[w]*cn;
|
||
}
|
||
}
|
||
|
||
for (i=0; i<m_nNumColumns; ++i)
|
||
dblEigenValue[i] = GetElement(i,i);
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
//////////////////////////////////////////////////////////////////////
|
||
// 求实对称矩阵特征值与特征向量的雅可比过关法
|
||
//
|
||
// 参数:
|
||
// 1. float dblEigenValue[] - 一维数组,长度为矩阵的阶数,返回时存放特征值
|
||
// 2. CMatrix1& mtxEigenVector - 返回时存放特征向量矩阵,其中第i列为与
|
||
// 数组dblEigenValue中第j个特征值对应的特征向量
|
||
// 3. float eps - 计算精度,默认值为0.000001
|
||
//
|
||
// 返回值:BOOL型,求解是否成功
|
||
//////////////////////////////////////////////////////////////////////
|
||
BOOL CMatrix1::JacobiEigenv2(float dblEigenValue[], CMatrix1& mtxEigenVector, float eps /*= 0.000001*/)
|
||
{
|
||
int i,j,p,q,u,w,t,s;
|
||
float ff,fm,cn,sn,omega,x,y,d;
|
||
|
||
if (! mtxEigenVector.Init(m_nNumColumns, m_nNumColumns))
|
||
return FALSE;
|
||
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
mtxEigenVector.m_pData[i*m_nNumColumns+i]=1.0;
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
if (i!=j)
|
||
mtxEigenVector.m_pData[i*m_nNumColumns+j]=0.0;
|
||
}
|
||
|
||
ff=0.0;
|
||
for (i=1; i<=m_nNumColumns-1; i++)
|
||
{
|
||
for (j=0; j<=i-1; j++)
|
||
{
|
||
d=m_pData[i*m_nNumColumns+j];
|
||
ff=ff+d*d;
|
||
}
|
||
}
|
||
|
||
ff=sqrt(2.0*ff);
|
||
|
||
Loop_0:
|
||
|
||
ff=ff/(1.0*m_nNumColumns);
|
||
|
||
Loop_1:
|
||
|
||
for (i=1; i<=m_nNumColumns-1; i++)
|
||
{
|
||
for (j=0; j<=i-1; j++)
|
||
{
|
||
d=fabs(m_pData[i*m_nNumColumns+j]);
|
||
if (d>ff)
|
||
{
|
||
p=i;
|
||
q=j;
|
||
goto Loop_2;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (ff<eps)
|
||
{
|
||
for (i=0; i<m_nNumColumns; ++i)
|
||
dblEigenValue[i] = GetElement(i,i);
|
||
return TRUE;
|
||
}
|
||
|
||
goto Loop_0;
|
||
|
||
Loop_2:
|
||
|
||
u=p*m_nNumColumns+q;
|
||
w=p*m_nNumColumns+p;
|
||
t=q*m_nNumColumns+p;
|
||
s=q*m_nNumColumns+q;
|
||
x=-m_pData[u];
|
||
y=(m_pData[s]-m_pData[w])/2.0;
|
||
omega=x/sqrt(x*x+y*y);
|
||
if (y<0.0)
|
||
omega=-omega;
|
||
|
||
sn=1.0+sqrt(1.0-omega*omega);
|
||
sn=omega/sqrt(2.0*sn);
|
||
cn=sqrt(1.0-sn*sn);
|
||
fm=m_pData[w];
|
||
m_pData[w]=fm*cn*cn+m_pData[s]*sn*sn+m_pData[u]*omega;
|
||
m_pData[s]=fm*sn*sn+m_pData[s]*cn*cn-m_pData[u]*omega;
|
||
m_pData[u]=0.0; m_pData[t]=0.0;
|
||
|
||
for (j=0; j<=m_nNumColumns-1; j++)
|
||
{
|
||
if ((j!=p)&&(j!=q))
|
||
{
|
||
u=p*m_nNumColumns+j;
|
||
w=q*m_nNumColumns+j;
|
||
fm=m_pData[u];
|
||
m_pData[u]=fm*cn+m_pData[w]*sn;
|
||
m_pData[w]=-fm*sn+m_pData[w]*cn;
|
||
}
|
||
}
|
||
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
if ((i!=p)&&(i!=q))
|
||
{
|
||
u=i*m_nNumColumns+p;
|
||
w=i*m_nNumColumns+q;
|
||
fm=m_pData[u];
|
||
m_pData[u]=fm*cn+m_pData[w]*sn;
|
||
m_pData[w]=-fm*sn+m_pData[w]*cn;
|
||
}
|
||
}
|
||
|
||
for (i=0; i<=m_nNumColumns-1; i++)
|
||
{
|
||
u=i*m_nNumColumns+p;
|
||
w=i*m_nNumColumns+q;
|
||
fm=mtxEigenVector.m_pData[u];
|
||
mtxEigenVector.m_pData[u]=fm*cn+mtxEigenVector.m_pData[w]*sn;
|
||
mtxEigenVector.m_pData[w]=-fm*sn+mtxEigenVector.m_pData[w]*cn;
|
||
}
|
||
|
||
goto Loop_1;
|
||
}
|
||
/*
|
||
|
||
for(i=0;i<4*p*N;i++)
|
||
{
|
||
matrixA[i]=Z[i/(2*N)][(i%(2*p)];
|
||
}
|
||
CMatrix1 mtxA(2*N,2*p,matrixA[]);
|
||
CMatrix1& mtxResult;
|
||
CMatrix1& mtxQ;
|
||
CMatrix1& mtxR;
|
||
CLEquation leq()
|
||
GetRootsetMqr( mtxResult, mtxQ, mtxR);
|
||
f=mtxResult.GetData()
|
||
for(i=0;i<p;i++)
|
||
{
|
||
h[i].real=f[i];
|
||
}
|
||
for(i=p;i<2*p;i++)
|
||
{
|
||
h[i].real=f[i];
|
||
}
|
||
return h;
|
||
}
|
||
*/ |