贪心算法-宣讲会问题

贪心算法(又称贪婪算法)是指,在对问题求解时,总是做出在当前看来是最好的选择。
也就是说,不从整体最优上加以考虑,算法得到的是在某种意义上的局部最优解 

贪心算法的在笔试时的解题套路
1,实现一个不依靠贪心策略的解法X,可以用最暴力的尝试
2,脑补出贪心策略A、贪心策略B、贪心策略C..
3,用解法X和对数器,去验证每一个贪心策略,用实验的方式得知哪个贪心策略正确
4,不要去纠结贪心策略的证明 

题目一:

一些项目要占用一个会议室宣讲,会议室不能同时容纳两个项目的宣讲。
给你每一个项目开始的时间和结束的时间(给你一个数组,里面是一个个具体的项目),你来安排宣讲的日程,要求会议室进行的宣讲的场次最多。
返回这个最多的宣讲场次。

solution:

先把会议按照结束时间从小到大排序,
优先安排会议结束早的宣讲,然后删除和此会议时间冲突的宣讲,再从剩余的宣讲中选择结束时间早的宣讲......

算法代码:

public class BestArrange {

    //宣讲项目模型
    public static class Program {
        public int start;
        public int end;

        public Program(int start, int end) {
            this.start = start;
            this.end = end;
        }
    }

    //自定义比较器
    public static class ProgramComparator implements Comparator<Program> {
        @Override
        public int compare(Program o1, Program o2) {
            return o1.end - o2.end;
        }

    }

    /**
     * 贪心算法:宣讲会议安排
     * @param programs :所有的会议
     * @param start :目前来到的时间点
     * @return :返回最多安排次数
     */
    public static int bestArrange(Program[] programs, int start) {
        Arrays.sort(programs, new ProgramComparator()); //先把所有的项目根据结束时间谁早进行排序
        System.out.println("排序后的会议为:");
        for (int i = 0; i < programs.length; i++) {
            System.out.print(programs[i]+" ");
        }
        System.out.println();
        int result = 0;
        //从左往右依次遍历所有的会议
        for (int i = 0; i < programs.length; i++) {
            if (start <= programs[i].start) { //现在来到的时间点有没有早于当前考虑会议的开始时间
                result++;
                start = programs[i].end; // start来到会议结束的时候
            }
        }
        return result;
    }   
}

测试:

package Algorithms.greed;

import java.util.Arrays;
import java.util.Comparator;

public class BestArrange {

    //宣讲项目模型
    public static class Program {
        public int start;
        public int end;

        public Program(int start, int end) {
            this.start = start;
            this.end = end;
        }

        @Override
        public String toString() {
            return "Program{" +
                    "start=" + start +
                    ", end=" + end +
                    '}';
        }
    }

    //自定义比较器
    public static class ProgramComparator implements Comparator<Program> {
        @Override
        public int compare(Program o1, Program o2) {
            return o1.end - o2.end;
        }

    }

    /**
     * 贪心算法:宣讲会议安排
     * @param programs :所有的会议
     * @param start :目前来到的时间点
     * @return :返回最多安排次数
     */
    public static int bestArrange(Program[] programs, int start) {
        Arrays.sort(programs, new ProgramComparator()); //先把所有的项目根据结束时间谁早进行排序
        System.out.println("排序后的会议为:");
        for (int i = 0; i < programs.length; i++) {
            System.out.println(programs[i]+" ");
        }
        System.out.println();
        int result = 0;
        int n = 0;
        //从左往右依次遍历所有的会议        
        for (int i = 0; i < programs.length; i++) {
            if (start <= programs[i].start) { //现在来到的时间点有没有早于当前考虑会议的开始时间
                result++;
                n++;
                System.out.println("第 "+n+" 场会议为"+programs[i]);
                start = programs[i].end; // start来到会议结束的时候
            }
        }
        return result;
    }

    public static void main(String[] args) {

        Program[] programs = {new Program(3, 10),new Program(3, 5),new Program(4, 11),
                new Program(7, 13),new Program(8, 12),new Program(14, 15)};
        int maxNumbers = bestArrange(programs, 2);
        System.out.println("会议的最大安排次数为:"+maxNumbers);
    }
}

/**
 * 排序后的会议为:
 * Program{start=3, end=5}
 * Program{start=3, end=10}
 * Program{start=4, end=11}
 * Program{start=8, end=12}
 * Program{start=7, end=13}
 * Program{start=14, end=15}
 *
 * 第 1 场会议为Program{start=3, end=5}
 * 第 2 场会议为Program{start=8, end=12}
 * 第 3 场会议为Program{start=14, end=15}
 * 会议的最大安排次数为:3
 */

 

posted @ 2021-08-15 12:37  zh_小猿  阅读(88)  评论(0编辑  收藏  举报