解题思路:

本题采用了邻接表+贪心+排序+尺取法的方法解题。

首先用邻接表存储给出的线段。

排序用在:

        1.遍历邻接表时,与遍历到的点所连接的点采用升序排序,这样取第0个元素,即为最小的点

        2.对读入的点按照升序排序。

贪心用在遍历到点i时,每次遍历时选择长度最短的,那么它可以满足以点i为起点,最多的线段

尺取法,定义尺子[start, end], start为选取到的点的最大值,end为此时可满足的最近端点:

        与一般尺取法不同的的是,我设计的尺子的起点(start)为选择到的点中最大的点。由于定义读入的点序列是升序的,所以start只能前移、不能后退,而end则是可以双向移动的。对于本题来说,end值意义不大,主要是用来判断是否出现了找不到的情况。

        在循环中,首先判断遍历到的端点i是否小于start,如果小于start,

                    在有解的情况下,目前选取到的点足够满足端点i的所有情况

                    假定端点i的一个区间[i, k],尺子区间为[start, end],只要证明k大于等于start即可:

                           如果start>k,由于start为递增点序列中选取到的某点,如果start>k,那么后续所有的点均大于k,则不可能满足[i, k]这个区间,则无解。

                    所以在有解的情况下,一定有start包含在[i, k]这个区间这个真命题。

        不断更新start和end,遇到start<i的情况,则更新start,并根据实际的点序列确定是否要增加选取到的点序列的长度。

        如果在循环中出现了start>end的情况,那么证明出现了上述假设的情况,输出-1即可。


注意事项:详情可见我的注释。

参考代码:

import java.io.*;
import java.util.*;

// https://www.dotcpp.com/oj/problem1555.html
// (邻接表+贪心+排序+尺取法)
public class Main {
    static StreamTokenizer cin;
    static PrintWriter out;
    static int n;
    static int m;
    static Set  points;
    static ArrayList[] edges;
    public static void main(String[] args) throws IOException{
        cin = new StreamTokenizer(new BufferedReader(new InputStreamReader(System.in)));
        out = new PrintWriter(new OutputStreamWriter(System.out));
        n = nextInt();
        m = nextInt();
        points = new TreeSet<>();
        edges = new ArrayList[50001];
        for(int i = 0; i < 50001; i++){
            edges[i] = new ArrayList<>();
        }
        for(int i = 0; i < n; i++){
            points.add(nextInt());
        }
        for(int i = 0; i < m; i++){
            int a = nextInt();
            int b = nextInt();
            edges[a].add(b);
        }
        int end = Integer.MAX_VALUE;  // 此时可满足的最近端点
        int count = 0; // 所需点的个数
        int start = 0; // 起始点
        boolean flag = false;
        // start为选取到的点的最大值
        Iterator iterator = points.iterator();
        for(int i = 0; i < 50001; i++){  // 遍历邻接表
            if(start > end){  // 此时无解, 具体分析见下
                flag = true;
                break;
            }
            int len = edges[i].size();
            if(len != 0){
                Collections.sort(edges[i]);  // 排序
                int min = edges[i].get(0);  // 选取该点对应线段最小值
                if(start >= i){
                    // 此时不需要更新选择到的点, 因为start为选取到的所有点中的最大值:
                        // 当start大于i时,
                            // 如果有解, 一定会存在[i, min]包含start,
                                // 因为点是升序排序的, 如果现在的最大点都没包含在这个区间内, 那么后续的点更大, 更不可能包含在此区间内
                            // 如果没解, 则会出现end > start的情况, 此时退出循环, 输出-1
                    if(min < end) // 只需要更新上限, 不需选取下一个点
                        end = min;
                    continue;
                }
                while(iterator.hasNext()){
                    int t = iterator.next();
                    // 进入循环只能是因为下限不够, 即start=i && t  end  || end > Integer.MAX_VALUE/2){
                            // 加入新点, 同时更新区间
                            count++;
                            end = min;  // 更新区间
                        }
                        if(min < end)  // 只更新终点
                            end = min;
                        break;
                    }
                }
            }
        }
        if(flag)
            out.println(-1);
        else
            out.println(count);
        out.println();
        out.flush();
    }
    static int nextInt() throws IOException{
        cin.nextToken();
        return (int)cin.nval;
    }

}
点赞(0)
 

0.0分

1 人评分

C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:

一点编程也不会写的:零基础C语言学练课程

解决困扰你多年的C语言疑难杂症特性的C语言进阶课程

从零到写出一个爬虫的Python编程课程

只会语法写不出代码?手把手带你写100个编程真题的编程百练课程

信息学奥赛或C++选手的 必学C++课程

蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程

手把手讲解近五年真题的蓝桥杯辅导课程

评论列表 共有 0 条评论

暂无评论