Duplicate code is one of the most pungent code smells. A rule that is often used is to re-structure code once it is duplicated in three or more places.
Common duplication problems, and corresponding solutions are:
Complex classes like Map often do a lot of different things. To break such a class down, we need to identify a cohesive component within that class. A common approach to find such a component is to look for fields/methods that share the same prefixes, or suffixes. You can also have a look at the cohesion graph to spot any un-connected, or weakly-connected components.
Once you have determined the fields that belong together, you can apply the Extract Class refactoring. If the component makes sense as a sub-class, Extract Subclass is also a candidate, and is often faster.
While breaking up the class, it is a good idea to analyze how other classes use Map, and based on these observations, apply Extract Interface, too.
| 1 | <?php | ||
| 14 | final class Map implements \IteratorAggregate, \ArrayAccess, Collection | ||
| 15 | { | ||
| 16 | use Traits\Collection; | ||
| 17 | use Traits\SquaredCapacity; | ||
| 18 | |||
| 19 | const MIN_CAPACITY = 8; | ||
| 20 | |||
| 21 | /** | ||
| 22 | * @var Pair[] | ||
| 23 | */ | ||
| 24 | private $pairs; | ||
| 25 | |||
| 26 | /** | ||
| 27 | * Creates an instance using the values of an array or Traversable object. | ||
| 28 | * | ||
| 29 | * Should an integer be provided the Map will allocate the memory capacity | ||
| 30 | * to the size of $values. | ||
| 31 | * | ||
| 32 | * @param array|\Traversable|int|null $values | ||
| 33 | */ | ||
| 34 | View Code Duplication | public function __construct($values = null) | |
|  | |||
| 35 |     { | ||
| 36 | $this->reset(); | ||
| 37 | |||
| 38 |         if (is_array($values) || $values instanceof Traversable) { | ||
| 39 | $this->putAll($values); | ||
| 40 | |||
| 41 |         } else if (is_integer($values)) { | ||
| 42 | $this->allocate($values); | ||
| 43 | } | ||
| 44 | } | ||
| 45 | |||
| 46 | private function reset() | ||
| 47 |     { | ||
| 48 | $this->pairs = []; | ||
| 49 | $this->capacity = self::MIN_CAPACITY; | ||
| 50 | } | ||
| 51 | |||
| 52 | /** | ||
| 53 | * @inheritDoc | ||
| 54 | */ | ||
| 55 | public function clear() | ||
| 56 |     { | ||
| 57 | $this->reset(); | ||
| 58 | } | ||
| 59 | |||
| 60 | /** | ||
| 61 | * Removes all Pairs from the Map | ||
| 62 | * | ||
| 63 | * @param mixed[] $keys | ||
| 64 | */ | ||
| 65 | public function removeAll($keys) | ||
| 66 |     { | ||
| 67 |         foreach ($keys as $key) { | ||
| 68 | $this->remove($key); | ||
| 69 | } | ||
| 70 | } | ||
| 71 | |||
| 72 | /** | ||
| 73 | * Return the first Pair from the Map | ||
| 74 | * | ||
| 75 | * @return Pair | ||
| 76 | * | ||
| 77 | * @throws UnderflowException | ||
| 78 | */ | ||
| 79 | public function first(): Pair | ||
| 80 |     { | ||
| 81 |         if ($this->isEmpty()) { | ||
| 82 | throw new UnderflowException(); | ||
| 83 | } | ||
| 84 | |||
| 85 | return $this->pairs[0]; | ||
| 86 | } | ||
| 87 | |||
| 88 | /** | ||
| 89 | * Return the last Pair from the Map | ||
| 90 | * | ||
| 91 | * @return Pair | ||
| 92 | * | ||
| 93 | * @throws UnderflowException | ||
| 94 | */ | ||
| 95 | public function last(): Pair | ||
| 96 |     { | ||
| 97 |         if ($this->isEmpty()) { | ||
| 98 | throw new UnderflowException(); | ||
| 99 | } | ||
| 100 | |||
| 101 | return end($this->pairs); | ||
| 102 | } | ||
| 103 | |||
| 104 | /** | ||
| 105 | * Return the pair at a specified position in the Map | ||
| 106 | * | ||
| 107 | * @param int $position | ||
| 108 | * | ||
| 109 | * @return Pair | ||
| 110 | * | ||
| 111 | * @throws OutOfRangeException | ||
| 112 | */ | ||
| 113 | public function skip(int $position): Pair | ||
| 114 |     { | ||
| 115 |         if ($position < 0 || $position >= count($this->pairs)) { | ||
| 116 | throw new OutOfRangeException(); | ||
| 117 | } | ||
| 118 | |||
| 119 | return $this->pairs[$position]->copy(); | ||
| 120 | } | ||
| 121 | |||
| 122 | /** | ||
| 123 | * Merge an array of values with the current Map | ||
| 124 | * | ||
| 125 | * @param array|\Traversable $values | ||
| 126 | * | ||
| 127 | * @return Map | ||
| 128 | */ | ||
| 129 | public function merge($values): Map | ||
| 130 |     { | ||
| 131 | $merged = $this->copy(); | ||
| 132 | $merged->putAll($values); | ||
| 133 | |||
| 134 | return $merged; | ||
| 135 | } | ||
| 136 | |||
| 137 | /** | ||
| 138 | * Intersect | ||
| 139 | * | ||
| 140 | * @param Map $map | ||
| 141 | * | ||
| 142 | * @return Map | ||
| 143 | */ | ||
| 144 | public function intersect(Map $map): Map | ||
| 145 |     { | ||
| 146 |         return $this->filter(function($key) use ($map) { | ||
| 147 | return $map->containsKey($key); | ||
| 148 | }); | ||
| 149 | } | ||
| 150 | |||
| 151 | /** | ||
| 152 | * Diff | ||
| 153 | * | ||
| 154 | * @param Map $map | ||
| 155 | * | ||
| 156 | * @return Map | ||
| 157 | */ | ||
| 158 | public function diff(Map $map): Map | ||
| 159 |     { | ||
| 160 |         return $this->filter(function($key) use ($map) { | ||
| 161 | return ! $map->containsKey($key); | ||
| 162 | }); | ||
| 163 | } | ||
| 164 | |||
| 165 | /** | ||
| 166 | * XOR | ||
| 167 | * | ||
| 168 | * @param Map $map | ||
| 169 | * | ||
| 170 | * @return Map | ||
| 171 | */ | ||
| 172 | public function xor(Map $map): Map | ||
| 173 |     { | ||
| 174 |         return $this->merge($map)->filter(function($key) use ($map) { | ||
| 175 | return $this->containsKey($key) ^ $map->containsKey($key); | ||
| 176 | }); | ||
| 177 | } | ||
| 178 | |||
| 179 | /** | ||
| 180 | * Identical | ||
| 181 | * | ||
| 182 | * @param mixed $a | ||
| 183 | * @param mixed $b | ||
| 184 | * | ||
| 185 | * @return bool | ||
| 186 | */ | ||
| 187 | private function keysAreEqual($a, $b): bool | ||
| 188 |     { | ||
| 189 |         if (is_object($a) && $a instanceof Hashable) { | ||
| 190 | return $a->equals($b); | ||
| 191 | } | ||
| 192 | |||
| 193 | return $a === $b; | ||
| 194 | } | ||
| 195 | |||
| 196 | /** | ||
| 197 | * @param $key | ||
| 198 | * | ||
| 199 | * @return Pair|null | ||
| 200 | */ | ||
| 201 | private function lookupKey($key) | ||
| 202 |     { | ||
| 203 |         foreach ($this->pairs as $pair) { | ||
| 204 |             if ($this->keysAreEqual($pair->key, $key)) { | ||
| 205 | return $pair; | ||
| 206 | } | ||
| 207 | } | ||
| 208 | } | ||
| 209 | |||
| 210 | /** | ||
| 211 | * @param $value | ||
| 212 | * | ||
| 213 | * @return Pair|null | ||
| 214 | */ | ||
| 215 | private function lookupValue($value) | ||
| 216 |     { | ||
| 217 |         foreach ($this->pairs as $pair) { | ||
| 218 |             if ($pair->value === $value) { | ||
| 219 | return $pair; | ||
| 220 | } | ||
| 221 | } | ||
| 222 | } | ||
| 223 | |||
| 224 | /** | ||
| 225 | * | ||
| 226 | */ | ||
| 227 | View Code Duplication | private function contains(string $lookup, array $values): bool | |
| 228 |     { | ||
| 229 |         if (empty($values)) { | ||
| 230 | return false; | ||
| 231 | } | ||
| 232 | |||
| 233 |         foreach ($values as $value) { | ||
| 234 |             if ( ! $this->$lookup($value)) { | ||
| 235 | return false; | ||
| 236 | } | ||
| 237 | } | ||
| 238 | |||
| 239 | return true; | ||
| 240 | } | ||
| 241 | |||
| 242 | /** | ||
| 243 | * Returns whether an association for all of zero or more keys exist. | ||
| 244 | * | ||
| 245 | * @param mixed ...$keys | ||
| 246 | * | ||
| 247 | * @return bool true if at least one value was provided and the map | ||
| 248 | * contains all given keys, false otherwise. | ||
| 249 | */ | ||
| 250 | public function containsKey(...$keys): bool | ||
| 251 |     { | ||
| 252 |         return $this->contains('lookupKey', $keys); | ||
| 253 | } | ||
| 254 | |||
| 255 | /** | ||
| 256 | * Returns whether an association for all of zero or more values exist. | ||
| 257 | * | ||
| 258 | * @param mixed ...$values | ||
| 259 | * | ||
| 260 | * @return bool true if at least one value was provided and the map | ||
| 261 | * contains all given values, false otherwise. | ||
| 262 | */ | ||
| 263 | public function containsValue(...$values): bool | ||
| 264 |     { | ||
| 265 |         return $this->contains('lookupValue', $values); | ||
| 266 | } | ||
| 267 | |||
| 268 | /** | ||
| 269 | * @inheritDoc | ||
| 270 | */ | ||
| 271 | public function copy() | ||
| 272 |     { | ||
| 273 | return new self($this); | ||
| 274 | } | ||
| 275 | |||
| 276 | /** | ||
| 277 | * @inheritDoc | ||
| 278 | */ | ||
| 279 | public function count(): int | ||
| 280 |     { | ||
| 281 | return count($this->pairs); | ||
| 282 | } | ||
| 283 | |||
| 284 | /** | ||
| 285 | * Returns a new map containing only the values for which a predicate | ||
| 286 | * returns true. A boolean test will be used if a predicate is not provided. | ||
| 287 | * | ||
| 288 | * @param callable|null $predicate Accepts a key and a value, and returns: | ||
| 289 | * true : include the value, | ||
| 290 | * false: skip the value. | ||
| 291 | * | ||
| 292 | * @return Map | ||
| 293 | */ | ||
| 294 | View Code Duplication | public function filter(callable $predicate = null): Map | |
| 295 |     { | ||
| 296 | $filtered = new self(); | ||
| 297 | |||
| 298 |         foreach ($this as $key => $value) { | ||
| 299 |             if ($predicate ? $predicate($key, $value) : $value) { | ||
| 300 | $filtered->put($key, $value); | ||
| 301 | } | ||
| 302 | } | ||
| 303 | |||
| 304 | return $filtered; | ||
| 305 | } | ||
| 306 | |||
| 307 | /** | ||
| 308 | * Returns the value associated with a key, or an optional default if the | ||
| 309 | * key is not associated with a value. | ||
| 310 | * | ||
| 311 | * @param mixed $key | ||
| 312 | * @param mixed $default | ||
| 313 | * | ||
| 314 | * @return mixed The associated value or fallback default if provided. | ||
| 315 | * | ||
| 316 | * @throws OutOfBoundsException if no default was provided and the key is | ||
| 317 | * not associated with a value. | ||
| 318 | */ | ||
| 319 | public function get($key, $default = null) | ||
| 331 | |||
| 332 | /** | ||
| 333 | * Returns a set of all the keys in the map. | ||
| 334 | * | ||
| 335 | * @return Set | ||
| 336 | */ | ||
| 337 | public function keys(): Set | ||
| 347 | |||
| 348 | /** | ||
| 349 | * Returns a new map using the results of applying a callback to each value. | ||
| 350 | * The keys will be keysAreEqual in both maps. | ||
| 351 | * | ||
| 352 | * @param callable $callback Accepts two arguments: key and value, should | ||
| 353 | * return what the updated value will be. | ||
| 354 | * | ||
| 355 | * @return Map | ||
| 356 | */ | ||
| 357 | public function map(callable $callback): Map | ||
| 367 | |||
| 368 | /** | ||
| 369 | * Returns a sequence of pairs representing all associations. | ||
| 370 | * | ||
| 371 | * @return Sequence | ||
| 372 | */ | ||
| 373 | public function pairs(): Sequence | ||
| 374 |     { | ||
| 375 | $sequence = new Vector(); | ||
| 376 | |||
| 377 |         foreach ($this->pairs as $pair) { | ||
| 378 | $sequence[] = $pair->copy(); | ||
| 379 | } | ||
| 380 | |||
| 381 | return $sequence; | ||
| 382 | } | ||
| 383 | |||
| 384 | /** | ||
| 385 | * Associates a key with a value, replacing a previous association if there | ||
| 386 | * was one. | ||
| 387 | * | ||
| 388 | * @param mixed $key | ||
| 389 | * @param mixed $value | ||
| 390 | */ | ||
| 391 | public function put($key, $value) | ||
| 403 | |||
| 404 | /** | ||
| 405 | * Creates associations for all keys and corresponding values of either an | ||
| 406 | * array or iterable object. | ||
| 407 | * | ||
| 408 | * @param array|\Traversable $values | ||
| 409 | */ | ||
| 410 | public function putAll($values) | ||
| 416 | |||
| 417 | /** | ||
| 418 | * Iteratively reduces the map to a single value using a callback. | ||
| 419 | * | ||
| 420 | * @param callable $callback Accepts the carry, key, and value, and | ||
| 421 | * returns an updated carry value. | ||
| 422 | * | ||
| 423 | * @param mixed|null $initial Optional initial carry value. | ||
| 424 | * | ||
| 425 | * @return mixed The carry value of the final iteration, or the initial | ||
| 426 | * value if the map was empty. | ||
| 427 | */ | ||
| 428 | public function reduce(callable $callback, $initial = null) | ||
| 438 | |||
| 439 | private function delete(int $position) | ||
| 440 |     { | ||
| 441 | $pair = $this->pairs[$position]; | ||
| 442 | $value = $pair->value; | ||
| 443 | |||
| 444 | array_splice($this->pairs, $position, 1, null); | ||
| 445 | |||
| 446 | $this->adjustCapacity(); | ||
| 447 | return $value; | ||
| 448 | } | ||
| 449 | |||
| 450 | /** | ||
| 451 | * Removes a key's association from the map and returns the associated value | ||
| 452 | * or a provided default if provided. | ||
| 453 | * | ||
| 454 | * @param mixed $key | ||
| 455 | * @param mixed $default | ||
| 456 | * | ||
| 457 | * @return mixed The associated value or fallback default if provided. | ||
| 458 | * | ||
| 459 | * @throws \OutOfBoundsException if no default was provided and the key is | ||
| 460 | * not associated with a value. | ||
| 461 | */ | ||
| 462 | public function remove($key, $default = null) | ||
| 477 | |||
| 478 | /** | ||
| 479 | * Returns a reversed copy of the map. | ||
| 480 | */ | ||
| 481 | public function reverse(): Map | ||
| 488 | |||
| 489 | /** | ||
| 490 | * Returns a sub-sequence of a given length starting at a specified offset. | ||
| 491 | * | ||
| 492 | * @param int $offset If the offset is non-negative, the map will | ||
| 493 | * start at that offset in the map. If offset is | ||
| 494 | * negative, the map will start that far from the | ||
| 495 | * end. | ||
| 496 | * | ||
| 497 | * @param int|null $length If a length is given and is positive, the | ||
| 498 | * resulting set will have up to that many pairs in | ||
| 499 | * it. If the requested length results in an | ||
| 500 | * overflow, only pairs up to the end of the map | ||
| 501 | * will be included. | ||
| 502 | * | ||
| 503 | * If a length is given and is negative, the map | ||
| 504 | * will stop that many pairs from the end. | ||
| 505 | * | ||
| 506 | * If a length is not provided, the resulting map | ||
| 507 | * will contains all pairs between the offset and | ||
| 508 | * the end of the map. | ||
| 509 | * | ||
| 510 | * @return Map | ||
| 511 | */ | ||
| 512 | public function slice(int $offset, int $length = null): Map | ||
| 528 | |||
| 529 | /** | ||
| 530 | * Returns a sorted copy of the map, based on an optional callable | ||
| 531 | * comparator. The map will be sorted by key if a comparator is not given. | ||
| 532 | * | ||
| 533 | * @param callable|null $comparator Accepts two values to be compared. | ||
| 534 | * Should return the result of a <=> b. | ||
| 535 | * | ||
| 536 | * @return Map | ||
| 537 | */ | ||
| 538 | public function sort(callable $comparator = null): Map | ||
| 551 | |||
| 552 | /** | ||
| 553 | * @inheritDoc | ||
| 554 | */ | ||
| 555 | public function toArray(): array | ||
| 565 | |||
| 566 | /** | ||
| 567 | * Returns a sequence of all the associated values in the Map. | ||
| 568 | * | ||
| 569 | * @return Sequence | ||
| 570 | */ | ||
| 571 | public function values(): Sequence | ||
| 581 | |||
| 582 | /** | ||
| 583 | * Get iterator | ||
| 584 | */ | ||
| 585 | public function getIterator() | ||
| 591 | |||
| 592 | /** | ||
| 593 | * Debug Info | ||
| 594 | */ | ||
| 595 | public function __debugInfo() | ||
| 599 | |||
| 600 | /** | ||
| 601 | * @inheritdoc | ||
| 602 | */ | ||
| 603 | public function offsetSet($offset, $value) | ||
| 607 | |||
| 608 | /** | ||
| 609 | * @inheritdoc | ||
| 610 | * | ||
| 611 | * @throws OutOfBoundsException | ||
| 612 | */ | ||
| 613 | public function &offsetGet($offset) | ||
| 623 | |||
| 624 | /** | ||
| 625 | * @inheritdoc | ||
| 626 | */ | ||
| 627 | public function offsetUnset($offset) | ||
| 631 | |||
| 632 | /** | ||
| 633 | * @inheritdoc | ||
| 634 | */ | ||
| 635 | public function offsetExists($offset) | ||
| 639 | } | ||
| 640 | 
Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.
You can also find more detailed suggestions in the “Code” section of your repository.