【笔试】美团2023年秋招第1场笔试(后端&数开&软件方向)

文章目录

      • [T1 小美玩排列](#T1 小美玩排列)
      • [T2 小美走公路](#T2 小美走公路)
      • [T3 小美切蛋糕](#T3 小美切蛋糕)
      • [T4 小美将字符串平铺成矩阵](#T4 小美将字符串平铺成矩阵)
      • [T5 小美染色](#T5 小美染色)

23秋招,美团笔试1(技术)

美团2024届秋招笔试第一场编程真题

时间:2023.08,牛客补题

美团是少有的整份卷子5题都是算法题的,20*5=100分,其他家一般会有408选择题再配2个编程题。

所以美团的T4、T5有时候会稍微有点卡点,我也没法保证每次都ak,有时候会有只能拿个九十几的情况,不过好在笔试可以做两次(虽然好像AK也不给面)

T1 小美玩排列

有一个排列,一共有n个数,还有特殊的两个数x和y,请你帮助小美判断x和y在排列中是否相邻,是则输出"Yes",不是则输出"No"

数据范围:

1 ≤ n ≤ 1e5

cpp 复制代码
//AC
#include <iostream>
#include <bits/stdc++.h>
using namespace std;
const int maxn = 200010;
int p[maxn];
int main() {
    int n;  cin>>n;
    for(int i = 1; i <= n; i++){
        int x;  cin>>x;
        p[x] = i;
    }
    int x, y;
    cin>>x>>y;
    if(abs(p[x]-p[y])==1){
        cout<<"Yes\n";
    }else{
        cout<<"No\n";
    }
}

T2 小美走公路

现有一条环形公路,总共有n个站点,a[i]代表第i个站点与第i+1个站点之间的距离,特殊的,a[n]表示第n个站点与第一个站点之间的距离。小美的出发地为x,目的地为y,请你求出x到y的最短距离

1 ≤ n ≤ 1e5

1 ≤ a[i] ≤ 1e9

cpp 复制代码
//AC
#include <iostream>
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
const int maxn = 200010;
LL a[maxn], s[maxn];
int main() {
    LL n;  cin>>n;
    LL sum = 0;
    for(LL i = 1; i <= n; i++){
        cin>>a[i];
        sum += a[i];
    }
    LL x,y ;  cin>>x>>y;
    if(x > y)swap(x,y);
    LL res = 0;
    for(LL i = x; i < y; i++){
        res += a[i];
    }
    res = min(res, sum-res);
    cout<<res<<"\n";
}

T3 小美切蛋糕

现有一个n*m的蛋糕矩阵a,a[i][j]代表一小块蛋糕的美味度,现在小美要和一个好朋友分享蛋糕,因此需要把这个蛋糕矩阵切成两半,并且要求分成两半后的两块蛋糕的美味度尽可能相等,即求出分成两半后的两块蛋糕的abs(s1 - s2)的最小值,s1代表第一块蛋糕的美味度,s2代表第二块蛋糕的美味度。要求:必须保证每一小块蛋糕的完整性(即不能斜着切,如果把整个大蛋糕正着放)

1 ≤ n , m≤ 1e3

1 ≤ a[i][j] ≤ 1e5

cpp 复制代码
//AC
#include <iostream>
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
LL a[1010][1010];
// LL b[1010][1010], c[1010][1010];
LL b[1010], c[1010];
int main() {
    LL n, m;  cin>>n>>m;
    LL sum = 0;
    for(LL i = 1; i <= n; i++){
        for(LL j = 1; j <= m; j++){
            cin>>a[i][j];
            b[i] += a[i][j];
            c[j] += a[i][j];
            sum += a[i][j];
            // b[i][j] += a[i][j-1];
            // c[i][j] += a[i-1][j];
        }
    }
    LL res = 1e9+10, tmp = 0;
    for(LL i = 0; i <= n; i++){
        tmp += b[i];
        res = min(res, abs((sum-tmp)-tmp));
    }
    tmp = 0;
    for(LL j = 0; j <= m; j++){
        tmp += c[j];
        res = min(res, abs((sum-tmp)-tmp));
    }
    cout<<res<<"\n";
}

T4 小美将字符串平铺成矩阵

现有一个长度为n的且仅包含小写字母的字符串s,小美想把这个字符串s平铺成一个xy的矩阵,要求xy == n,平铺的方法为:将字符串前y个字符按顺序放到第一行,将字符串第y+1到第2*y个字符按顺序放到第二行,以此类推。现规定矩阵的权值为连通量的数目,连通量代表的是从一个点出发,上下左右若存在相同字符则可以继续扩展该连通量(类似于一个岛屿,上下左右若存在相同的字符则可以扩展这个岛屿,或者可以理解为上下左右如果是相同字符则可以合并成一个连通量),求矩阵的最小权值

1 ≤ n ≤ 1e4

cpp 复制代码
//WA-最开始想直接set乱来,最后还是被迫换成dfs写
#include <iostream>
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
int dx[] = {-1, 1, 0 ,0 };
int dy[] = {0, 0, -1, 1};
int main() {
    int n;  cin>>n;
    string s;  cin>>s;
    set<char>se;
    for(char ch : s){
        se.insert(ch);
    }
    // cout<<s[5]<<"\n";
    n = s.size();
    int res = 1e9+10;
    for(int i = 2; i < n; i++){
        if(n%i!=0)continue;
        int ts = 0;
        for(int j = 0; j < n; j++){
            set<char>tmp;
            int z = 0;
            int x = j/i, y = j%i;
            // tmp.insert(s[j]);
            int ok = 0;
            // z++;
            for(int k = 0; k < 4; k++){
                int nx = x+dx[k], ny = y+dy[k];
                if(nx<0||nx>=n/i || ny<0||ny>=i)continue;
                if(s[nx*i+ny]==s[j]){
                    // cout<<j<<" "<<nx<<" "<<ny<<" "<<s[nx*i+ny]<<" "<<s[j]<<"\n";
                    ok = 1;
                }
                
                // tmp.insert(s[nx*i+ny]);
                // z++;
                if(i==3 && j==2){
                    // cout<<nx<<" "<<ny<<" "<<s[nx*i+ny]<<" "<<s[j]<<"\n";
                    // cout<<x<<" "<<y<<" ";
                    // cout<<nx<<' '<<ny<<" asd\n";
                }
            }
            // if(j>=0){tmp.insert(s[j-1]); z++; }
            // if(j<n){tmp.insert(s[j+1]); z++; }
            // if(j-i>=0){tmp.insert(s[j-i]); z++; }
            // if(j+i<n){tmp.insert(s[j+i]); z++; }
            // cout<<i<<","<<j<<","<<tmp.size()<<" "<<z<<"\n";
            // for(char ch : tmp)cout<<ch; cout<<"\n";
            // if(tmp.size()==z){
            //     ts++;
            //     // cout<<i<<", "<<j<<"\n";
            // }
            if(ok!=1){
                ts++;
                // cout<<i<<", "<<j<<"\n";
            }
        }
        if(ts<res){
            // cout<<i<<" "<<ts<<"\n";
        }
        res = min(res, ts);
    }
    // cout<<se.size()<<"\n";
    cout<<(int)(res+se.size())<<"\n";
}
cpp 复制代码
//AC
//T4-AC
#include <iostream>
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
int dx[] = {-1, 1, 0 ,0 };
int dy[] = {0, 0, -1, 1};
string s;
int n; 
int vis[10010];
// int ts = 0;
int ttt;
void dfs(int x, int y,int d){
        // cout<<x<<" "<<y<<"\n";
    for(int i = 0; i < 4; i++){
        int nx = x+dx[i], ny = y+dy[i];
        int nj = nx*d+ny;
        if(nx<0||nx>=n/d || ny<0||ny>=d || vis[nj])continue;
        if(s[nj] != s[ttt])continue;
        if(!vis[nj]){
            vis[nj] = 1;
            dfs(nx,ny,d);
        }
    }
}
int main() {
     cin>>n;
      cin>>s;
    int res = 1e9+10;
    for(int i = 1; i <= n; i++){
        if(n%i!=0)continue;
        int ts = 0;
        for(int j = 0; j < n; j++)vis[j] = 0;
        for(int j = 0; j < n; j++){
            int x = j/i, y = j%i;
            if(!vis[j]){
                // cout<<x<<' '<<y<<" asfd\n";
                ttt = j;
                dfs(x, y, i);
                ts++;
            }
        }
        // cout<<i<<" "<<ts<<"\n";
        res = min(res, ts);
    }
    // cout<<se.size()<<"\n";
    cout<<(int)(res)<<"\n";
}

T5 小美染色

现有一颗包含n个节点的树,节点i的权值为w[i],d[i]代表节点i相邻的节点集合。一开始所有节点均为白色,若两个相邻节点均为白色且权值的乘积为完全平方数则可以将两个节点都染成红色,问最多可以将多少个节点染成红色

1 ≤ n ≤ 1e5

1 ≤ w[i] ≤ 1e9

cpp 复制代码
// 输出0有10%(数据3/30),暴力+剪枝可以再多骗一部分
// AC 
//1、给出n个点和n-1条边的一棵树
//2、每次操作,相邻且白色且权值的乘积是完全平方数的,改成红色。求最多可以染多少个红。
//3、每个点有两个状态,选和不选,那么就是0/1DP了,又给了个树,也就是树上01DP
// u是根节点 ,i是子节点。不选根时,子节点的最大方案数统计上来就行,dp[u][0] += max(dp[i][0],dp[i][1])
// 选根,需要符合条件(is_sqrt(a[u]*a[i])==1)的子节点同时不被选中,那么dp[u][1]=dp[u][0]-max(dp[i][0],dp[i][1])+dp[i][0]+2,即回退这个点的记录,并更新成同时选中这个子节点和根节点的状态。
// 最后答案就是 max(dp[0][0],dp[0][1])
#include <iostream>
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
const LL maxn = 1e5 + 10;
LL a[maxn], c[maxn];
vector<LL>G[maxn];
LL dp[maxn][2];
LL res = 0;
int checksqrt(int x) { int d = sqrtl(x)+1e-7; return d*d == x; };
void dfs(LL u, LL f) {
    for (LL i : G[u]) {
        if (i == f)continue;
        dfs(i, u);
        dp[u][0] += max(dp[i][0], dp[i][1]);
        // if(checksqrt(a[i]*a[u])){  //需要先把所有子节点的dp状态都更新完了,确保dp[i][0/1]都是最优的,才能更新这里,不然只有7%,别问我怎么知道的
        //     dp[u][1] = max(dp[u][1], dp[u][0] - max(dp[i][0], dp[i][1]) + dp[i][0] + 2);
        // }
    }
    for (LL i : G[u]) { 
        if (i == f)continue;
        if(checksqrt(a[i]*a[u])){ 
            dp[u][1] = max(dp[u][1], dp[u][0] - max(dp[i][0], dp[i][1]) + dp[i][0] + 2);
        }
    }
}
int main() {
    LL n;
    cin >> n;
    for (LL i = 1; i <= n; i++) {
        cin >> a[i];
    }
    for (LL i = 1; i < n; i++) {
        LL u, v;  
        cin >> u >> v;
        G[u].push_back(v);
        G[v].push_back(u);
    }
    dfs(1, -1);
    cout << max(dp[1][0], dp[1][1]) << "\n";
}
相关推荐
浅念同学5 分钟前
算法.图论-并查集上
java·算法·图论
何不遗憾呢13 分钟前
每日刷题(算法)
算法
立志成为coding大牛的菜鸟.18 分钟前
力扣1143-最长公共子序列(Java详细题解)
java·算法·leetcode
鱼跃鹰飞18 分钟前
Leetcode面试经典150题-130.被围绕的区域
java·算法·leetcode·面试·职场和发展·深度优先
liangbm324 分钟前
数学建模笔记——动态规划
笔记·python·算法·数学建模·动态规划·背包问题·优化问题
潮汐退涨月冷风霜29 分钟前
机器学习之非监督学习(四)K-means 聚类算法
学习·算法·机器学习
B站计算机毕业设计超人35 分钟前
计算机毕业设计Python+Flask微博情感分析 微博舆情预测 微博爬虫 微博大数据 舆情分析系统 大数据毕业设计 NLP文本分类 机器学习 深度学习 AI
爬虫·python·深度学习·算法·机器学习·自然语言处理·数据可视化
羊小猪~~39 分钟前
深度学习基础案例5--VGG16人脸识别(体验学习的痛苦与乐趣)
人工智能·python·深度学习·学习·算法·机器学习·cnn
limingade4 小时前
手机实时提取SIM卡打电话的信令和声音-新的篇章(一、可行的方案探讨)
物联网·算法·智能手机·数据分析·信息与通信
jiao000016 小时前
数据结构——队列
c语言·数据结构·算法