5.9美团笔试5道算法题

本人的战绩如下:
100%、100%,45%,100%,64%
第三题和第五题超时,有哪位ac的大佬欢迎在评论区讨论一下
求美团爸爸一次面试机会,非常非常非常想去美团,球球了

t1:100%
trick:将坐标“拉直”, 注意去重,不然卡82%
package meituan.t1;

import java.util.ArrayList;
import java.util.List;
import java.util.Scanner;

public class Main {
    static int n, m, tot;
    static List<Integer>[] link, cost;
    static int[] dis;

    public static void main(String[] args) {
        Scanner in = new Scanner(System.in);
        m = in.nextInt();
        n = in.nextInt();
        tot = m * n - 1;
        int k = in.nextInt();
        link = new ArrayList[n * m];
        cost = new ArrayList[n * m];
        dis = new int[n * m];
        for (int i = 0; i < n * m; i++) {
            link[i] = new ArrayList<>();
            cost[i] = new ArrayList<>();
            dis[i] = Integer.MAX_VALUE;
        }
        int x, y, u, v, w;
        for (int i = 1; i <= k; i++) {
            x = in.nextInt() - 1;
            y = in.nextInt() - 1;
            u = in.nextInt() - 1;
            v = in.nextInt() - 1;
            w = in.nextInt();
            if (x == u && y == v) continue; //注意去重,不然卡82%
            link[x * n + y].add(u * n + v);
            cost[x * n + y].add(w);
        }
        //初始值为0
        dis[0] = 0;
        dfs(0);
        System.out.println(dis[tot] == Integer.MAX_VALUE ? -1 : dis[tot]);
    }

    static void dfs(int cur) {
        if (cur == tot) return;
        for (int i = 0; i < link[cur].size(); i++) {
            dis[link[cur].get(i)] = Math.min(dis[link[cur].get(i)], dis[cur] + cost[cur].get(i));
            dfs(link[cur].get(i));
        }
    }
}

t2:100%
思路:滑动窗口。
package meituan.t2;

import java.util.ArrayDeque;
import java.util.Queue;
import java.util.Scanner;

public class Main {
    public static void main(String[] args) {
        Scanner in = new Scanner(System.in);
        int n = in.nextInt(), m = in.nextInt(), h = in.nextInt();
        int height;
        Queue<Integer> queue = new ArrayDeque<>();
        for (int i = 0; i < n; i++) {
            height = in.nextInt();
            if (height <= h) queue.add(i);
            if (queue.size() > 0) {
                if (i - queue.peek() + 1 == m) {
                    if (queue.size() == m) {
                        System.out.println(queue.peek() + 1);
                        return;
                    } else queue.remove();
                }
            }

        }
        System.out.println(-1);
    }
}
t3:45% ->TLE
思路:记忆化暴力搜索。。。
package meituan.t3;

import java.util.HashMap;
import java.util.Map;
import java.util.Objects;
import java.util.Scanner;

//t3
public class Main {
    static int x, a, b, n;
    static Map<Node, Long> map;

    public static void main(String[] args) {
        Scanner in = new Scanner(System.in);
        int t = in.nextInt();
        while (t-- > 0) {
            x = in.nextInt();
            a = in.nextInt();
            b = in.nextInt();
            n = in.nextInt();
            map = new HashMap<>();
            System.out.println(dfs(0, x));
        }
    }

    //记录两个值,cur 和 status
    static long dfs(int cur, int status) {
        if (status <= 0 || cur == n) return 0;
        Node node = new Node(cur, status);
        if (map.containsKey(node)) return map.get(node);
        long res = Math.max(dfs(cur + 1, status > a ? status - a : 0) + status, dfs(cur + 1, status + b));
        map.put(node, res);
        return res;
    }

}

class Node {
    int time, status;

    public Node(int time, int status) {
        this.time = time;
        this.status = status;
    }  public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        Node node = (Node) o;
        return time == node.time &&
                status == node.status;
    }
        public int hashCode() {
        return Objects.hash(time, status);
    }
}
t4:100%
思路:广搜,多一维记录即可
package meituan.t4;

import java.util.ArrayDeque;
import java.util.Queue;
import java.util.Scanner;

//t4
public class Main {
    static int n, m;
    static int[][] dir = new int[][]{{-1, 0}, {1, 0}, {0, 1}, {0, -1}};
    static char[][] map;
    static int[][][] dis;

    public static void main(String[] args) {
        Scanner in = new Scanner(System.in);
        m = in.nextInt();
        n = in.nextInt();
        in.nextLine();
        map = new char[m][n];
        //1表示有一次机会破坏,0表示没有机会破坏了
        dis = new int[m][n][2];
        for (int i = 0; i < m; i++) {
            for (int j = 0; j < n; j++) {
                dis[i][j][0] = dis[i][j][1] = Integer.MAX_VALUE;
            }
        }
        dis[0][0][0] = dis[0][0][1] = 0;
        for (int i = 0; i < m; i++) map[i] = in.nextLine().toCharArray();
        Queue<Node> queue = new ArrayDeque<>();
        queue.add(new Node(0, 0, 1));
        Node root;
        while (!queue.isEmpty()) {
            root = queue.remove();
            for (int i = 0; i < 4; i++) {
                int dx = root.x + dir[i][0];
                int dy = root.y + dir[i][1];
                if (check(dx, dy)) {
                    if (map[dx][dy] == '*' && root.cnt == 1 && dis[dx][dy][0] > dis[root.x][root.y][1] + 1) {
                        dis[dx][dy][0] = dis[root.x][root.y][1] + 1;
                        queue.add(new Node(dx, dy, 0));
                    }
                    if (map[dx][dy] == '.' && dis[dx][dy][root.cnt] > dis[root.x][root.y][root.cnt] + 1) {
                        dis[dx][dy][root.cnt] = dis[root.x][root.y][root.cnt] + 1;
                        queue.add(new Node(dx, dy, root.cnt));
                    }
                }
            }
        }
        //拥有破坏一次障碍物的机会
        int res = Math.min(dis[m - 1][n - 1][0], dis[m - 1][n - 1][1]);
        System.out.println(res == Integer.MAX_VALUE ? -1 : res);
    }


    static boolean check(int x, int y) {
        return x >= 0 && y >= 0 && x < m && y < n;
    }

}

class Node {
    int x, y, cnt;

    Node(int x, int y, int cnt) {
        this.x = x;
        this.y = y;
        this.cnt = cnt;
    }
}
t5:64%
思路:有点像力扣上的一道hard题,这里采用它的做法:记忆化暴力搜索。
package meituan.t5;

import java.util.Scanner;

//t5
public class Main {
    static long[] pre1, pre2;
    static long[] cost;

    public static void main(String[] args) {
        Scanner in = new Scanner(System.in);
        cost = new long[26];
        for (int i = 0; i < 26; i++) cost[i] = in.nextLong();
        in.nextLine();
        String s = in.nextLine();
        String t = in.nextLine();
        int len1 = s.length(), len2 = t.length();
        pre1 = new long[len1];
        pre2 = new long[len2];
        for (int i = 0; i < len1; i++) pre1[i] = (i > 0 ? pre1[i - 1] : 0) + cost[s.charAt(i) - 'a'];
        for (int i = 0; i < len2; i++) pre2[i] = (i > 0 ? pre2[i - 1] : 0) + cost[t.charAt(i) - 'a'];
        long[][] dp = new long[len1][len2];
        //删除的代价最少,那么剩余字符的价值总和应该是最大的
        long res = dfs(s, 0, len1, t, 0, len2, dp);
        System.out.println(pre1[len1 - 1] + pre2[len2 - 1] - res);
    }

    static long dfs(String s, int cur1, int len1, String t, int cur2, int len2, long[][] dp) {
        if (cur1 == len1) return pre2[len2 - 1] - (cur2 > 0 ? pre2[cur2 - 1] : 0);
        if (cur2 == len2) return pre1[len1 - 1] - (cur1 > 0 ? pre1[cur1 - 1] : 0);
        if (dp[cur1][cur2] > 0) return dp[cur1][cur2];
        long change;
        if (s.charAt(cur1) == t.charAt(cur2)) change = dfs(s, cur1 + 1, len1, t, cur2 + 1, len2, dp);
        else {
            //删除s[cur1] 或者 删除t[cur2]
            change = Math.min(dfs(s, cur1 + 1, len1, t, cur2, len2, dp) + cost[s.charAt(cur1) - 'a'],
                    dfs(s, cur1, len1, t, cur2 + 1, len2, dp) + cost[t.charAt(cur2) - 'a']);
        }
        return dp[cur1][cur2] = change;
    }
}
//7 -5 4 6 2 5 8 -4 -8 4 10 -1 3 3 -5 10 7 0 -5 -4 -9 3 8 0 8 -3
//nygfh
//lixawy





#美团##笔试题目#
全部评论
牛啊,我就写出来一道题😓
6 回复 分享
发布于 2021-05-09 12:21
其他题我没你高,但是我第三题91%,思路是这样的,如果要锻炼的话,收益是x-a*t,t是剩余天数,当天收益x,但是后面每天相当于损失a;休息的话收益是b*t,相当于后面每天的收益。然后每天比较他们大小选择锻炼还是休息,算是一种模糊求解吧...再注意下x一定大于等于0
3 回复 分享
发布于 2021-05-09 12:36
第三题我用等差数列做的,求了个导数把休息时间最大值算出来。。。
3 回复 分享
发布于 2021-05-09 16:18
楼主稳了
2 回复 分享
发布于 2021-05-09 12:27
突然发现自己好菜
2 回复 分享
发布于 2021-05-09 12:30
第一题 我用dp 27%超时 第二题 ac 第三题用dp一个小时没做出来,0通过率 第四题第五题没时间看 还有比我更菜的吗😥
2 回复 分享
发布于 2021-05-09 12:49
第三题直接三分,第五题最长公共子序列
1 回复 分享
发布于 2021-05-09 12:36
众所周知,能不能进面试和AC题目多少并没有关系
1 回复 分享
发布于 2021-05-09 18:07
第五题我直接偷鸡大法,记录两个字符串每个字符出现的频率,然后取最小值*2*价值,直接输出。好像过了40%还是60%。记不清了
1 回复 分享
发布于 2021-05-09 22:41
mark一下,收到面试通知说一下
1 回复 分享
发布于 2021-05-10 10:39
太强了😣
点赞 回复 分享
发布于 2021-05-09 12:29
第三题贪心,假设先连续休息i天,然后连续工作n-i天,后n-i天收益是个等差数列,套公式求和,解一元二次方程最大值即可
点赞 回复 分享
发布于 2021-05-09 12:55
哦对了楼主,第三题我记得要全用long好像?x变化后可能超出int
点赞 回复 分享
发布于 2021-05-09 13:07
不知就问,可以用本地编译器吗
点赞 回复 分享
发布于 2021-05-09 13:39
有笔试题目吗
点赞 回复 分享
发布于 2021-05-09 15:58
大佬
点赞 回复 分享
发布于 2021-05-09 18:35
第五题原型是,最长公共子序列,加上对权重的判断,可以ac
点赞 回复 分享
发布于 2021-05-09 18:45
你这也太稳了
点赞 回复 分享
发布于 2021-05-09 19:55
是在塞码网的吗,为什么我考试的时候并没有看到我的通过率😱只有调试自己的用例和保存代码
点赞 回复 分享
发布于 2021-05-09 21:20
大佬什么时候投的啊。。。
点赞 回复 分享
发布于 2021-05-09 21:51

相关推荐

2024-12-03 09:59
北京邮电大学 Java
点赞 评论 收藏
分享
评论
43
107
分享

创作者周榜

更多
牛客网
牛客企业服务