zoukankan      html  css  js  c++  java
  • 联赛模拟测试16(A&C未补)

    A.阴阳

    咕咕~

    B.简单的序列

    目前似乎有三种方法可过
    一种就是(Catalan),想了一下午
    还有两种是(DP)
    (Catalan)
    根据给定的(s)
    看其中有多少个是失配的
    比如())((()
    那么就要求左边的串中要出现(2)(()
    同理看,右边要有(3)())
    然后分别对于左边和右边求括号匹配,就是求卡特兰数
    然后答案是两边相乘的结果
    关于会不会有重复计算的答案
    因为枚举的是左边字符串的长度
    就相当于枚举了s在整个串当中的不同位置
    所以答案没有重复
    不用考虑去重

    Code
    #include <iostream>
    #include <cstring>
    #include <cstdio>
    #include <algorithm>
    #include <queue>
    using namespace std;
    
    inline int read(){
    	int x = 0, w = 1;
    	char ch = getchar();
    	for(; ch > '9' || ch < '0'; ch = getchar()) if(ch == '-') w = -1;
    	for(; ch >= '0' && ch <= '9'; ch = getchar()) x = x * 10 + ch - '0';
    	return x * w;
    }
    
    inline void write(register int x){
    	if(x < 0) x = ~x + 1, putchar('-');
    	if(x > 9) write(x / 10);
    	putchar(x % 10 + '0');
    }
    
    const int ss = 200010;
    const int mod = 1e9 + 7;
    
    int fac[ss], f[ss], inv[ss];
    inline void pre(register int n){
    	fac[1] = fac[0] = f[1] = f[0] = inv[1] = 1;
    	for(register int i = 2; i <= n; i++) fac[i] = 1ll * i * fac[i - 1] % mod;
    	for(register int i = 2; i <= n; i++) inv[i] = 1ll * (mod - mod / i) * inv[mod % i] % mod;
    	for(register int i = 2; i <= n; i++) f[i] = 1ll * f[i - 1] * inv[i] % mod;
    }
    
    inline int C(register int n, register int m){
    	if(!m || n == m) return 1;
    	return 1ll * fac[n] * f[m] % mod * f[n - m] % mod;
    }
    
    char a[ss];
    int stk[ss], top, l, r, maxx, len;
    long long ans;
    signed main(){
    	freopen("bracket.in", "r", stdin);
    	freopen("bracket.out", "w", stdout);
    	register int n = read(), m = read();
    	scanf("%s", a + 1);
    	for(register int i = 1; i <= m; i++){
    		if(a[i] == ')') top++, r++;
    		else top--, l++;
    		if(top > maxx) maxx = top;
    	}
    	if(n == m){
    		register bool flag = 0;
    		for(register int i = 1; i <= m; i++){
    			if(a[i] == '(') ++top;
    			else top--;
    			if(top < 0) flag = 1;
    		}
    		if(!flag && top == 0) return puts("1"), 0;
    		else return puts("0"), 0;
    	}
    	pre(n);
    	len = n - m;
    	for(register int i = 0; i <= len; i++)//p的长度
    		for(register int j = (i + maxx + 1) / 2; j <= i; j++){
    			register int now = l + j;
    			register int cur = r + i - j;
    			register int tmp = (n - i - m - now + cur) / 2;
    			ans += (((C(i, j) - C(i, j + 1) + mod) % mod) * ((C(n - i - m, tmp) - C(n - i - m, tmp - 1) + mod) % mod)) % mod;
    			if(ans >= mod) ans %= mod;
    		}
    	write(ans);
    	puts("");
    	return 0;
    }
    

    C.简单的期望

    咕咕~

    D.简单的操作

    考场上推导出来出现奇环就(-1)但是其他的操作都不会了(QWQ)
    正解(dfs)染色判断(-1)
    (bfs)直接跑联通块的最短路的最大值
    最后一遍直接统计每个联通块的答案即可
    学习完之后一遍码过表示很开心

    Code
    /* cinput
    4 6
    1 2
    2 3
    1 3
    3 4
    2 4
    1 4
    */
    #include <iostream>
    #include <cstring>
    #include <cstdio>
    #include <algorithm>
    #include <queue>
    using namespace std;
    
    inline int read(){
    	int x = 0, w = 1;
    	char ch = getchar();
    	for(; ch > '9' || ch < '0'; ch = getchar()) if(ch == '-') w = -1;
    	for(; ch >= '0' && ch <= '9'; ch = getchar()) x = x * 10 + ch - '0';
    	return x * w;
    }
    
    inline void write(register int x){
    	if(x < 0) x = ~x + 1, putchar('-');
    	if(x > 9) write(x / 10);
    	putchar(x % 10 + '0');
    }
    
    const int ss = 100010;
    
    struct node{
    	int to, nxt;
    }edge[ss << 1];
    
    int head[ss], tot;
    inline void add(register int u, register int v){
    	edge[++tot].to = v;
    	edge[tot].nxt = head[u];
    	head[u] = tot;
    }
    
    int color[ss], belong[ss];
    inline void dfs(register int u, register int col, register int cnt){
    	color[u] = col;
    	belong[u] = cnt;
    	for(register int i = head[u]; i; i = edge[i].nxt){
    		register int v = edge[i].to;
    		if(!color[v]) dfs(v, -col, cnt);
    		if(color[u] == color[v]){
    			puts("-1");
    			exit(0);
    		}
    	}
    }
    
    queue<int> q;
    bool vis[ss];
    int dis[ss];
    inline int bfs(register int s){
    	register int maxx = -1;
    	memset(vis, 0, sizeof vis);
    	q.push(s);
    	dis[s] = 0;
    	vis[s] = 1;
    	while(!q.empty()){
    		register int u = q.front();
    		q.pop();
    		maxx = max(maxx, dis[u]);
    		for(register int i = head[u]; i; i = edge[i].nxt){
    			register int v = edge[i].to;
    			if(!vis[v]){
    				vis[v] = 1;
    				q.push(v);
    				dis[v] = dis[u] + 1;
    			}
    		}
    	}
    	return maxx;
    }
    
    int tmp[ss], ans, tim;
    signed main(){
    	freopen("merge.in", "r", stdin);
    	freopen("merge.out", "w", stdout);
    	register int n = read(), m = read();
    	while(m--){
    		register int u = read(), v = read();
    		add(u, v);
    		add(v, u);
    	}
    	for(register int i = 1; i <= n; i++)
    		if(!color[i]) dfs(i, 1, ++tim);
    	for(register int i = 1; i <= n; i++)
    		tmp[belong[i]] = max(tmp[belong[i]], bfs(i));
    	for(register int i = 1; i <= tim; i++)
    		ans += tmp[i];
    	write(ans);
    	puts("");
    	return 0;
    }
    
  • 相关阅读:
    redis命令
    eclipse error pages 打红X的解决方法
    探究adroid活动
    Javascript基本算法演练 Seek and Destroy
    c语言结构体排序示例
    android studio 环境配置
    git学习
    栈用于2进制转换10进制
    html和js
    js
  • 原文地址:https://www.cnblogs.com/rui-4825/p/13814550.html
Copyright © 2011-2022 走看看