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 QueryBuilder 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 QueryBuilder, and based on these observations, apply Extract Interface, too.
| 1 | <?php | ||
| 18 | class QueryBuilder implements QueryInterface | ||
| 19 | { | ||
| 20 | /* @var string The modifier. SELECT, INSERT INTO, UPDATE or DELETE */ | ||
| 21 | private $modifier; | ||
| 22 | |||
| 23 | /* @var string The table name. */ | ||
| 24 | private $table; | ||
| 25 | |||
| 26 | /* @var array The columns. */ | ||
| 27 | private $columns; | ||
| 28 | |||
| 29 | /* @var array The values. */ | ||
| 30 | private $values; | ||
| 31 | |||
| 32 | /* @var array The join conditions. */ | ||
| 33 | private $join; | ||
| 34 | |||
| 35 | /* @var array The where conditions. */ | ||
| 36 | private $where; | ||
| 37 | |||
| 38 | /* @var array The group by conditions. */ | ||
| 39 | private $groupBy; | ||
| 40 | |||
| 41 | /* @var array The order by conditions. */ | ||
| 42 | private $orderBy; | ||
| 43 | |||
| 44 | /* @var string The limit. */ | ||
| 45 | private $limit; | ||
| 46 | |||
| 47 | /* @var string The offset. */ | ||
| 48 | private $offset; | ||
| 49 | |||
| 50 | /** | ||
| 51 | * Construct a query builder object with the given table. | ||
| 52 | * | ||
| 53 | * @param string $table | ||
| 54 | */ | ||
| 55 | 45 | public function __construct($table) | |
| 63 | |||
| 64 | /** | ||
| 65 | * Returns a string representation of the query object. | ||
| 66 | * | ||
| 67 | * @return string a string representation of the query object. | ||
| 68 | */ | ||
| 69 | 45 | public function __toString() | |
| 88 | |||
| 89 | /** | ||
| 90 | 	 * {@inheritdoc} | ||
| 91 | */ | ||
| 92 | 33 | public function select($columns = ['*']) | |
| 99 | |||
| 100 | /** | ||
| 101 | * Returns the select clause. | ||
| 102 | * | ||
| 103 | * @return string the select clause. | ||
| 104 | */ | ||
| 105 | 33 | private function getSelectClause() | |
| 109 | |||
| 110 | /** | ||
| 111 | * Returns the select query. | ||
| 112 | * | ||
| 113 | * @return string the select query. | ||
| 114 | */ | ||
| 115 | 33 | private function getSelectQuery() | |
| 145 | |||
| 146 | /** | ||
| 147 | 	 * {@inheritdoc} | ||
| 148 | */ | ||
| 149 | 3 | public function insert(array $values) | |
| 156 | |||
| 157 | /** | ||
| 158 | * Returns the insert clause. | ||
| 159 | * | ||
| 160 | * @return string the insert clause. | ||
| 161 | */ | ||
| 162 | 3 | private function getInsertClause() | |
| 174 | |||
| 175 | /** | ||
| 176 | * Returns the insert query. | ||
| 177 | * | ||
| 178 | * @return string the insert query. | ||
| 179 | */ | ||
| 180 | 3 | private function getInsertQuery() | |
| 184 | |||
| 185 | /** | ||
| 186 | 	 * {@inheritdoc} | ||
| 187 | */ | ||
| 188 | 4 | public function update(array $values) | |
| 195 | |||
| 196 | /** | ||
| 197 | * Returns the update clause. | ||
| 198 | * | ||
| 199 | * @return string the update clause. | ||
| 200 | */ | ||
| 201 | 4 | private function getUpdateClause() | |
| 211 | |||
| 212 | /** | ||
| 213 | * Returns the update query. | ||
| 214 | * | ||
| 215 | * @return string the update query. | ||
| 216 | */ | ||
| 217 | 4 | View Code Duplication | private function getUpdateQuery() | 
| 231 | |||
| 232 | /** | ||
| 233 | 	 * {@inheritdoc} | ||
| 234 | */ | ||
| 235 | 3 | public function delete() | |
| 241 | |||
| 242 | /** | ||
| 243 | * Returns the delete clause. | ||
| 244 | * | ||
| 245 | * @return string the delete clause. | ||
| 246 | */ | ||
| 247 | 3 | private function getDeleteClause() | |
| 251 | |||
| 252 | /** | ||
| 253 | * Returns the delete query. | ||
| 254 | * | ||
| 255 | * @return string the delete query. | ||
| 256 | */ | ||
| 257 | 3 | View Code Duplication | private function getDeleteQuery() | 
| 272 | |||
| 273 | /** | ||
| 274 | 	 * {@inheritdoc} | ||
| 275 | */ | ||
| 276 | 2 | public function join($table, $primary, $operator, $secondary) | |
| 282 | |||
| 283 | /** | ||
| 284 | 	 * {@inheritdoc} | ||
| 285 | */ | ||
| 286 | 2 | public function leftJoin($table, $primary, $operator, $secondary) | |
| 292 | |||
| 293 | /** | ||
| 294 | 	 * {@inheritdoc} | ||
| 295 | */ | ||
| 296 | 2 | public function rightJoin($table, $primary, $operator, $secondary) | |
| 302 | |||
| 303 | /** | ||
| 304 | 	 * {@inheritdoc} | ||
| 305 | */ | ||
| 306 | 2 | public function crossJoin($table, $primary, $operator, $secondary) | |
| 312 | |||
| 313 | 33 | private function getJoinClause() | |
| 323 | |||
| 324 | /** | ||
| 325 | 	 * {@inheritdoc} | ||
| 326 | */ | ||
| 327 | 15 | public function where($column, $operator, $value) | |
| 333 | |||
| 334 | /** | ||
| 335 | * Returns the where clause. | ||
| 336 | * | ||
| 337 | * @return string the where clause. | ||
| 338 | */ | ||
| 339 | 40 | private function getWhereClause() | |
| 353 | |||
| 354 | /** | ||
| 355 | * Returns the where condition. | ||
| 356 | * | ||
| 357 | * @param string $column | ||
| 358 | * @param string $operator | ||
| 359 | * @param mixed $value | ||
| 360 | * @return string the where condition. | ||
| 361 | */ | ||
| 362 | 15 | private function getWhereCondition($column, $operator, $value) | |
| 370 | |||
| 371 | /** | ||
| 372 | 	 * {@inheritdoc} | ||
| 373 | */ | ||
| 374 | 2 | public function groupBy($column) | |
| 380 | |||
| 381 | /** | ||
| 382 | * Returns the group by clause. | ||
| 383 | * | ||
| 384 | * @return string the group by clause. | ||
| 385 | */ | ||
| 386 | 33 | private function getGroupByClause() | |
| 394 | |||
| 395 | /** | ||
| 396 | 	 * {@inheritdoc} | ||
| 397 | */ | ||
| 398 | 4 | public function orderBy($column, $order = null) | |
| 410 | |||
| 411 | /** | ||
| 412 | * Returns the order by clause. | ||
| 413 | * | ||
| 414 | * @return string the order by clause. | ||
| 415 | */ | ||
| 416 | 33 | private function getOrderByClause() | |
| 424 | |||
| 425 | /** | ||
| 426 | 	 * {@inheritdoc} | ||
| 427 | */ | ||
| 428 | 6 | public function limit($limit) | |
| 434 | |||
| 435 | /** | ||
| 436 | * Returns the limit clause. | ||
| 437 | * | ||
| 438 | * @return string the limit clause. | ||
| 439 | */ | ||
| 440 | 40 | private function getLimitClause() | |
| 448 | |||
| 449 | /** | ||
| 450 | 	 * {@inheritdoc} | ||
| 451 | */ | ||
| 452 | 2 | public function offset($offset) | |
| 458 | |||
| 459 | /** | ||
| 460 | * Returns the offset clause. | ||
| 461 | * | ||
| 462 | * @return string the offset clause. | ||
| 463 | */ | ||
| 464 | 33 | private function getOffsetClause() | |
| 472 | } | ||
| 473 | 
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.