二叉树路径和为一固定值

二叉树类题目,第一反应肯定就是使用递归(dfs)。这道题也不例外。

还是先给出我的代码。主要注意数组的引用性。所以每次都要去掉尾数。

并在最后放入结果集中时,执行深copy。

import java.util.ArrayList;
/**
public class TreeNode {
    int val = 0;
    TreeNode left = null;
    TreeNode right = null;

    public TreeNode(int val) {
        this.val = val;

    }

}
*/
public class Solution {
    public ArrayList<ArrayList<Integer>> FindPath(TreeNode root,int target) {
        ArrayList<ArrayList<Integer>> result = new ArrayList<>();
        ArrayList<Integer> array = new ArrayList<>();
        find(root,target,result,array);
        
        return result;
    }
    void find(TreeNode root,int target,ArrayList<ArrayList<Integer>> result,ArrayList<Integer> array){
        if(null == root){
           return;
        }
        array.add(root.val);
        target-=root.val;
        if(0 == target && null == root.left && null == root.right){
            result.add(new ArrayList<>(array)); //直接赋值是引用赋值,指向相同对象。array最终回到顶层就只有一个10。
            return;
        }
        
        if(null !=root.left){
            find(root.left,target,result,array);
            array.remove(array.size() -1);
        }
        if(null != root.right){
            find(root.right,target,result,array);
            array.remove(array.size() -1);
        }
        
    }
}

利用类变量的简化版:

import java.util.ArrayList;
/**
public class TreeNode {
    int val = 0;
    TreeNode left = null;
    TreeNode right = null;

    public TreeNode(int val) {
        this.val = val;

    }

}
*/
public class Solution {
    ArrayList<ArrayList<Integer>> result = new ArrayList<>();
    ArrayList<Integer> array = new ArrayList<>();
    public ArrayList<ArrayList<Integer>> FindPath(TreeNode root,int target) {
         if(null == root){
           return result;
        }
        array.add(root.val);
        target-=root.val;
        if(0 == target && null == root.left && null == root.right){
            result.add(new ArrayList<>(array)); //直接赋值是引用赋值,指向相同对象。array最终回到顶层就只有一个10。
        }
        FindPath(root.left,target);
        FindPath(root.right,target);
        array.remove(array.size() -1);
        return result;
    }
}

非递归法:注意牛客的编译器C++越界不会报错。但是java肯定会抛异常的。一定小心数组越界。

import java.util.ArrayList;
import java.util.Stack;
/**
public class TreeNode {
    int val = 0;
    TreeNode left = null;
    TreeNode right = null;

    public TreeNode(int val) {
        this.val = val;

    }

}
*/
public class Solution {
   
    
    public ArrayList<ArrayList<Integer>> FindPath(TreeNode root,int target) {
         ArrayList<ArrayList<Integer>> result = new ArrayList<>();
         if(null == root){
           return result;
        }
        ArrayList<Integer> array = new ArrayList<>();
        Stack<TreeNode> stack = new Stack<>();
        stack.push(root);
        TreeNode cur = root;
        TreeNode last = null;
        int sum=0;
        while(!stack.isEmpty()){
            if(null != cur){
                stack.push(cur);
                array.add(cur.val);
                sum += cur.val;
                if(target == sum && null == cur.left && null == cur.right){
                    result.add(new ArrayList<Integer>(array));
                }
                cur = cur.left;
            }else{
                TreeNode temp = stack.peek();
                if(null != temp.right && temp.right != last){
                    cur = temp.right;
                }else{
                    last = temp;
                    stack.pop();
                    if(array.size() >0){
                        array.remove(array.size() - 1);
                    }
                    sum -= temp.val;
                }
            }
            
        }
        return result;
    }
}

这道题可以用递归的方式来解决,具体思路如下: 1. 如果二叉树为空,直接返回0。 2. 分别计算左右子树的最大深度maxDepth左右子树中的最长路径长度,分别为leftPathrightPath。 3. 比较leftPath + rightPath当前最长路径长度maxLength,如果大于maxLength,则更新maxLength,并记录当前路径上的结点值。 4. 最终返回maxDepthmaxLength中的较大值作为当前子树的最大深度。 下面是对应的C语言代码实现: ```c #include <stdio.h> #include <stdlib.h> // 二叉树结点定义 struct TreeNode { int val; struct TreeNode *left; struct TreeNode *right; }; void findLongestPath(struct TreeNode* root, int* maxLength, int** path, int* pathIndex) { if (!root) { return; } int leftDepth = findLongestPath(root->left, maxLength, path, pathIndex); int rightDepth = findLongestPath(root->right, maxLength, path, pathIndex); int leftPath = 0; if (root->left && root->left->val == root->val) { leftPath = leftDepth + 1; } int rightPath = 0; if (root->right && root->right->val == root->val) { rightPath = rightDepth + 1; } int currentPath = leftPath + rightPath; if (currentPath > *maxLength) { *maxLength = currentPath; *pathIndex = 0; (*path)[(*pathIndex)++] = root->val; } else if (currentPath == *maxLength) { (*path)[(*pathIndex)++] = root->val; } return 1 + (leftDepth > rightDepth ? leftDepth : rightDepth); } int* longestUnivaluePath(struct TreeNode* root, int* returnSize) { int maxLength = 0; int* path = (int*)malloc(sizeof(int) * 1000); int pathIndex = 0; findLongestPath(root, &maxLength, &path, &pathIndex); *returnSize = pathIndex; return path; } int main() { // 构造二叉树 struct TreeNode* root = (struct TreeNode*)malloc(sizeof(struct TreeNode)); root->val = 1; root->left = (struct TreeNode*)malloc(sizeof(struct TreeNode)); root->left->val = 4; root->left->left = NULL; root->left->right = NULL; root->right = (struct TreeNode*)malloc(sizeof(struct TreeNode)); root->right->val = 5; root->right->left = (struct TreeNode*)malloc(sizeof(struct TreeNode)); root->right->left->val = 5; root->right->left->left = NULL; root->right->left->right = NULL; root->right->right = (struct TreeNode*)malloc(sizeof(struct TreeNode)); root->right->right->val = 5; root->right->right->left = NULL; root->right->right->right = NULL; int returnSize; int* path = longestUnivaluePath(root, &returnSize); printf("The longest path length is: %d\n", returnSize); printf("The values on the path are: "); for (int i = 0; i < returnSize; i++) { printf("%d ", path[i]); } printf("\n"); free(path); free(root->left); free(root->right->left); free(root->right->right); free(root); return 0; } ``` 注:此代码中的二叉树是固定的,实际使用时需要根据题目要求构造对应的二叉树
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值