COCI Parametriziran —— 状压+dfs求相似字符串个数

2024-04-07 00:32

本文主要是介绍COCI Parametriziran —— 状压+dfs求相似字符串个数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Description
在这里插入图片描述

Input
在这里插入图片描述

Output

在这里插入图片描述

Sample Input

3 3
??b
c??
c?c

4 6
ab??c?
??kll?
a?k??c
?bcd??

5 2
??
b?
c?
?g
cg
Sample Output

2

3

8
Hint
在这里插入图片描述

题意:

给你n个长度相同的字符串,这些字符串中含有小写字母和’?’,’?'可以替换成任意字符,如果两个替换之后的字符串可以相等,那么就称为这两个字符串相似,求相似字符串的对数。

题解:

一个字符串,我们可以只考虑它数字的情况,例如a?b?c我们可以只考虑abc这三个位置,用状压来存,因为只有1<<6种可能,所以开这么大的unordered_map,那么对于上面的情况我们要查询的就是10101这个map,abc有很多对应的可能:abc,?bc,a?c,ab?,??c,?b?,a??,???都是可以的,所以我们dfs找他的所有可能,之后在将原字符串的所有可能的情况加入map:10000的位置加入a,01000的位置加入?,11000的位置加入a?,以此类推,注意string是会mle的,需要用hash。

#pragma gcc optimize(2)
#include<bits/stdc++.h>
using namespace std;
#define ll unsigned int
unordered_map<ll,int>mp[(1<<6)];
int p[10];
char s[10];
int have,n,m;
long long ans;
void dfs(int pos,int val)
{if(pos>=m){int has=0;for(int i=0;i<m;i++){if(!(have&p[i]))continue;if((val&p[i]))has=has*31+s[i];elsehas=has*31+'?';}ans+=mp[have][has];return ;}if(!(have&p[pos]))dfs(pos+1,val);else{for(int i=0;i<=1;i++)dfs(pos+1,(val|(i==0?0:p[pos])));}
}
int main()
{p[0]=1;for(int i=1;i<=6;i++)p[i]=p[i-1]*2;scanf("%d%d",&n,&m);int maxn=(1<<m),has;for(int i=1;i<=n;i++){scanf("%s",s);have=0;for(int j=0;j<m;j++)if(s[j]!='?')have|=p[j];dfs(0,0);for(int i=0;i<maxn;i++){has=0;for(int j=0;j<m;j++){if((i&p[j]))has=has*31+s[j];}mp[i][has]++;}}printf("%lld\n",ans);return 0;
}

这篇关于COCI Parametriziran —— 状压+dfs求相似字符串个数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/881228

相关文章

spoj705( 求不相同的子串个数)

题意:求串s的不同子串的个数 解题思路:任何子串都是某个后缀的前缀,对n个后缀排序,求某个后缀的前缀的个数,减去height[i](第i个后缀与第i-1 个后缀有相同的height[i]个前缀)。 代码如下: #include<iostream>#include<algorithm>#include<stdio.h>#include<math.h>#include<cstrin

hdu 2489 (dfs枚举 + prim)

题意: 对于一棵顶点和边都有权值的树,使用下面的等式来计算Ratio 给定一个n 个顶点的完全图及它所有顶点和边的权值,找到一个该图含有m 个顶点的子图,并且让这个子图的Ratio 值在所有m 个顶点的树中最小。 解析: 因为数据量不大,先用dfs枚举搭配出m个子节点,算出点和,然后套个prim算出边和,每次比较大小即可。 dfs没有写好,A的老泪纵横。 错在把index在d

XTU 1233 n个硬币连续m个正面个数(dp)

题面: Coins Problem Description: Duoxida buys a bottle of MaiDong from a vending machine and the machine give her n coins back. She places them in a line randomly showing head face or tail face o

poj 3050 dfs + set的妙用

题意: 给一个5x5的矩阵,求由多少个由连续6个元素组成的不一样的字符的个数。 解析: dfs + set去重搞定。 代码: #include <iostream>#include <cstdio>#include <set>#include <cstdlib>#include <algorithm>#include <cstring>#include <cm

ural 1149. Sinus Dances dfs

1149. Sinus Dances Time limit: 1.0 second Memory limit: 64 MB Let  An = sin(1–sin(2+sin(3–sin(4+…sin( n))…) Let  Sn = (…( A 1+ n) A 2+ n–1) A 3+…+2) An+1 For given  N print  SN Input One

hdu 6198 dfs枚举找规律+矩阵乘法

number number number Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Problem Description We define a sequence  F : ⋅   F0=0,F1=1 ; ⋅   Fn=Fn

每日一题|牛客竞赛|四舍五入|字符串+贪心+模拟

每日一题|四舍五入 四舍五入 心有猛虎,细嗅蔷薇。你好朋友,这里是锅巴的C\C++学习笔记,常言道,不积跬步无以至千里,希望有朝一日我们积累的滴水可以击穿顽石。 四舍五入 题目: 牛牛发明了一种新的四舍五入应用于整数,对个位四舍五入,规则如下 12345->12350 12399->12400 输入描述: 输入一个整数n(0<=n<=109 ) 输出描述: 输出一个整数

深度优先(DFS)和广度优先(BFS)——算法

深度优先 深度优先搜索算法(英语:Depth-First-Search,DFS)是一种用于遍历或搜索树或图的算法。 沿着树的深度遍历树的节点,尽可能深的搜索树的分支,当节点v的所在边都己被探寻过,搜索将回溯到发现节点v的那条边的起始节点。这一过程一直进行到已发现从源节点可达的所有节点为止。如果还存在未被发现的节点,则选择其中一个作为源节点并重复以上过程,整个进程反复进行直到所有节点都被访

C和指针:字符串

字符串、字符和字节 字符串基础 字符串就是一串零个或多个字符,并且以一个位模式为全0的NUL字节结尾。 字符串长度就是字符串中字符数。 size_t strlen( char const *string ); string为指针常量(const修饰string),指向的string是常量不能修改。size_t是无符号数,定义在stddef.h。 #include <stddef.h>

PHP字符串全排列

方法一: $str = 'abc';$a =str_split($str);perm($a, 0, count($a)-1);function perm(&$ar, $k, $m) {if($k == $m){ echo join('',$ar), PHP_EOL;}else {for($i=$k; $i<=$m; $i++) {swap($ar[$k], $ar[$i]);perm($ar