You are here

class MemoryEfficientImplementation in Zircon Profile 8

Same name and namespace in other branches
  1. 8.0 vendor/sebastian/diff/src/LCS/MemoryEfficientLongestCommonSubsequenceImplementation.php \SebastianBergmann\Diff\LCS\MemoryEfficientImplementation

Memory-efficient implementation of longest common subsequence calculation.

@package Diff @author Sebastian Bergmann <sebastian@phpunit.de> @author Denes Lados <lados.denes@gmail.com> @copyright Sebastian Bergmann <sebastian@phpunit.de> @license http://www.opensource.org/licenses/BSD-3-Clause The BSD 3-Clause License @link http://www.github.com/sebastianbergmann/diff

Hierarchy

Expanded class hierarchy of MemoryEfficientImplementation

2 files declare their use of MemoryEfficientImplementation
Differ.php in vendor/sebastian/diff/src/Differ.php
DifferTest.php in vendor/sebastian/diff/tests/DifferTest.php

File

vendor/sebastian/diff/src/LCS/MemoryEfficientLongestCommonSubsequenceImplementation.php, line 23

Namespace

SebastianBergmann\Diff\LCS
View source
class MemoryEfficientImplementation implements LongestCommonSubsequence {

  /**
   * Calculates the longest common subsequence of two arrays.
   *
   * @param  array $from
   * @param  array $to
   * @return array
   */
  public function calculate(array $from, array $to) {
    $cFrom = count($from);
    $cTo = count($to);
    if ($cFrom == 0) {
      return array();
    }
    elseif ($cFrom == 1) {
      if (in_array($from[0], $to)) {
        return array(
          $from[0],
        );
      }
      else {
        return array();
      }
    }
    else {
      $i = intval($cFrom / 2);
      $fromStart = array_slice($from, 0, $i);
      $fromEnd = array_slice($from, $i);
      $llB = $this
        ->length($fromStart, $to);
      $llE = $this
        ->length(array_reverse($fromEnd), array_reverse($to));
      $jMax = 0;
      $max = 0;
      for ($j = 0; $j <= $cTo; $j++) {
        $m = $llB[$j] + $llE[$cTo - $j];
        if ($m >= $max) {
          $max = $m;
          $jMax = $j;
        }
      }
      $toStart = array_slice($to, 0, $jMax);
      $toEnd = array_slice($to, $jMax);
      return array_merge($this
        ->calculate($fromStart, $toStart), $this
        ->calculate($fromEnd, $toEnd));
    }
  }

  /**
   * @param array $from
   * @param array $to
   * @return array
   */
  private function length(array $from, array $to) {
    $current = array_fill(0, count($to) + 1, 0);
    $cFrom = count($from);
    $cTo = count($to);
    for ($i = 0; $i < $cFrom; $i++) {
      $prev = $current;
      for ($j = 0; $j < $cTo; $j++) {
        if ($from[$i] == $to[$j]) {
          $current[$j + 1] = $prev[$j] + 1;
        }
        else {
          $current[$j + 1] = max($current[$j], $prev[$j + 1]);
        }
      }
    }
    return $current;
  }

}

Members

Namesort descending Modifiers Type Description Overrides
MemoryEfficientImplementation::calculate public function Calculates the longest common subsequence of two arrays. Overrides LongestCommonSubsequence::calculate
MemoryEfficientImplementation::length private function