-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathrrefs.m
102 lines (96 loc) · 2.62 KB
/
rrefs.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
function out = rrefs (mat, reduce)
% RREFS Reduced Row Echelon Form with Steps
% License: GPL-3.0
% out = rrefs(mat, reduce) show step by step procedure to get to (reduced)
% row echelon form of matrix mat, then return manipulated matrix.
% WARNING: This function may not be reliable for some matrices.
% The user would usually benefit from rational formatting
format rat;
% Save a copy of old matrix for comparizion
oldmat = mat;
% Cache size
[row, col] = size(mat);
% Leading entry row counter
leading = 1;
% Loop through each colomn
if (reduce)
a = col;
else
a = col - 1;
end % if
for c = 1:a
% Find the first non-zero entry and put to top
if (mat(leading, c) == 0)
% Skip if all zeros
if (mat(leading:row, c) == zeros(size(leading:row))')
continue;
end % if
% Get leading entry to 1
for r = leading:row
if (mat(r, c) ~= 0)
mat = ro_swap(mat, leading, r);
break;
end % if
end % for
end % if
% Check if the leading entry is here
if (mat(leading, c) ~= 0)
% Scale the leading entry
if (mat(leading, c) ~= 1)
mat = ro_scale(mat, leading, 1/mat(leading, c));
end % if
% Zero out other rows
if (reduce)
a = 1;
else
a = leading + 1;
end % if
for r = a:row
% Don't remove leading entry, ignore if it's zero already
if (r == leading || mat(r, c) == 0)
continue;
end % if
% Zero out the row
mat = ro_add(mat, r, leading, -1 * mat(r, c));
end % for
end % if
% Change leading counter
leading = leading + 1;
if (leading > row)
break;
end % if
end % if
% Assign return matrix
out = mat;
% Show results from built in function
disp('Done! Result of built in rref(): ');
disp(rref(oldmat));
% Reset formatting
format short;
end % function
function out = ro_add (mat, r_target, r_source, factor)
% Add row r_source times factor to row r_target
out = mat;
out(r_target, :) = out(r_target, :) + factor * out(r_source, :);
% Log
fprintf('R%d + (%s)R%d -> R%d\n', r_target, strtrim(rats(factor)), ...
r_source, r_target);
disp(out);
end % function
function out = ro_scale (mat, r, factor)
% Multiply row r by factor
out = mat;
out(r, :) = out(r, :) * factor;
% Log
fprintf('(%s)R%d -> R%d\n', strtrim(rats(factor)), r, r);
disp(out);
end % function
function out = ro_swap (mat, r1, r2)
% Swap row r1 with row r2
out = mat;
out(r1, :) = mat(r2, :);
out(r2, :) = mat(r1, :);
% Log
fprintf('R%d <-> R%d\n', r1, r2);
disp(out);
end % function