Skip to content

Commit bb393e9

Browse files
Merge pull request #287 from PolyMathOrg/qr-decomposition-extract-class
Refactor: Extract QR Decomposition Class
2 parents fc2a331 + aa0e360 commit bb393e9

File tree

3 files changed

+112
-35
lines changed

3 files changed

+112
-35
lines changed

src/Math-Matrix/PMMatrix.class.st

+7-35
Original file line numberDiff line numberDiff line change
@@ -409,6 +409,12 @@ PMMatrix >> columnAt: anInteger [
409409
^ rows collect: [ :each | each at: anInteger ]
410410
]
411411

412+
{ #category : #'cell accessing' }
413+
PMMatrix >> columnVectorAt: col size: dimension [
414+
415+
^ (self columnAt: col) copyFrom: col to: dimension
416+
]
417+
412418
{ #category : #iterators }
413419
PMMatrix >> columnsCollect: aBlock [
414420
"Perform the collect: operation on the rows of the receiver."
@@ -751,41 +757,7 @@ PMMatrix >> productWithVector: aVector [
751757
{ #category : #'as yet unclassified' }
752758
PMMatrix >> qrFactorization [
753759

754-
| identMat q r hh colSize i |
755-
self numberOfRows < self numberOfColumns ifTrue: [
756-
self error: 'numberOfRows<numberOfColumns' ].
757-
r := PMMatrix rows: rows deepCopy.
758-
colSize := self numberOfRows.
759-
q := PMSymmetricMatrix identity: colSize.
760-
identMat := q deepCopy.
761-
1 to: self numberOfColumns do: [ :col |
762-
hh := ((r columnAt: col) copyFrom: col to: colSize) householder.
763-
i := (PMVector new: col - 1 withAll: 0) , (hh at: 2).
764-
q := q * (identMat - ((hh at: 1) * i tensorProduct: i)). "not really necessary, should be simplified"
765-
i := PMMatrix rows:
766-
((r rows allButFirst: col - 1) collect: [ :aRow |
767-
aRow allButFirst: col - 1 ]).
768-
i := i - ((hh at: 2) tensorProduct: (hh at: 1) * (hh at: 2) * i).
769-
i rows withIndexDo: [ :aRow :index |
770-
aRow withIndexDo: [ :n :c |
771-
r
772-
rowAt: col + index - 1
773-
columnAt: col + c - 1
774-
put: ((n closeTo: 0)
775-
ifTrue: [ 0 ]
776-
ifFalse: [ n ]) ] ]
777-
"col <colSize ifTrue: [i :=(hh at: 2) copyFrom: 2 to: colSize -col +1. i withIndexDo: [:n :index| r rowAt: col columnAt: index put: n ] ]""and this part is not correct, dont uncomment before the bug is corrected! useful if q is not explicitely necessary" ].
778-
"r rows allButFirst withIndexDo: [:aRow :ri|1 to: (ri min: self numberOfColumns ) do: [:ci|aRow at: ci put:0 ] ] ""not necessary with equalsTo:0"
779-
i := 0.
780-
[ (r rowAt: colSize) allSatisfy: [ :n | n = 0 ] ] whileTrue: [
781-
i := i + 1.
782-
colSize := colSize - 1 ].
783-
i > 0 ifTrue: [
784-
r := PMMatrix rows: (r rows copyFrom: 1 to: colSize).
785-
i := q numberOfColumns - i.
786-
q := PMMatrix rows:
787-
(q rows collect: [ :row | row copyFrom: 1 to: i ]) ].
788-
^ Array with: q with: r
760+
^ (PMQRDecomposition of: self) decompose
789761
]
790762

791763
{ #category : #'as yet unclassified' }
+91
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,91 @@
1+
"
2+
I am responsible for the decomposition of a matrix, A say, into a product A = QR of an orthogonal matrix Q and an upper triangular matrix R
3+
"
4+
Class {
5+
#name : #PMQRDecomposition,
6+
#superclass : #Object,
7+
#instVars : [
8+
'matrixToDecompose',
9+
'colSize'
10+
],
11+
#category : #'Math-Matrix'
12+
}
13+
14+
{ #category : #'instance creation' }
15+
PMQRDecomposition class >> of: matrix [
16+
matrix numberOfRows < matrix numberOfColumns ifTrue: [
17+
self error: 'numberOfRows<numberOfColumns' ].
18+
^ self new of: matrix
19+
]
20+
21+
{ #category : #private }
22+
PMQRDecomposition >> decompose [
23+
24+
"
25+
The method appears to be using Householder reflection. There is a wiki page
26+
that describes the mechanics:
27+
https://en.wikipedia.org/wiki/QR_decomposition#Using_Householder_reflections
28+
"
29+
30+
| identityMatrix q r householderVector i |
31+
r := self initialRMatrix.
32+
q := self initialQMatrix.
33+
identityMatrix := PMSymmetricMatrix identity: colSize.
34+
1 to: self numberOfColumns do: [ :col |
35+
| columnVectorFromRMatrix householderMatrix |
36+
columnVectorFromRMatrix := r columnVectorAt: col size: colSize.
37+
householderVector := columnVectorFromRMatrix householder.
38+
i := (PMVector zeros: col - 1) , (householderVector at: 2).
39+
householderMatrix := identityMatrix
40+
-
41+
((householderVector at: 1) * i tensorProduct: i).
42+
q := q * householderMatrix.
43+
i := PMMatrix rows:
44+
((r rows allButFirst: col - 1) collect: [ :aRow |
45+
aRow allButFirst: col - 1 ]).
46+
i := i - ((householderVector at: 2) tensorProduct:
47+
(householderVector at: 1) * (householderVector at: 2) * i).
48+
i rows withIndexDo: [ :aRow :index |
49+
aRow withIndexDo: [ :n :c |
50+
r
51+
rowAt: col + index - 1
52+
columnAt: col + c - 1
53+
put: ((n closeTo: 0)
54+
ifTrue: [ 0 ]
55+
ifFalse: [ n ]) ] ] ].
56+
i := 0.
57+
[ (r rowAt: colSize) allSatisfy: [ :n | n = 0 ] ] whileTrue: [
58+
i := i + 1.
59+
colSize := colSize - 1 ].
60+
i > 0 ifTrue: [
61+
r := PMMatrix rows: (r rows copyFrom: 1 to: colSize).
62+
i := q numberOfColumns - i.
63+
q := PMMatrix rows:
64+
(q rows collect: [ :row | row copyFrom: 1 to: i ]) ].
65+
^ Array with: q with: r
66+
]
67+
68+
{ #category : #private }
69+
PMQRDecomposition >> initialQMatrix [
70+
71+
^ PMSymmetricMatrix identity: colSize
72+
]
73+
74+
{ #category : #private }
75+
PMQRDecomposition >> initialRMatrix [
76+
77+
^ PMMatrix rows: matrixToDecompose rows deepCopy
78+
]
79+
80+
{ #category : #private }
81+
PMQRDecomposition >> numberOfColumns [
82+
83+
^ matrixToDecompose numberOfColumns
84+
]
85+
86+
{ #category : #'instance creation' }
87+
PMQRDecomposition >> of: matrix [
88+
89+
matrixToDecompose := matrix.
90+
colSize := matrixToDecompose numberOfRows.
91+
]

src/Math-Tests-Matrix/PMQRTest.class.st

+14
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,20 @@ PMQRTest >> assert: inverse isMoorePenroseInverseOf: aMatrix [
2727
closeTo: aMatrix mpInverse transpose.
2828
]
2929

30+
{ #category : #tests }
31+
PMQRTest >> testHorizontalRectangularMatrixCannotBeDecomposed [
32+
33+
| horizontalRectangularMatrix |
34+
horizontalRectangularMatrix := PMMatrix rows: {
35+
{ 1. 2. 4 }.
36+
{ 5. 6. 7 } }.
37+
38+
self
39+
should: [
40+
(PMQRDecomposition of: horizontalRectangularMatrix) ]
41+
raise: Error
42+
]
43+
3044
{ #category : #tests }
3145
PMQRTest >> testMoorePenroseInverseOfLargeNonRandomMatrixAndItsTranspose [
3246
| a inverse transposeOfA |

0 commit comments

Comments
 (0)