Problem Description

度度熊为了完成毕业论文,需要收集一些数据来支撑他的论据,于是设计了一份包含 mm 个问题的调查问卷,每个问题只有 'A' 和 'B' 两种选项。

将问卷散发出去之后,度度熊收到了 nn 份互不相同的问卷,在整理结果的时候,他发现可以只保留其中的一部分问题,使得这 nn 份问卷仍然是互不相同的。这里认为两张问卷是不同的,当且仅当存在至少一个被保留的问题在这两份问卷中的回答不同。

现在度度熊想知道,存在多少个问题集合,使得这 nn 份问卷在只保留这个集合的问题之后至少有 kk 对问卷是不同的。

Input

第一行包含一个整数 TT,表示有 TT 组测试数据。

接下来依次描述 TT 组测试数据。对于每组测试数据:

第一行包含三个整数 nn,mm 和 kk,含义同题目描述。

接下来 nn 行,每行包含一个长度为 mm 的只包含 'A' 和 'B' 的字符串,表示这份问卷对每个问题的回答。

保证 1 \leq T \leq 1001≤T≤100,1 \leq n \leq 10^31≤n≤10​3​​,1 \leq m \leq 101≤m≤10,1 \leq k \leq 10^61≤k≤10​6​​,给定的 nn 份问卷互不相同。

Output

对于每组测试数据,输出一行信息 "Case #x: y"(不含引号),其中 x 表示这是第 xx 组测试数据,y 表示满足条件的问题集合的个数,行末不要有多余空格。

Sample Input
2
2 2 1
AA
BB
2 2 2
AA
BB
Sample Output
Copy

Case #1: 3
Case #2: 0

参考博客:https://blog.csdn.net/nuclear_physis/article/details/81414833

分析:我们将A-B的问题序列转化成为0-1二进制序列从而转为整数特征,而问题子集情况也可以使用0-1进制序列表示。通过位运算就可以表示问题子集对应的问题结果0-1序列,进而转为整数特征。先将n个问卷的问题特征值排序。排序后就可以清晰得到相同问题问卷数目,比如有这样的排序结果[1 2 2 3 3],那么问卷2 和 3的问题是一样的,问卷 4 和 5 的是一样的。通过相同问题数目就可以很容易得到相同问题问卷的对数。我们想如果将总的问题对数减去相同的问题的问卷对数那么就得到了不一样问卷的对数。上面的例子就是5 * 4 / 2 - 2 * 1 / 2 - 2 * 1 / 2 等于8。

AC代码:

#include <map>
#include <set>
#include <stack>
#include <cmath>
#include <queue>
#include <cstdio>
#include <vector>
#include <string>
#include <bitset>
#include <cstring>
#include <iomanip>
#include <iostream>
#include <algorithm>
#define ls (r<<1)
#define rs (r<<1|1)
#define debug(a) cout << #a << " " << a << endl
using namespace std;
typedef long long ll;
const ll maxn = 2*1e3+10;
const ll mod = 998244353;
ll n, m, k, ans;
ll a[maxn], b[maxn];
bool ok( ll x ) {
for( ll i = 1; i <= n; i ++ ) {
b[i] = a[i]&x;
}
map<ll,ll> mp;
for( ll i = 1; i <= n; i ++ ) { //二进制整数值相容代表调查问卷相同
mp[b[i]] ++;
}
ll total = n*(n-1)/2;
for( auto it : mp ) {
if( it.second > 1 ) {
total = total - it.second*(it.second-1)/2;
}
}
if( total >= k ) {
return true;
} else {
return false;
}
}
int main() {
ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
ll T, t = 1;
scanf("%lld",&T);
while( T -- ) {
scanf("%lld%lld%lld",&n,&m,&k);
for( ll i = 1; i <= n; i ++ ) {
char str[maxn];
scanf("%s",str);
for( ll i = 0; i < strlen(str); i ++ ) {
if( str[i] == 'A' ) {
str[i] = '0';
} else {
str[i] = '1';
}
}
bitset<12> bs(str);
a[i] = bs.to_ulong();
}
ans = 0;
for( ll i = 1; i <= (1<<m)-1; i ++ ) { //问题的所有子集转化为求每个子集对应的二进制整数
if( ok(i) ) {
ans ++;
}
}
printf("Case #%lld: %lld\n",t++,ans);
}
return 0;
}

  

  

最新文章

  1. jQuery专题
  2. Servlet-RequestDispatcher.forward方法
  3. Light OJ 1364 Expected Cards (期望dp,好题)
  4. 使用git submodule管理一个需要多个分立开发或者第三方repo的项目
  5. QString转换成LPCWSTR
  6. memcached学习——大纲简介 &amp;&amp; 安装(基于centos6.5)、启动、关闭memcached(一)
  7. JAVA 计算地球上任意两点(经纬度)距离
  8. PhpSrom安装xdebug
  9. javaweb之session过期验证
  10. docker - 容器里安装ssh
  11. vue使用tips记录
  12. asp.net json,对象,字符串的相互转换
  13. red hat防火墙的开启与关闭及状态查看方法
  14. 图-&gt;存储结构-&gt;邻接表
  15. MySQL分析数据运行状态利器【SHOW PROCESSLIST】
  16. 本地推送UILocalNotification的一些简单方法
  17. 【JVM】2、JVM调优总结
  18. 堆(Heap)
  19. Filters: before, after, around, 常用到before_action,执行顺序是从外到内。
  20. BigDecimal 两种方式

热门文章

  1. window下不用安装虚拟机,也可以玩转linux,玩转最新redis
  2. C++ protobuffer 前后端通信 简单应用
  3. Cell Phone Networ (树形dp-最小支配集)
  4. CountDownLatch实现多线程并发请求
  5. django数据库事务
  6. [ubuntu][deepin]系统增加自定义开机启动项
  7. 消息中间件和JMS介绍(一)
  8. IOS7.0唯一“设备ID”的获取方法
  9. 【Kubernetes 系列五】在 AWS 中使用 Kubernetes:EKS
  10. bottombar——Fragment