2021/11/27 总结

一天跟了三场比赛;记录一下部分题目;
https://ac.nowcoder.com/acm/contest/24803/D

题意

nxn的矩阵只能旋转最外面一层,问最后能不能使得整体有序;

题解

可以把最外层全部抠出来,然后可以观察到,他和有序的最外层的最小表示实际上是一样的,如果外面相等,里面可以按照顺序判断,复杂度n*n,(数据范围只有10,感觉可以开到1000.。。)

const int N = 20;
int g[N][N];
int d[N][N];
vector<int> v;
vector<int> get_min(vector<int>& s) {
    int n = s.size();
    for (int i = 0; i < n; i++) s.push_back(s[i]);
    int i = 0, j = 1, k = 0;
    while (i < n and j < n) {
        for (k = 0; s[i + k] == s[j + k] and k < n; k++);
        if (k == n)  break;
        if (s[i + k] > s[j + k]) {
            i += k + 1;
            if (i == j) i++;
        } else {
            j += k + 1;
            if (i == j) j++;
        }
    }
    k = min(i, j);
    vector<int> v;
    for (int i = k; i < k + n; i++) v.push_back(s[i]);
    return v;
}
int n;
vector<int> init1() {
    vector<int> v;
    for (int i = 1; i <= n; i++) {
        v.eb(d[1][i]);
    }
    for (int i = 2; i <= n; i++) {
        v.eb(d[i][n]);
    }
    for (int i = n - 1; i >= 1; i--) {
        v.eb(d[n][i]);
    }
    for (int i = n - 1; i >= 2; i--) {
        v.eb(d[i][1]);
    }
    return v;
}
void solve() {
 
    cin >> n;
    int idx = 0;
    for(int i=1;i<=n;i++){
        for(int j=1;j<=n;j++){
            cin>>d[i][j];
        }
    }
    for (int i = 1; i <= n; i++) {
        for (int j = 1; j <= n; j++) {
            g[i][j] = ++idx;
        }
    }
    for (int i = 1; i <= n; i++) {
        v.eb(g[1][i]);
    }
    for (int i = 2; i <= n; i++) {
        v.eb(g[i][n]);
    }
    for (int i = n - 1; i >= 1; i--) {
        v.eb(g[n][i]);
    }
    for (int i = n - 1; i >= 2; i--) {
        v.eb(g[i][1]);
    }
    auto v1 = init1();
    if (get_min(v) == get_min(v1)) {
        int cnt=0;
        bool ok=true;
        for(int i=1;i<=n;i++){
            for(int j=1;j<=n;j++){
                ++cnt;
                if(i==1 or i==n or j==1 or j==n) continue;
                if(d[i][j]!=cnt) ok=false;
            }
        }
        if(ok) cout<<"YES"<<endl;
        else cout<<"NO"<<endl;
    } else cout << "NO" << endl;
}

https://ac.nowcoder.com/acm/contest/24803/B

题意

问一个字母出现k次且最短的序列长度是多少

题解

可以双指针也可以二分

#include <bits/stdc++.h>
#define eb emplace_back
#define divUp(a,b) (a+b-1)/b
#define mkp(x,y) make_pair(x,y)
#define all(v) begin(v),end(v)
#define int long long
#define deb(x) cout<<#x<<" "<<x<<endl
#define fi first
#define se second
using namespace std;
typedef pair<int, int> pii;
bool checkP2(int n) {return n > 0 and (n & (n - 1)) == 0;};

const int N=100010;
char s[N];
int c[30];
int cnt[30];
void solve() {
    int m;
    cin>>(s+1);
    int n=strlen(s+1);
    memset(c,0,sizeof c);
    cin>>m;
    int res=2e9;
    for(int i=1;i<=n;i++) c[s[i]-'a']++;
    for(int i=0;i<26;i++){
        if(c[i]>=m){
            memset(cnt,0,sizeof cnt);
            for(int j=1,k=1;j<=n;j++){
                while(cnt[i]<m and k<=n) {
                    if(s[k]==i+'a') cnt[i]++;
                    k++;
                }
                if(cnt[i]>=m) {
                    res=min(res,k-j);
                }
                if(s[j]==i+'a') cnt[i]--;
            }
        }
    }
    if(res==2e9) cout<<-1<<endl;
    else cout<<res<<endl;

}
signed main() {

    std::ios::sync_with_stdio(false); cin.tie(0); cout.tie(0);
    int _; cin >> _; while (_--)
        solve();
    return 0;
}

https://atcoder.jp/contests/abc229/tasks/abc229_dhttps://atcoder.jp/contests/abc229/tasks/abc229_d

题意

一个序列只有x和.最多可以花费k次把.变成x,问连续x的最长序列长度是多少

题解

发现可以二分序列长度,只要再这个长度内花费小于等于k即可,利用前缀和统计x就可以知道.的个数

const int N = 200010;
char s[N];
int sum[N];
int k;
int n;
bool check(int mid) {
    for (int i = mid, j = 1;i <= n;i++, j++) {
        int cur = sum[i] - sum[j - 1];
        int len = i - j + 1;
        if (len - cur <= k) {
           
            return true;
        }
    }
    return false;
}
void solve() {
    cin >> (s + 1)>>k;
    n=strlen(s+1);
    for (int i = 1;i <= n;i++) {
        if (s[i] == 'X') sum[i] = 1;
        sum[i] += sum[i - 1];
    }
    int l = 0, r = n;
    while (l < r) {
        int mid = (l + r + 1) >> 1;
        if (check(mid)) l = mid;
        else r = mid - 1;
    }
    cout << l << endl;
}

https://atcoder.jp/contests/abc229/tasks/abc229_e

题意

给了一个图,每次按照顺序删除编号为i的点,问删除第i个点之后的连通块有多少

思路

求连通块的话很明显可以用并查集,但是每次按照顺序的话,后面删除操作比较困难,所以可以选择倒着跑,每次加进来一个点,如果发现这个点的根节点是他自己,就说明这是一个新的块,每个父节点可以使用set来记录,也方便删除,还有注意一点,加边的时候是较小的边指向较大的边,考虑较大边的时候较小边已经被删除;

int p[N];
int find(int x){
    if(x!=p[x]) p[x]=find(p[x]);
    return p[x];
}
set<int> s;
pii a[N];
int ans[N];
void merge(int x,int y){
    int pa=find(x),pb=find(y);
    if(pa!=pb){
        p[pa]=pb;
        s.erase(pa);
    }
}
vector<int> h[N];
void solve() {
    int n,m;
    cin>>n>>m;
    int x,y;
    for(int i=1;i<=m;i++){
        cin>>x>>y;
        h[min(x,y)].eb(max(x,y));
    }
    for(int i=1;i<=n;i++) p[i]=i;
    for(int i=n;i>=1;i--){
        ans[i]=s.size();
        if(find(i)==i) s.insert(i);
        for(auto j:h[i]){
            merge(i,j);
        }
    }
    for(int i=1;i<=n;i++) cout<<ans[i]<<endl;
}

leetcode双周赛66T4

题意

给定一个矩阵,让你找到所有的正金字塔和倒金字塔,(底边是3,5,7,9.。。。);

思路

直接暴力查找,使用前缀和做优化即可

class Solution {
public:
    vector<vector<int>> g,d;
    int n,m;
    int sum(int x,int y,int cnt){
        if(x>=n or y<0 or y+cnt>=m) return 0;
        if(!d[x][y]) return 0;
        if(g[x][y+cnt]-g[x][y]!=cnt) return 0;
        return sum(x+1,y-1,cnt+2)+1;
    }
    int sum1(int x,int y,int cnt){
        if(x<0 or y<0 or y+cnt>=m) return 0;
        if(!d[x][y]) return 0;
        if(g[x][y+cnt]-g[x][y]!=cnt) return 0;
        return sum1(x-1,y-1,cnt+2)+1;
    }
    int countPyramids(vector<vector<int>>& grid) {
        d=grid;
        g=grid;
        m=grid[0].size();
        n=grid.size();
        int res=0;
        for(int i=0;i<n;i++){
            g[i][0]=grid[i][0];
            for(int j=1;j<m;j++){
                g[i][j]=grid[i][j]+g[i][j-1];
            }
        }
        for(int i=0;i<n;i++){
            for(int j=0;j<m;j++){
                if(grid[i][j]){
                    res+=sum1(i,j,0)-1+sum(i,j,0)-1;
                }
            }
        }
        return res;
    }
};

上一篇:BZOJ 5125 小Q的书架


下一篇:在报表中录入数据时如何实现行列转换