PHP 一个树为另一棵树的子结构 [TO BE CONTINUED]

输入两棵二叉树A,B,判断B是不是A的子结构。(ps:我们约定空树不是任意一个树的子结构)

   

<?php
class TreeNode {
    private $val;
    private $left;
    private $right;

    public function __construct($val=null, $left = null, $right = null) {
        $this->val = $val;
        $this->left = $left;
        $this->right = $right;
    }

    /**
     * @param $x : val
     * @param $t TreeNode or: null
     */
    public function insert($x) {
        if ($this->val ===null) {
            $this->val = $x;
        }
        if ($x < $this->val) {
            $this->left = is_null($this->left) ? new self($x) :
                $this->left->insert($x);
        } else if ($x > $this->val) {
            $this->right = is_null($this->right) ? new self($x):
                $this->right->insert($x);
        }
        /* else x is in the tree already; we'll do nothing */
        return $this;
    }

    /**
     * 先序遍历
     * @param $callback
     */
    public function traversePre($callback) {
        if (is_null($this->val)) {
            return;
        }
        call_user_func($callback, $this->val);
        if (!is_null($this->left)) {
            $this->left->traversePre($callback);
        }
        if (!is_null($this->right)) {
            $this->right->traversePre($callback);
        }
    }

    /**
     * 中序遍历
     * @param $callback
     */
    public function traverseMid($callback) {
        if (is_null($this->val)) {
            return;
        }
        if (!is_null($this->left)) {
            $this->left->traverseMid($callback);
        }
        call_user_func($callback, $this->val);
        if (!is_null($this->right)) {
            $this->right->traverseMid($callback);
        }
    }

    /**
     * @return null
     */
    public function makeEmpty() {
        if ($this->left !== null) {
            $this->left->makeEmpty();
            unset($this->left);
        }
        if ($this->right !== null) {
            $this->right->makeEmpty();
            unset($this->right);
        }
        $this->val = null;
        return null;
    }

    /**
     * fixme: find(7) => find(11)?
     * @return null
     */
    public function find($x) {
        if ($this->val===null) {
            return null;
        }
        return ($x < $this->val) ? (is_null($this->left) ? null : $this->left->find($x)) :
            ($x > $this->val) ? (is_null($this->right) ? null: $this->right->find($x)) : $this;
    }

    /**
     * 输入两棵二叉树A,B,判断B是不是A的子结构。(ps:我们约定空树不是任意一个树的子结构)
     * @param $pRoot1
     * @param $pRoot2
     * @return bool
     */
    public static function HasSubtree($pRoot1, $pRoot2) {
        if ($pRoot2 === null || $pRoot1===null) {
            return false;
        }
        if ($pRoot1->val===$pRoot2->val) {
            return self::isSubTree($pRoot1, $pRoot2) ||
            self::isSubTree($pRoot1->left, $pRoot2) || self::isSubTree($pRoot1->right, $pRoot2);
        }
        return false;
    }

    private static function isSubTree($root1, $root2) {
        if ($root2===null) {return true;}
        if ($root1 === null) {return false;}
        if ($root1->val === $root2->val) {
            return self::isSubTree($root1->left, $root2->left) && self::isSubTree($root1->right, $root2->right);
        }
        return false;
    }

}


  

test:

/**
    8
  /  \
 6   10
/ \  / \
5  7 9 11
*/

$tree = new TreeNode();

$a = [8,6,10,5,7,9,11];
array_walk($a, function($value, $index, $tree) {
    $tree->insert($value);
}, $tree);

$tree->traversePre(function($elem) {echo $elem.',';});
echo PHP_EOL;
$tree->traverseMid(function($elem) {echo $elem.',';});
echo PHP_EOL;
var_dump($tree->find(7));
// $tree->makeEmpty();

  

  

posted @ 2018-07-24 22:08  zhanghui_ming  阅读(150)  评论(0编辑  收藏  举报