算法基础学习3

一、图

基本模板

Graph

public class Graph {
    public HashMap nodes;
    public HashSet edges;

    public Graph() {
        nodes = new HashMap();
        edges = new HashSet<>();
    }
}

Edge

public class Edge {
    public int weight;
    public Node from;
    public Node to;

    public Edge(int weight, Node from, Node to) {
        this.weight = weight;
        this.from = from;
        this.to = to;
    }
}

Node

public class Node {
    public int value;
    public int in;
    public int out;
    public ArrayList nexts;
    public ArrayList edges;

    public Node(int value) {
        this.value = value;
        in = 0;
        out = 0;
        nexts = new ArrayList<>();
        edges = new ArrayList<>();
    }
}

遍历

广度优先

/**
 * 图的广度优先遍历
 * 使用队列,为了保证不重复遍历,所以使用了set进行重复判断
 */
public static void BFS(Node node){
    Queue queue = new LinkedList<>();
    HashSet set = new HashSet<>();
    queue.add(node);
    set.add(node);
    while (! queue.isEmpty()){
        //弹出打印
        Node poll = queue.poll();
        System.out.print(poll.value+" ");
        //获取节点的next点
        ArrayList nexts = poll.nexts;
        for (Node next: nexts){
            //是否在set中
            if (! set.contains(next)){
                set.add(next);
                queue.add(next);
            }
        }
    }

}

深度优先

/**
 * 图的深度优先遍历
 * 准备栈和set集合
 * 在添加元素进入set时,就处理打印
 * 步骤L
 * 依次弹出栈一直到空
 *  获取node的next,遍历next,判断next是否在set中
 *      1.在set中,继续判断下一个next
 *      2.不在set中,将本node和该next压入栈,然后将next加入到set,再处理打印set
 *
 */
public static void DFS(Node node){
    Stack stack = new Stack<>();
    HashSet set = new HashSet<>();
    stack.add(node);
    set.add(node);
    System.out.print(node.value+" ");
    while (!set.isEmpty()){
        Node pop = stack.pop();
        ArrayList nexts = pop.nexts;
        for (Node next: nexts){
            //不在set中
            if (!set.contains(next)){
                //将原node和该next压入栈
                stack.push(pop);
                stack.push(next);
                set.add(next);
                System.out.print(next.value+" ");
                //不在处理后面的next节点,退出进行深度遍历
                break;
            }
        }

    }

}

拓扑排序

/**
 * 拓扑排序
 * 准备HashMap容器,记录每个点的入度
 * 准备queue容器用来实现遍历
 * 将所有node遍历,将每个点的入度对应记录在map中
 *      此时然后将入度为零的点加入到queue中,
 * 遍历从queue开始,遍历到的点,那么该点的nexts中的next点入度--,更新map
 *      更新后,如果该点入度为0,加入到queue中
 * 循环至queue为空
 */
public static List TopologySort(Graph graph){
    List result = new ArrayList<>();
    HashMap inMap = new HashMap<>();
    Queue zeroInQueue = new LinkedList<>();
    //node遍历,将每个点的入度对应记录在map中
    for (Node node: graph.nodes.values()){
        inMap.put(node,node.in);
        if (node.in == 0){
            zeroInQueue.add(node);
        }
    }
    //遍历queue
    while (!zeroInQueue.isEmpty()){
        Node poll = zeroInQueue.poll();
        result.add(poll);
        //该点的nexts中的next点入度--,更新map
        for (Node next: poll.nexts){
            //入度更新
            Integer in = inMap.get(next);
            inMap.put(next,in--);
            //入度为0,加入到queue中
            if (in == 0){
                zeroInQueue.add(next);
            }
        }
    }
    return result;
}

Kruskal最小生成树

public class Kruskal {
    public static class MySets{
        //每个节点的所属集合
        private HashMap> setMap;

        //构造
        public MySets(List nodes){
            for (Node node: nodes){
                //每个节点初始所属集合中只有自己
                List list = new ArrayList<>();
                list.add(node);
                //为属性赋值
                setMap.put(node,list);
            }
        }
        //接口:判断是否在同一集合中
        public boolean isSameSet(Node from,Node to){
            List fromList = setMap.get(from);
            List toList = setMap.get(to);
            return fromList == toList;
        }
        //接口:将集合合并
        public void unionSet(Node from,Node to){
            List fromList = setMap.get(from);
            List toList = setMap.get(to);
            //将toList中的节点加入到from集合中,
            //并且! 更改to集合节点所属
            for (Node node: toList){
                fromList.add(node);
                setMap.put(node,fromList);
            }
        }

    }

    /**
     * kruskal算法最小生成树
     * 接口:1.是否处于一个集合
     *      2.将集合合并
     *
     * 按照边的权值从小到大开始连接图,查看时候会出现环(也就是对应代码中两个节点的所属集合是同一个)
     * 如果在连接后出现环,则不连接
     * 如果不出现环,则连接,并更新from节点的所属集合(将集合合并,用于判断环),将节点添加到结果集中
     */
    public static Set kruskalMST(Graph graph){
        List list = (List) graph.nodes.values();
        //初始化mysets
        MySets mySets = new MySets(list);
        //按照边的权值大小,加入到queue中并排序
        PriorityQueue queue = new PriorityQueue<>(Comparator.comparingInt(x -> x.weight));
        queue.addAll(graph.edges);
        //结果集
        Set result = new HashSet<>();

        while (! queue.isEmpty()){
            Edge edge = queue.poll();
            //不在一个集合中
            if (!mySets.isSameSet(edge.from, edge.to)){
                //则连接,并更新from节点的所属集合
                result.add(edge);
                mySets.unionSet(edge.from,edge.to);
            }
        }
        return result;
    }
}

Prim

/**
 * 准备queue用来遍历
 * 准备set用来判断是否已经加入过
 * 先将所有边从小到大加入到queue中
 * 循环:从queue中取出边,判断该边的目的点是否已经在set中(k算法在此步是判断是否会成环)
 *      如果不在set中,则说明没有,加入到set中,并且将该点的edges加入到queue中,此时会产生重复,但set会判断
 *      如果在set中,弹出下一个,继续循环
 */
public static Set primMST(Graph graph) {
    PriorityQueue priorityQueue = new PriorityQueue<>(Comparator.comparingInt(x -> x.weight));
    //判断旧值是否添加过
    HashSet set = new HashSet<>();
    Set result = new HashSet<>();
    //循环是防止图为森林,可以不加
    for (Node node : graph.nodes.values()) {
        if (!set.contains(node)) {
            set.add(node);

            //在队列中加入所有边,排序
            priorityQueue.addAll(node.edges);
            while (!priorityQueue.isEmpty()) {
                //弹出最小权值的边
                Edge edge = priorityQueue.poll();
                Node toNode = edge.to;
                //toNode没加入过set
                if (!set.contains(toNode)) {
                    //加入set
                    set.add(toNode);
                    result.add(edge);
                    //结果集中再加入该node的边(会有重复边,但是set会将已经添加过的点忽略)
                    priorityQueue.addAll(toNode.edges);
                }
            }


        }
    }
    return result;
}

Dijkstra

public class Dijkstra {    /**     *  准备一个hashMap用来维持,源点到该node的最短路径,     *  准备一个set,用来表示已经锁住的node     *     *  步骤:     *  将源节点的对应路径加入map     *  从这些路径中找到最小权值的边,获取对应的node     *  获取node的edege对应的toNode     *      如果toNode不在map中,说明距离为无穷大,直接更新距离为 源到该node+node到toNode的距离     *      如果在map中,判断  源到该node+node到toNode的距离  和 原来源到该toNode的距离大小     *       如果原来大,则不更新,反之更新最小路径     *  遍历完该node的所有边后,就将该node加入到set中,代表已经扫描过,后续不在扫描     *     *  重新从这些路径中找到最小权值的边,获取对应的node     *     */    public static HashMap dijkstra1(Node head) {        //源点到该node的最短路径        HashMap map = new HashMap<>();        //锁住的node        HashSet set = new HashSet<>();        //加入第一个节点到map中,到自己的距离为零        map.put(head,0);        //获取最小权值的node        Node minNode = getMinDisUnSelect(map, set);        //能获取到没被锁定的权值最小边        while (minNode != null){            int distance = map.get(minNode);            for (Edge edge: minNode.edges){                //获取node的edeges对应的toNode                Node toNode = edge.to;                //不在map中                if (!map.containsKey(toNode)){                    map.put(toNode, distance+edge.weight);                }                else {                    map.put( toNode,Math.min( distance+edge.weight,map.get(toNode) ) );                }            }            //当前节点完成遍历,加入到被锁set中            set.add(minNode);            //再获取一个minNode            minNode = getMinDisUnSelect(map, set);        }        return map;    }    //从没有被锁定的点中,获取权值最小的边,并返回对应节点    public static Node getMinDisUnSelect(HashMap map,HashSet set){        int minDistance = Integer.MAX_VALUE;        Node minNode = null;        for (Entry entry: map.entrySet()){            Node node = entry.getKey();            int distance = entry.getValue();            //没被锁住,并且较小            if (! set.contains(node) && distance

你可能感兴趣的:(算法基础学习3)