Skip to content
This repository was archived by the owner on Jun 21, 2024. It is now read-only.

Commit c4361a3

Browse files
committed
feat: add BisectMethod impl
1 parent 83dae88 commit c4361a3

File tree

5 files changed

+151
-8
lines changed

5 files changed

+151
-8
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,104 @@
1+
//
2+
// Created by TianKai Ma on 2023/12/19.
3+
//
4+
5+
#include "BisectMethod.h"
6+
7+
ull CalculateSignChange(const Vector &x, const Vector &y, lld mu) {
8+
#ifdef DEBUG
9+
if (x.size != y.size + 1) {
10+
throw std::invalid_argument("x.size != y.size + 1");
11+
}
12+
#endif
13+
14+
auto n = x.size;
15+
auto s = 0;
16+
auto q = x.array[0] - mu;
17+
for (ull k = 0; k < n; k++) {
18+
if (q < 0) {
19+
s++;
20+
}
21+
if (k != n - 1) {
22+
if (q == 0) {
23+
q = std::abs(y.array[k]) * 1e-10;
24+
}
25+
26+
q = x.array[k + 1] - mu - y.array[k] * y.array[k] / q;
27+
}
28+
}
29+
return s;
30+
}
31+
32+
std::vector<lld> BisectMethodCall(const Vector &x, const Vector &y, lld start, lld end, lld precision) {
33+
#ifdef DEBUG
34+
if (start > end) {
35+
throw std::invalid_argument("start > end");
36+
}
37+
if (precision < 0) {
38+
throw std::invalid_argument("precision < 0");
39+
}
40+
#endif
41+
42+
if (end - start < precision) {
43+
auto l = CalculateSignChange(x, y, start);
44+
auto r = CalculateSignChange(x, y, end);
45+
46+
if (l > r) {
47+
return {};
48+
}
49+
50+
auto result = std::vector<lld>(r - l, (start + end) / 2);
51+
return result;
52+
}
53+
54+
auto mid = (start + end) / 2;
55+
auto l = CalculateSignChange(x, y, start);
56+
auto r = CalculateSignChange(x, y, end);
57+
58+
if (l == r) {
59+
return {};
60+
}
61+
62+
auto m = CalculateSignChange(x, y, mid);
63+
auto result = std::vector<lld>();
64+
if (l != m) {
65+
auto left = BisectMethodCall(x, y, start, mid, precision);
66+
result.insert(result.end(), left.begin(), left.end());
67+
}
68+
if (m != r) {
69+
auto right = BisectMethodCall(x, y, mid, end, precision);
70+
result.insert(result.end(), right.begin(), right.end());
71+
}
72+
return result;
73+
}
74+
75+
Vector BisectMethod(const Vector &x, const Vector &y, lld precision) {
76+
#ifdef DEBUG
77+
if (x.size != y.size + 1) {
78+
throw std::invalid_argument("x.size != y.size + 1");
79+
}
80+
if (precision < 0) {
81+
throw std::invalid_argument("precision < 0");
82+
}
83+
#endif
84+
85+
// determine Infinity norm of T: x_n + y_n + y_(n-1)
86+
auto n = x.size;
87+
auto max = x.array[0];
88+
for (ull i = 1; i < n; i++) {
89+
lld temp;
90+
if (i == n - 1) {
91+
temp = std::abs(x.array[i]) + std::abs(y.array[i]);
92+
} else {
93+
temp = std::abs(x.array[i]) + std::abs(y.array[i - 1]) + std::abs(y.array[i]);
94+
}
95+
if (temp > max) {
96+
max = temp;
97+
}
98+
}
99+
100+
auto l = -max;
101+
auto r = max;
102+
auto result = BisectMethodCall(x, y, l, r, precision);
103+
return Vector(result);
104+
}
+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
//
2+
// Created by TianKai Ma on 2023/12/19.
3+
//
4+
5+
#ifndef NUMERICAL_ALGEBRA_BISECTMETHOD_H
6+
#define NUMERICAL_ALGEBRA_BISECTMETHOD_H
7+
8+
#include "CustomMath_lib.h"
9+
10+
ull CalculateSignChange(const Vector &x, const Vector &y, lld mu);
11+
12+
Vector BisectMethod(const Vector &x, const Vector &y, lld precision = 1e-10);
13+
14+
#endif //NUMERICAL_ALGEBRA_BISECTMETHOD_H

CustomMath_lib/CMakeLists.txt

+2
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,8 @@ set(SOURCE_FILES
4343
Matrix/MatrixOperations.cpp
4444
JacobiMethod/JacobiMethod.cpp
4545
JacobiMethod/JacobiMethod.h
46+
BisectMethod/BisectMethod.cpp
47+
BisectMethod/BisectMethod.h
4648
)
4749

4850
add_library(CustomMath_lib STATIC ${SOURCE_FILES} ${HEADER_FILES})

CustomMath_lib/CustomMath_lib.h

+1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010
#include "Vector/Vector.h"
1111
#include "Matrix/Matrix.h"
1212

13+
#include "BisectMethod/BisectMethod.h"
1314
#include "CholeskyMethod/CholeskyMethod.h"
1415
#include "GaussMethod/GaussMethod.h"
1516
#include "HessenbergMethod/HessenbergMethod.h"

main.cpp

+30-8
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,38 @@
11
#include "CustomMath_lib.h"
22

33
int main() {
4-
auto m = Matrix("[5 1 -2; 1 2 0; -2 0 -10]");
5-
auto I = Matrix::identity(m.rows);
6-
auto lambda = -10.263471;
7-
auto A = m - I * lambda;
8-
auto x = Vector("[1 1 1]");
4+
auto n = 3;
5+
auto x = Vector(n, 2);
6+
auto y = Vector(n - 1, -1);
97

10-
auto result = RevPowerIteration(PowerIterationInput{A, x, 1000});
8+
auto A = Matrix(n, n);
9+
for(ull i = 0; i < n; i++) {
10+
for(ull j = 0; j < n; j++) {
11+
if(i == j) {
12+
A.matrix[i][j] = 2;
13+
} else if(i == j + 1 || i == j - 1) {
14+
A.matrix[i][j] = -1;
15+
}
16+
}
17+
}
18+
auto I = Matrix::identity(n);
19+
auto x_default = Vector(n, 1);
1120

12-
std::cout << "Eigenvector: " << std::endl;
13-
result.result.print();
21+
// for (lld k = -4.0; k <= 4.0; k += 0.1) {
22+
// auto r = CalculateSignChange(x, y, k);
23+
// std::cout << k << " " << r << std::endl;
24+
// }
25+
26+
auto r = BisectMethod(x, y);
27+
r.print();
28+
29+
for (auto &lambda: r.array) {
30+
auto B = A - I * lambda;
31+
auto input = PowerIterationInput{B, x_default, 1000,};
32+
auto k = RevPowerIteration(input);
33+
k.result.print();
34+
(B * k.result).print();
35+
}
1436

1537
return 0;
1638
}

0 commit comments

Comments
 (0)