Blame view

vendor/sebastian/diff/src/LCS/TimeEfficientLongestCommonSubsequenceImplementation.php 2.22 KB
ad2e91f7   Mihail   move multyparser ...
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
  <?php
  /*
   * This file is part of the Diff package.
   *
   * (c) Sebastian Bergmann <sebastian@phpunit.de>
   *
   * For the full copyright and license information, please view the LICENSE
   * file that was distributed with this source code.
   */
  
  namespace SebastianBergmann\Diff\LCS;
  
  /**
   * Time-efficient implementation of longest common subsequence calculation.
   *
   * @package    Diff
   * @author     Sebastian Bergmann <sebastian@phpunit.de>
   * @author     Kore Nordmann <mail@kore-nordmann.de>
   * @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
   */
  class TimeEfficientImplementation 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)
      {
          $common     = array();
          $fromLength = count($from);
          $toLength   = count($to);
          $width      = $fromLength + 1;
          $matrix     = new \SplFixedArray($width * ($toLength + 1));
  
          for ($i = 0; $i <= $fromLength; ++$i) {
              $matrix[$i] = 0;
          }
  
          for ($j = 0; $j <= $toLength; ++$j) {
              $matrix[$j * $width] = 0;
          }
  
          for ($i = 1; $i <= $fromLength; ++$i) {
              for ($j = 1; $j <= $toLength; ++$j) {
                  $o = ($j * $width) + $i;
                  $matrix[$o] = max(
                      $matrix[$o - 1],
                      $matrix[$o - $width],
                      $from[$i - 1] === $to[$j - 1] ? $matrix[$o - $width - 1] + 1 : 0
                  );
              }
          }
  
          $i = $fromLength;
          $j = $toLength;
  
          while ($i > 0 && $j > 0) {
              if ($from[$i-1] === $to[$j-1]) {
                  $common[] = $from[$i-1];
                  --$i;
                  --$j;
              } else {
                  $o = ($j * $width) + $i;
                  if ($matrix[$o - $width] > $matrix[$o - 1]) {
                      --$j;
                  } else {
                      --$i;
                  }
              }
          }
  
          return array_reverse($common);
      }
  }