1 module nxt.combinations; 2 3 /** Given non-negative integers $(D m) and $(D n), generate all size $(D m) 4 combinations of the integers from 0 to $(D n)-1 in sorted order (each 5 combination is sorted and the entire table is sorted). 6 7 For example, 3 comb 5 is 8 0 1 2 9 0 1 3 10 0 1 4 11 0 2 3 12 0 2 4 13 0 3 4 14 1 2 3 15 1 2 4 16 1 3 4 17 2 3 4 18 19 See_Also: http://rosettacode.org/wiki/Combinations 20 */ 21 struct Combinations(T, bool copy = true, bool useArray = true) 22 { 23 import std.container: Array; 24 import std.traits: Unqual; 25 26 static if (useArray) 27 alias Indices = Array!size_t; 28 else 29 alias Indices = size_t[]; 30 31 Unqual!T[] pool, front; 32 size_t r, n; 33 bool empty = false; 34 35 Indices indices; 36 37 size_t len; 38 bool lenComputed = false; 39 40 this(T[] pool_, in size_t r_) // pure nothrow @safe 41 { 42 this.pool = pool_.dup; 43 this.r = r_; 44 this.n = pool.length; 45 if (r > n) 46 empty = true; 47 48 indices.length = r; 49 50 size_t i; 51 52 i = 0; 53 foreach (ref ini; indices[]) 54 ini = i++; 55 56 front.length = r; 57 58 i = 0; 59 foreach (immutable idx; indices[]) 60 front[i++] = pool[idx]; 61 } 62 63 @property size_t length() /*logic_const*/ // pure nothrow @nogc 64 { 65 static size_t binomial(size_t n, size_t k) // pure nothrow @safe @nogc 66 in 67 { 68 assert(n > 0, "binomial: n must be > 0."); 69 } 70 do 71 { 72 if (k < 0 || k > n) 73 return 0; 74 if (k > (n / 2)) 75 k = n - k; 76 size_t result = 1; 77 foreach (size_t d; 1 .. k + 1) { 78 result *= n; 79 n--; 80 result /= d; 81 } 82 return result; 83 } 84 85 if (!lenComputed) 86 { 87 // Set cache. 88 len = binomial(n, r); 89 lenComputed = true; 90 } 91 return len; 92 } 93 94 void popFront() // pure nothrow @safe 95 { 96 if (!empty) 97 { 98 bool broken = false; 99 size_t pos = 0; 100 foreach_reverse (immutable i; 0 .. r) 101 { 102 pos = i; 103 if (indices[i] != i + n - r) 104 { 105 broken = true; 106 break; 107 } 108 } 109 if (!broken) 110 { 111 empty = true; 112 return; 113 } 114 indices[pos]++; 115 foreach (immutable j; pos + 1 .. r) 116 indices[j] = indices[j - 1] + 1; 117 static if (copy) 118 front = new Unqual!T[front.length]; 119 120 size_t i = 0; 121 foreach (immutable idx; indices[]) 122 { 123 front[i] = pool[idx]; 124 i++; 125 } 126 } 127 } 128 } 129 130 auto combinations(bool copy = true, T, bool useArray = false)(T[] items, in size_t k) 131 in(items.length) 132 => Combinations!(T, copy, useArray)(items, k); 133 134 unittest 135 { 136 import std.algorithm: equal, map; 137 // assert(equal([1, 2, 3, 4].combinations!false(2), [[3, 4], [3, 4], [3, 4], [3, 4], [3, 4], [3, 4]])); 138 enum solution = [[1, 2], 139 [1, 3], 140 [1, 4], 141 [2, 3], 142 [2, 4], 143 [3, 4]]; 144 assert(equal([1, 2, 3, 4].combinations!true(2), solution)); 145 assert(equal([1, 2, 3, 4].combinations(2).map!(x => x), solution)); 146 } 147 148 import std.range.primitives : isInputRange; 149 150 /** All Unordered Element Pairs (2-Element Subsets) of a $(D Range). 151 152 TODO: Add template parameter to decide if .array should be used internally. 153 154 See_Also: http://forum.dlang.org/thread/iqkybajwdzcvdytakgvw@forum.dlang.org#post-vhufbwsqbssyqwfxxbuu:40forum.dlang.org 155 See_Also: https://issues.dlang.org/show_bug.cgi?id=6788 156 See_Also: https://issues.dlang.org/show_bug.cgi?id=7128 157 */ 158 auto pairwise(R)(R r) 159 if (isInputRange!R) 160 { 161 struct Pairwise(R) 162 { 163 import core.internal.traits : Unqual; 164 import std.traits : ForeachType; 165 import std.typecons: Tuple; 166 167 alias UR = Unqual!R; 168 alias E = ForeachType!UR; 169 alias Pair = Tuple!(E, E); 170 171 import std.range.primitives : isRandomAccessRange, hasLength; 172 import std.traits : isNarrowString; 173 174 static if (isRandomAccessRange!R && 175 hasLength!R && 176 !isNarrowString!R) 177 { 178 179 this(R r_) 180 { 181 this._input = r_; 182 j = 1; 183 } 184 @property bool empty() => j >= _input.length; 185 @property Pair front() => typeof(return)(_input[i], _input[j]); 186 void popFront() 187 { 188 if (j >= _input.length - 1) 189 { 190 i++; 191 j = i + 1; 192 } 193 else 194 j++; 195 } 196 private size_t i, j; 197 } 198 else // isInputRange!UR 199 { 200 import std.range : dropOne; 201 this(R r_) 202 { 203 this._input = r_; 204 i = r_; 205 if (!i.empty) 206 j = i.dropOne; 207 else 208 j = UR.init; 209 } 210 @property bool empty() => j.empty; 211 @property Pair front() => typeof(return)(i.front, j.front); 212 void popFront() 213 { 214 j.popFront(); 215 if (j.empty) 216 { 217 i.popFront(); 218 j = i.dropOne; 219 } 220 } 221 private UR i, j; // temporary copies of $(D _input) 222 } 223 224 private: 225 UR _input; 226 } 227 228 return Pairwise!R(r); 229 } 230 231 /// test RandomAccessRange input 232 unittest 233 { 234 import std.algorithm: equal, filter; 235 import std.typecons: Tuple; 236 237 assert((new int[0]).pairwise.empty); 238 assert([1].pairwise.empty); 239 240 alias T = Tuple!(int, int); 241 assert(equal([1, 2].pairwise, 242 [T(1, 2)])); 243 assert(equal([1, 2, 3].pairwise, 244 [T(1, 2), T(1, 3), T(2, 3)])); 245 assert(equal([1, 2, 3, 4].pairwise, 246 [T(1, 2), T(1, 3), T(1, 4), 247 T(2, 3), T(2, 4), T(3, 4)])); 248 } 249 250 /// test ForwardRange input 251 unittest 252 { 253 import std.algorithm: equal, filter; 254 import std.array : array; 255 256 auto p = [1].filter!"a < 4".pairwise; 257 assert(p.empty); 258 259 assert(equal(p.array, 260 [1].pairwise.array)); 261 262 assert(equal([1, 2, 3, 4].filter!"a < 4".pairwise, 263 [1, 2, 3].pairwise)); 264 }