mirror of
https://github.com/NGSolve/netgen.git
synced 2025-01-12 22:20:35 +05:00
787 lines
14 KiB
C++
787 lines
14 KiB
C++
#ifdef abc
|
|
#include <mystdlib.h>
|
|
#include <linalg.hpp>
|
|
#include <algorithm>
|
|
|
|
namespace netgen
|
|
{
|
|
|
|
double BaseVector :: shit = 0;
|
|
|
|
// %FD Constructs a vector of length zero
|
|
BaseVector :: BaseVector ()
|
|
{
|
|
length = 0;
|
|
}
|
|
|
|
// %FD Constructs a vector of given length
|
|
BaseVector :: BaseVector (
|
|
INDEX alength // length of the vector
|
|
)
|
|
{
|
|
length = alength;
|
|
}
|
|
|
|
// %FD Sets length of the vector, old vector will be destroyed
|
|
void
|
|
BaseVector :: SetLength (
|
|
INDEX alength // new length of the vector
|
|
)
|
|
{
|
|
length = alength;
|
|
}
|
|
|
|
// %FD Changes length of the vector, old values stay valid
|
|
void
|
|
BaseVector :: ChangeLength (
|
|
INDEX alength // new length of the vector
|
|
)
|
|
{
|
|
length = alength;
|
|
}
|
|
|
|
|
|
|
|
// %FD { Write a vector with the help of the '<<' operator onto a stream }
|
|
ostream & // stream for further use
|
|
operator<< (
|
|
ostream & s, // stream to write vector onto
|
|
const BaseVector & v // vector to write
|
|
)
|
|
{
|
|
return v.Print (s);
|
|
}
|
|
|
|
|
|
// %FD{ Divides every component of the vector by the scalar c.
|
|
// The function checks for division by zero }
|
|
BaseVector & // result vector
|
|
BaseVector :: operator/= (
|
|
double c // scalar to divide by
|
|
)
|
|
{
|
|
if (c)
|
|
return (*this) *= (1/c);
|
|
else
|
|
{
|
|
(*myerr) << "operator/=: division by zero" << endl;
|
|
return *this;
|
|
}
|
|
}
|
|
|
|
|
|
// %FD Creates a copy of the object
|
|
BaseVector * // pointer to the new vector
|
|
BaseVector :: Copy () const
|
|
{
|
|
cerr << "Base_vector::Copy called" << endl << flush;
|
|
return NULL;
|
|
}
|
|
|
|
|
|
|
|
|
|
void BaseVector :: GetElementVector (const Array<INDEX> & pnum,
|
|
BaseVector & elvec) const
|
|
{
|
|
int i;
|
|
for (i = 1; i <= pnum.Size(); i++)
|
|
elvec(i) = (*this)(pnum.Get(i));
|
|
}
|
|
|
|
void BaseVector :: SetElementVector (const Array<INDEX> & pnum,
|
|
const BaseVector & elvec)
|
|
{
|
|
int i;
|
|
for (i = 1; i <= pnum.Size(); i++)
|
|
(*this)(pnum.Get(i)) = elvec(i);
|
|
}
|
|
|
|
|
|
void BaseVector :: AddElementVector (const Array<INDEX> & pnum,
|
|
const BaseVector & elvec)
|
|
{
|
|
int i;
|
|
for (i = 1; i <= pnum.Size(); i++)
|
|
(*this)(pnum.Get(i)) += elvec(i);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
TempVector :: ~TempVector ()
|
|
{
|
|
delete vec;
|
|
}
|
|
|
|
TempVector BaseVector :: operator+ (const BaseVector & v2) const
|
|
{
|
|
return (*Copy()) += v2;
|
|
}
|
|
|
|
TempVector BaseVector :: operator- (const BaseVector & v2) const
|
|
{
|
|
return (*Copy()) -= v2;
|
|
}
|
|
|
|
TempVector BaseVector :: operator- () const
|
|
{
|
|
return (*Copy()) *= -1;
|
|
}
|
|
|
|
|
|
TempVector operator* (const BaseVector & v1, double scal)
|
|
{
|
|
return (*v1.Copy()) *= scal;
|
|
}
|
|
|
|
TempVector operator/ (const BaseVector & v1, double scal)
|
|
{
|
|
return (*v1.Copy()) /= scal;
|
|
}
|
|
|
|
|
|
TempVector operator* (double scal, const BaseVector & v1)
|
|
{
|
|
return v1 * scal;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BaseVector * TempVector :: Copy () const
|
|
{
|
|
return vec->Copy();
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
double Vector :: shit = 0;
|
|
|
|
class clVecpool
|
|
{
|
|
public:
|
|
Array<double *> vecs;
|
|
Array<INDEX> veclens;
|
|
|
|
~clVecpool();
|
|
};
|
|
|
|
clVecpool :: ~clVecpool()
|
|
{
|
|
int i;
|
|
for (i = 1; i <= vecs.Size(); i++)
|
|
delete vecs.Elem(i);
|
|
}
|
|
|
|
static clVecpool vecpool;
|
|
|
|
|
|
|
|
static double * NewDouble (INDEX len)
|
|
{
|
|
if (len < 10)
|
|
return new double[len];
|
|
else
|
|
{
|
|
int i;
|
|
for (i = 1; i <= vecpool.veclens.Size(); i++)
|
|
if (vecpool.veclens.Get(i) == len)
|
|
{
|
|
double * hvec = vecpool.vecs.Get(i);
|
|
vecpool.vecs.DeleteElement(i);
|
|
vecpool.veclens.DeleteElement(i);
|
|
return hvec;
|
|
}
|
|
|
|
return new double[len];
|
|
}
|
|
}
|
|
|
|
static void DeleteDouble (INDEX len, double * dp)
|
|
{
|
|
if (len < 10)
|
|
delete [] dp;
|
|
else
|
|
{
|
|
vecpool.vecs.Append (dp);
|
|
vecpool.veclens.Append (len);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
Vector :: Vector () : BaseVector()
|
|
{
|
|
data = NULL;
|
|
}
|
|
|
|
Vector :: Vector (INDEX alength) : BaseVector (alength)
|
|
{
|
|
if (length)
|
|
{
|
|
// data = new double[length];
|
|
data = NewDouble (length);
|
|
|
|
if (!data)
|
|
{
|
|
length = 0;
|
|
(*myerr) << "Vector not allocated" << endl;
|
|
}
|
|
}
|
|
else
|
|
data = NULL;
|
|
}
|
|
|
|
|
|
Vector :: Vector (const Vector & v2)
|
|
{
|
|
length = v2.length;
|
|
|
|
if (length)
|
|
{
|
|
// data = new double[length];
|
|
data = NewDouble (length);
|
|
|
|
if (data)
|
|
{
|
|
memcpy (data, v2.data, length * sizeof (double));
|
|
}
|
|
else
|
|
{
|
|
length = 0;
|
|
(*myerr) << "Vector::Vector : Vector not allocated" << endl;
|
|
}
|
|
}
|
|
else
|
|
data = NULL;
|
|
}
|
|
|
|
|
|
Vector :: ~Vector ()
|
|
{
|
|
// veclenfile << "~Vector delete: " << length << endl;
|
|
if (data)
|
|
{
|
|
DeleteDouble (length, data);
|
|
// delete [] data;
|
|
}
|
|
|
|
}
|
|
|
|
void Vector :: SetLength (INDEX alength)
|
|
{
|
|
if (length == alength) return;
|
|
|
|
if (data)
|
|
{
|
|
DeleteDouble (length, data);
|
|
// delete [] data;
|
|
}
|
|
data = NULL;
|
|
length = alength;
|
|
|
|
if (length == 0) return;
|
|
// data = new double[length];
|
|
data = NewDouble (length);
|
|
|
|
if (!data)
|
|
{
|
|
length = 0;
|
|
(*myerr) << "Vector::SetLength: Vector not allocated" << endl;
|
|
}
|
|
}
|
|
|
|
void Vector :: ChangeLength (INDEX alength)
|
|
{
|
|
(*mycout) << "Vector::ChangeLength called" << endl;
|
|
if (length == alength) return;
|
|
|
|
if (alength == 0)
|
|
{
|
|
// delete [] data;
|
|
DeleteDouble (length, data);
|
|
length = 0;
|
|
return;
|
|
}
|
|
|
|
double * olddata = data;
|
|
|
|
data = NewDouble (alength);
|
|
// data = new double[alength];
|
|
if (!data)
|
|
{
|
|
length = 0;
|
|
(*myerr) << "Vector::SetLength: Vector not allocated" << endl;
|
|
delete [] olddata;
|
|
}
|
|
|
|
memcpy (data, olddata, min2(alength, length));
|
|
|
|
delete [] olddata;
|
|
length = alength;
|
|
}
|
|
|
|
/// NEW RM
|
|
void Vector::SetBlockLength (INDEX /* blength */)
|
|
{
|
|
MyError("BaseVector::SetBlockLength was called for a Vector");
|
|
}
|
|
|
|
|
|
double & Vector :: operator() (INDEX i)
|
|
{
|
|
if (i >= 1 && i <= length) return Elem(i);
|
|
else (*myerr) << "\nindex " << i << " out of range ("
|
|
<< 1 << "," << Length() << ")\n";
|
|
return shit;
|
|
}
|
|
|
|
double Vector :: operator() (INDEX i) const
|
|
{
|
|
if (i >= 1 && i <= length) return Get(i);
|
|
else (*myerr) << "\nindex " << i << " out of range ("
|
|
<< 1 << "," << Length() << ")\n" << flush;
|
|
return shit;
|
|
}
|
|
|
|
|
|
|
|
double Vector :: SupNorm () const
|
|
{
|
|
double sup = 0;
|
|
|
|
for (INDEX i = 1; i <= Length(); i++)
|
|
if (fabs (Get(i)) > sup)
|
|
sup = fabs(Get(i));
|
|
|
|
return sup;
|
|
}
|
|
|
|
double Vector :: L2Norm () const
|
|
{
|
|
double sum = 0;
|
|
|
|
for (INDEX i = 1; i <= Length(); i++)
|
|
sum += Get(i) * Get(i);
|
|
|
|
return sqrt (sum);
|
|
}
|
|
|
|
double Vector :: L1Norm () const
|
|
{
|
|
double sum = 0;
|
|
|
|
for (INDEX i = 1; i <= Length(); i++)
|
|
sum += fabs (Get(i));
|
|
|
|
return sum;
|
|
}
|
|
|
|
double Vector :: Max () const
|
|
{
|
|
if (!Length()) return 0;
|
|
double m = Get(1);
|
|
for (INDEX i = 2; i <= Length(); i++)
|
|
if (Get(i) > m) m = Get(i);
|
|
return m;
|
|
}
|
|
|
|
double Vector :: Min () const
|
|
{
|
|
if (!Length()) return 0;
|
|
double m = Get(1);
|
|
for (INDEX i = 2; i <= Length(); i++)
|
|
if (Get(i) < m) m = Get(i);
|
|
return m;
|
|
}
|
|
|
|
|
|
/*
|
|
ostream & operator<<(ostream & s, const Vector & v)
|
|
{
|
|
int w = s.width();
|
|
if (v.Length())
|
|
{
|
|
s.width(0);
|
|
s << '(';
|
|
for (INDEX i = 1; i < v.Length(); i++)
|
|
{
|
|
s.width(w);
|
|
s << v.Get(i) << ",";
|
|
if (i % 8 == 0) s << endl << ' ';
|
|
}
|
|
s.width(w);
|
|
s << v.Get(v.Length()) << ')';
|
|
}
|
|
else
|
|
s << "(Vector not allocated)";
|
|
|
|
return s;
|
|
}
|
|
*/
|
|
|
|
ostream & Vector :: Print (ostream & s) const
|
|
{
|
|
int w = s.width();
|
|
if (Length())
|
|
{
|
|
s.width(0);
|
|
s << '(';
|
|
for (INDEX i = 1; i < Length(); i++)
|
|
{
|
|
s.width(w);
|
|
s << Get(i) << ",";
|
|
if (i % 8 == 0) s << endl << ' ';
|
|
}
|
|
s.width(w);
|
|
s << Get(Length()) << ')';
|
|
}
|
|
else
|
|
s << "(Vector not allocated)";
|
|
|
|
return s;
|
|
}
|
|
|
|
|
|
|
|
BaseVector & Vector :: operator+= (const BaseVector & v2)
|
|
{
|
|
const Vector & hv2 = v2.CastToVector();
|
|
|
|
if (Length() == hv2.Length())
|
|
for (INDEX i = 1; i <= Length(); i++)
|
|
Elem (i) += hv2.Get(i);
|
|
else
|
|
(*myerr) << "operator+= illegal dimension" << endl;
|
|
return *this;
|
|
}
|
|
|
|
BaseVector & Vector :: operator-= (const BaseVector & v2)
|
|
{
|
|
const Vector & hv2 = v2.CastToVector();
|
|
|
|
if (Length() == hv2.Length())
|
|
for (INDEX i = 1; i <= Length(); i++)
|
|
Elem (i) -= hv2.Get(i);
|
|
else
|
|
(*myerr) << "operator-= illegal dimension" << endl;
|
|
return *this;
|
|
}
|
|
|
|
BaseVector & Vector :: operator*= (double c)
|
|
{
|
|
for (INDEX i = 1; i <= Length(); i++)
|
|
Elem(i) *= c;
|
|
return *this;
|
|
}
|
|
|
|
|
|
|
|
BaseVector & Vector :: Add (double scal, const BaseVector & v2)
|
|
{
|
|
const Vector & hv2 = v2.CastToVector();
|
|
|
|
if (Length() == hv2.Length())
|
|
{
|
|
double * p1 = data;
|
|
double * p2 = hv2.data;
|
|
|
|
for (INDEX i = Length(); i > 0; i--)
|
|
{
|
|
(*p1) += scal * (*p2);
|
|
p1++; p2++;
|
|
}
|
|
}
|
|
else
|
|
(*myerr) << "Vector::Add: illegal dimension" << endl;
|
|
return *this;
|
|
}
|
|
|
|
BaseVector & Vector :: Add2 (double scal, const BaseVector & v2,
|
|
double scal3, const BaseVector & v3)
|
|
{
|
|
const Vector & hv2 = v2.CastToVector();
|
|
const Vector & hv3 = v3.CastToVector();
|
|
|
|
if (Length() == hv2.Length())
|
|
{
|
|
double * p1 = data;
|
|
double * p2 = hv2.data;
|
|
double * p3 = hv3.data;
|
|
|
|
for (INDEX i = Length(); i > 0; i--)
|
|
{
|
|
(*p1) += (scal * (*p2) + scal3 * (*p3));
|
|
p1++; p2++; p3++;
|
|
}
|
|
}
|
|
else
|
|
(*myerr) << "Vector::Add: illegal dimension" << endl;
|
|
return *this;
|
|
}
|
|
|
|
BaseVector & Vector :: Set (double scal, const BaseVector & v2)
|
|
{
|
|
const Vector & hv2 = v2.CastToVector();
|
|
|
|
if (Length() == hv2.Length())
|
|
{
|
|
double * p1 = data;
|
|
double * p2 = hv2.data;
|
|
|
|
for (INDEX i = Length(); i > 0; i--)
|
|
{
|
|
(*p1) = scal * (*p2);
|
|
p1++; p2++;
|
|
}
|
|
}
|
|
else
|
|
(*myerr) << "Vector::Set: illegal dimension" << endl;
|
|
return *this;
|
|
}
|
|
|
|
|
|
BaseVector & Vector :: Set2 (double scal , const BaseVector & v2,
|
|
double scal3, const BaseVector & v3)
|
|
{
|
|
const Vector & hv2 = v2.CastToVector();
|
|
const Vector & hv3 = v3.CastToVector();
|
|
|
|
if (Length() == hv2.Length() && Length() == hv3.Length())
|
|
{
|
|
double * p1 = data;
|
|
double * p2 = hv2.data;
|
|
double * p3 = hv3.data;
|
|
|
|
for (INDEX i = Length(); i > 0; i--)
|
|
{
|
|
(*p1) = scal * (*p2) + scal3 * (*p3);
|
|
p1++; p2++; p3++;
|
|
}
|
|
}
|
|
else
|
|
(*myerr) << "Vector::Set: illegal dimension" << endl;
|
|
return *this;
|
|
}
|
|
|
|
|
|
void Vector :: GetPart (int startpos, BaseVector & v2) const
|
|
{
|
|
Vector & hv2 = v2.CastToVector();
|
|
|
|
if (Length() >= startpos + v2.Length() - 1)
|
|
{
|
|
const double * p1 = &Get(startpos);
|
|
double * p2 = &hv2.Elem(1);
|
|
|
|
memcpy (p2, p1, hv2.Length() * sizeof(double));
|
|
}
|
|
else
|
|
MyError ("Vector::GetPart: Vector to short");
|
|
}
|
|
|
|
|
|
// NEW RM
|
|
void Vector :: SetPart (int startpos, const BaseVector & v2)
|
|
{
|
|
const Vector & hv2 = v2.CastToVector();
|
|
INDEX i;
|
|
INDEX n = v2.Length();
|
|
|
|
if (Length() >= startpos + n - 1)
|
|
{
|
|
double * p1 = &Elem(startpos);
|
|
const double * p2 = &hv2.Get(1);
|
|
|
|
for (i = 1; i <= n; i++)
|
|
{
|
|
(*p1) = (*p2);
|
|
p1++;
|
|
p2++;
|
|
}
|
|
}
|
|
else
|
|
MyError ("Vector::SetPart: Vector to short");
|
|
}
|
|
|
|
void Vector :: AddPart (int startpos, double val, const BaseVector & v2)
|
|
{
|
|
const Vector & hv2 = v2.CastToVector();
|
|
INDEX i;
|
|
INDEX n = v2.Length();
|
|
|
|
if (Length() >= startpos + n - 1)
|
|
{
|
|
double * p1 = &Elem(startpos);
|
|
const double * p2 = &hv2.Get(1);
|
|
|
|
for (i = 1; i <= n; i++)
|
|
{
|
|
(*p1) += val * (*p2);
|
|
p1++;
|
|
p2++;
|
|
}
|
|
}
|
|
else
|
|
MyError ("Vector::AddPart: Vector to short");
|
|
}
|
|
|
|
|
|
|
|
|
|
double Vector :: operator* (const BaseVector & v2) const
|
|
{
|
|
const Vector & hv2 = v2.CastToVector();
|
|
|
|
double sum = 0;
|
|
double * p1 = data;
|
|
double * p2 = hv2.data;
|
|
|
|
if (Length() == hv2.Length())
|
|
{
|
|
for (INDEX i = Length(); i > 0; i--)
|
|
{
|
|
sum += (*p1) * (*p2);
|
|
p1++; p2++;
|
|
}
|
|
}
|
|
else
|
|
(*myerr) << "Scalarproduct illegal dimension" << endl;
|
|
return sum;
|
|
}
|
|
|
|
void Vector :: SetRandom ()
|
|
{
|
|
INDEX i;
|
|
for (i = 1; i <= Length(); i++)
|
|
Elem(i) = rand ();
|
|
|
|
double l2 = L2Norm();
|
|
if (l2 > 0)
|
|
(*this) /= l2;
|
|
// Elem(i) = 1.0 / double(i);
|
|
// Elem(i) = drand48();
|
|
}
|
|
|
|
|
|
/*
|
|
TempVector Vector :: operator- () const
|
|
{
|
|
Vector & sum = *(Vector*)Copy();
|
|
|
|
if (sum.Length () == Length())
|
|
{
|
|
for (INDEX i = 1; i <= Length(); i++)
|
|
sum.Set (i, Get(i));
|
|
}
|
|
else
|
|
(*myerr) << "operator+ (Vector, Vector): sum.Length() not ok" << endl;
|
|
return sum;
|
|
}
|
|
*/
|
|
|
|
BaseVector & Vector::operator= (const Vector & v2)
|
|
{
|
|
SetLength (v2.Length());
|
|
|
|
if (data == v2.data) return *this;
|
|
|
|
if (v2.Length() == Length())
|
|
memcpy (data, v2.data, sizeof (double) * Length());
|
|
else
|
|
(*myerr) << "Vector::operator=: not allocated" << endl;
|
|
|
|
return *this;
|
|
}
|
|
|
|
BaseVector & Vector::operator= (const BaseVector & v2)
|
|
{
|
|
const Vector & hv2 = v2.CastToVector();
|
|
|
|
SetLength (hv2.Length());
|
|
|
|
if (data == hv2.data) return *this;
|
|
|
|
if (hv2.Length() == Length())
|
|
memcpy (data, hv2.data, sizeof (double) * Length());
|
|
else
|
|
(*myerr) << "Vector::operator=: not allocated" << endl;
|
|
|
|
return *this;
|
|
}
|
|
|
|
|
|
BaseVector & Vector::operator= (double scal)
|
|
{
|
|
if (!Length()) (*myerr) << "Vector::operator= (double) : data not allocated"
|
|
<< endl;
|
|
|
|
for (INDEX i = 1; i <= Length(); i++)
|
|
Set (i, scal);
|
|
|
|
return *this;
|
|
}
|
|
|
|
|
|
BaseVector * Vector :: Copy () const
|
|
{
|
|
return new Vector (*this);
|
|
}
|
|
|
|
|
|
void Vector :: Swap (BaseVector & v2)
|
|
{
|
|
Vector & hv2 = v2.CastToVector();
|
|
swap (length, hv2.length);
|
|
swap (data, hv2.data);
|
|
}
|
|
|
|
|
|
|
|
|
|
void Vector :: GetElementVector (const Array<INDEX> & pnum,
|
|
BaseVector & elvec) const
|
|
{
|
|
int i;
|
|
Vector & helvec = elvec.CastToVector();
|
|
for (i = 1; i <= pnum.Size(); i++)
|
|
helvec.Elem(i) = Get(pnum.Get(i));
|
|
}
|
|
|
|
void Vector :: SetElementVector (const Array<INDEX> & pnum,
|
|
const BaseVector & elvec)
|
|
{
|
|
int i;
|
|
const Vector & helvec = elvec.CastToVector();
|
|
for (i = 1; i <= pnum.Size(); i++)
|
|
Elem(pnum.Get(i)) = helvec.Get(i);
|
|
}
|
|
|
|
|
|
void Vector :: AddElementVector (const Array<INDEX> & pnum,
|
|
const BaseVector & elvec)
|
|
{
|
|
int i;
|
|
const Vector & helvec = elvec.CastToVector();
|
|
for (i = 1; i <= pnum.Size(); i++)
|
|
Elem(pnum.Get(i)) += helvec.Get(i);
|
|
}
|
|
}
|
|
#endif
|