Newer
Older
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
#ifndef MANTID_GEOMETRY_HKLGENERATOR_H_
#define MANTID_GEOMETRY_HKLGENERATOR_H_
#include "MantidGeometry/DllConfig.h"
#include "MantidGeometry/Crystal/UnitCell.h"
#include "MantidKernel/V3D.h"
#include <boost/iterator/iterator_facade.hpp>
namespace Mantid {
namespace Geometry {
/** HKLGenerator
HKLGenerator is a pseudo-container that helps in generating actual
containers with V3D-objects representing Miller indices (HKL).
It's a common task to generate lists of Miller indices. The simplest
way of doing that is to simply create a nested loop structure that
goes through all combinations of H, K, L within some limits and then
put them into a container (vector, list, set, ...):
for(int h = -hmin; h <= hmax; ++h) {
for(int k = -kmin; k <= kmax; ++k) {
for(int l = -lmin; l <= lmax; ++l) {
hkls.push_back(V3D(h, k, l));
}
}
}
In most cases the list is not used like that, instead it's filtered
using some criteria, for example a certain range of d-spacings or
others, so the reflection needs to be checked first:
...
hkl = V3D(h, k, l)
if(isOk(hkl)) {
hkls.push_back(hkl);
}
...
Instead of explicitly stating the triple-loop, HKLGenerator provides
a shorter way for this process using a const_iterator. The first code
example then becomes this:
HKLGenerator generator(V3D(hmin, kmin, lmin), V3D(hmax, kmax, lmax));
for(auto hkl = generator.begin(); hkl != generator.end(); ++hkl) {
hkls.push_back(*hkl);
}
Or even shorter, using std::copy:
HKLGenerator generator(V3D(hmin, kmin, lmin), V3D(hmax, kmax, lmax));
std::copy(generator.begin(), generator.end(), std::back_inserter(hkls));
It's also possible to use filters this way, but in a much easier fashion,
using std::copy_remove_if (pre C++11) or std::copy_if (C++11):
// pre C++11
std::copy_remove_if(generator.begin(), generator.end(),
std::back_inserter(hkls), isNotOk)
// C++11
std::copy_if(generator.begin(), generator.end(), std::back_inserter(hkls),
isOk)
See the documentation for HKLFilter for more details on how to perform
actual filtering.
Please be aware that the iterator increments infinitely if it passes the
specified maximimum HKL. In that case K and L remain constant while H is
incremented (until it overflows).
@author Michael Wedel, ESS
@date 23/09/2015
Copyright © 2015 ISIS Rutherford Appleton Laboratory, NScD Oak Ridge
National Laboratory & European Spallation Source
This file is part of Mantid.
Mantid is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
Mantid is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
File change history is stored at: <https://github.com/mantidproject/mantid>
Code Documentation is available at: <http://doxygen.mantidproject.org>
*/
class MANTID_GEOMETRY_DLL HKLGenerator {
public:
/**
* @brief The const_iterator class
*
* This class uses boost::iterator_facade to provide a const_iterator
* interface to HKLGenerator. This makes it very easy to use the standard
* library algorithms. It's a forward iterator, so it's not possible to use
* operator[] for random access of specific HKLs (this would have to be
* done on the generated container if that supports it).
*
* While the iterators can be instantiated directly, the intention is to
* use HKLGenerator::begin() and HKLGenerator::end().
*/
class MANTID_GEOMETRY_DLL const_iterator
: public boost::iterator_facade<const_iterator, const Kernel::V3D &,
boost::forward_traversal_tag> {
public:
const_iterator();
explicit const_iterator(const Kernel::V3D ¤t);
explicit const_iterator(const Kernel::V3D &hklMin,
const Kernel::V3D &hklMax);
private:
// Required for boost::iterator_facade to work
friend class boost::iterator_core_access;
void increment();
/// Returns true if other is at the same position
inline bool equal(const const_iterator &other) const {
return this->m_h == other.m_h && this->m_k == other.m_k &&
this->m_l == other.m_l;
}
/// Returns a const reference to the currently pointed at HKL.
inline const Kernel::V3D &dereference() const { return m_hkl; }
/// Required for compilation in VS. Forward iterator can not be used
/// that way, so the implementation does nothing.
inline void advance(difference_type) {}
inline void decrement() {}
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
int m_h, m_k, m_l;
Kernel::V3D m_hkl;
int m_hMin, m_hMax;
int m_kMin, m_kMax;
int m_lMin, m_lMax;
};
HKLGenerator(const Kernel::V3D &hklMin, const Kernel::V3D &hklMax);
HKLGenerator(const Kernel::V3D &hklMinMax);
HKLGenerator(int hMinMax, int kMinMax, int lMinMax);
HKLGenerator(const UnitCell &unitCell, double dMin);
virtual ~HKLGenerator() {}
/// Returns the number of HKLs to be generated.
inline size_t size() const { return m_size; }
/// Returns an iterator to the beginning of the sequence.
inline const const_iterator &begin() const { return m_begin; }
/// Returns an iterator which "points at" one element past the end.
inline const const_iterator &end() const { return m_end; }
private:
size_t getSize(const Kernel::V3D &min, const Kernel::V3D &max) const;
const_iterator getBeginIterator() const;
const_iterator getEndIterator() const;
Kernel::V3D getEndHKL() const;
Kernel::V3D m_hklMin;
Kernel::V3D m_hklMax;
size_t m_size;
const_iterator m_begin;
const_iterator m_end;
};
} // namespace Geometry
} // namespace Mantid
#endif /* MANTID_GEOMETRY_HKLGENERATOR_H_ */