-
Notifications
You must be signed in to change notification settings - Fork 11
/
KmerIterator.cpp
115 lines (100 loc) · 2.58 KB
/
KmerIterator.cpp
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
103
104
105
106
107
108
109
110
111
112
113
114
115
#include <iterator>
#include <utility>
#include "Kmer.hpp"
#include "KmerIterator.hpp"
/* Note: That an iter is exhausted means that (iter._invalid == true) */
// use: ++iter;
// pre:
// post: *iter is now exhausted
// OR *iter is the next valid pair of kmer and location
KmerIterator& KmerIterator::operator++() {
int pos_ = p_.second;
if (!invalid_) {
if (s_[pos_+Kmer::k] == 0) {
invalid_ = true;
return *this;
} else {
find_next(pos_,pos_+Kmer::k-1,true);
return *this;
}
}
return *this;
}
// use: iter++;
// pre:
// post: iter has been incremented by one
KmerIterator KmerIterator::operator++(int) {
KmerIterator tmp(*this);
operator++();
return tmp;
}
// use: val = (a == b);
// pre:
// post: (val == true) if a and b are both exhausted
// OR a and b are in the same location of the same string.
// (val == false) otherwise.
bool KmerIterator::operator==(const KmerIterator& o) {
if (invalid_ || o.invalid_) {
return invalid_ && o.invalid_;
} else {
return (s_ == o.s_) && (p_.second == o.p_.second);
}
}
// use: p = *iter;
// pre:
// post: p is NULL or a pair of Kmer and int
std::pair<Kmer, int>& KmerIterator::operator*() {
return p_;
}
// use: example 1: km = iter->first;
// example 2: i = iter->second;
// pre: *iter is not NULL
// post: km will be (*iter).first, i will be (*iter).second
std::pair<Kmer, int> *KmerIterator::operator->() {
return &(operator*());
}
// use: iter.raise(km, rep);
// post: iter has been incremented by one
// if iter is not invalid, km is iter->first and rep is km.rep()
void KmerIterator::raise(Kmer& km, Kmer& rep) {
operator++();
if (!invalid_) {
km = p_.first;
rep = km.rep();
}
}
// use: find_next(i,j, last_valid);
// pre:
// post: *iter is either invalid or is a pair of:
// 1) the next valid kmer in the string that does not have any 'N'
// 2) the location of that kmer in the string
void KmerIterator::find_next(size_t i, size_t j, bool last_valid) {
++i;
++j;
while (s_[j] != 0) {
char c = s_[j];
if (c == 'A' || c == 'C' || c == 'G' || c == 'T') {
if (last_valid) {
p_.first = p_.first.forwardBase(c);
break; // default case,
} else {
if (i + Kmer::k - 1 == j) {
p_.first = Kmer(s_+i);
last_valid = true;
break; // create k-mer from scratch
} else {
++j;
}
}
} else {
++j;
i = j;
last_valid = false;
}
}
if (i+Kmer::k-1 == j && s_[j] != 0) {
p_.second = i;
} else {
invalid_ = true;
}
}