CF895C: Square Subsets && 【BZOJ2844】albus就是要第一个出场

这两道题很类似,都是线性基的计数问题,解题的核心思想也一样。

CF895C Square Subsets

题目链接

题意

给定\(n\)个数,求多少种选数方案使得选出来的数乘积为完全平方数。\(n\leq 100000,a_i\leq70\)。

完全平方数的本质就是每个质因子的次数为偶数。

所以我们将每一个数唯一分解,然后记录每个质因子的奇偶状态,就得到了一个个01串。问题就变成了有多少个集合中的数异或为0。

我们建好线性基,设线性基的秩为\(m\),则答案就是\(2^{n-m}-1\)。再判线性基能否组成0,如果能,答案再\(+1\)

因为线性基中的\(m\)个元素是线性无关的,并且可以表达出其他数的所有线性组合,所以其他\(2^{n-m}-1\)(减掉都不选的一种情况)的所有数都能在线性基中找到唯一的一个组合与之异或起来为0。

代码:

#include<bits/stdc++.h>
#define ll long long using namespace std; int n;
ll p[50],cnt;
bool Insert(ll x) {
for(int i=20;i>=0;i--) {
if(!(x>>i)&1) continue ;
if(!p[i]) return p[i]=x,1;
x^=p[i];
}
return 0;
}
ll pri[100];
bool vis[100];
void pre() {
for(int i=2;i<=70;i++) {
if(!vis[i]) pri[++pri[0]]=i;
for(int j=1;j<=pri[0]&&i*pri[j]<=70;j++) {
vis[i*pri[j]]=1;
if(i%pri[j]==0) break;
}
}
}
void break_down(ll v) {
ll x=0;
for(int i=1;i<=pri[0];i++) {
while(v%pri[i]==0) {
x^=1<<i;
v/=pri[i];
}
}
if(!Insert(x)) cnt++;
}
int main() {
pre();
scanf("%d",&n);
ll a;
for(int i=1;i<=n;i++) {
scanf("%lld",&a);
break_down(a);
}
ll t=2,ans=1;
for(;cnt;cnt>>=1,t=t*t%1000000007) {
if(cnt&1)ans=ans*t%1000000007;
}
cout<<(ans-1+1000000007)%1000000007;
return 0;
}

【BZOJ2844】albus就是要第一个出场

题目链接

我们就是要求比\(Q\)小的集合有多少个。

和上一道题一样,我们就是要求比\(Q\)小,且能被线性基表示出来的数有多少个,我们设为\(lower\),则答案为\(lower\cdot 2^{n-m}+1\)。

问题其实就是求\(\leq Q-1\)的的最大的数的\(Rank\)。

我的实现方式不太一样。我们就找\(\leq Q-1\)的数的数量。\(\leq Q-1\)一定是从高到低的前\(i-1\)位相同,第\(i\)位小一些,后面的几位随便。于是我们就考虑\(Q\)的前\(i\)位,我们设为\(Q'\),然后将所有数的前\(i\)位建线性基。问题就变成求有多少种方法能组合出\(Q'\)。

这样做复杂度要比网上的一般写法多一个\(log\),不过个人感觉好理解些。

代码:

#include<bits/stdc++.h>
#define ll long long
#define N 100005
#define mod 10086 using namespace std;
inline int Get() {int x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9') {if(ch=='-') f=-1;ch=getchar();}while('0'<=ch&&ch<='9') {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}return x*f;} int n;
int p[35],a[N];
int Q;
bool Insert(int a) {
for(int i=30;i>=0;i--) {
if(a&(1<<i)) {
if(!p[i]) return p[i]=a,1;
else a^=p[i];
}
}
return 0;
}
int cnt;
void build(int k) {
cnt=0;
memset(p,0,sizeof(p));
for(int i=1;i<=n;i++) {
cnt+=Insert(a[i]>>k);
}
}
ll ans;
ll pw[N]; int main() {
n=Get();
pw[0]=1;
for(int i=1;i<=n;i++) pw[i]=(pw[i-1]<<1)%mod;
for(int i=1;i<=n;i++) a[i]=Get();
Q=Get();
for(int i=30;i>=0;i--) {
if(Q&(1<<i)) {
build(i);
if(!Insert((Q>>i)^1)) {
(ans+=pw[n-cnt])%=mod;
}
}
}
cout<<(ans+1)%mod;
return 0;
}

CF895C: Square Subsets && 【BZOJ2844】albus就是要第一个出场的更多相关文章

  1. BZOJ2844: albus就是要第一个出场

    Description 已知一个长度为n的正整数序列A(下标从1开始), 令 S = { x | 1 <= x <= n }, S 的幂集2^S定义为S 所有子集构成的集合. 定义映射 f ...

  2. BZOJ2844: albus就是要第一个出场(线性基)

    Time Limit: 6 Sec  Memory Limit: 128 MBSubmit: 2054  Solved: 850[Submit][Status][Discuss] Descriptio ...

  3. bzoj千题计划195:bzoj2844: albus就是要第一个出场

    http://www.lydsy.com/JudgeOnline/problem.php?id=2844 题意:给定 n个数,把它的所有子集(可以为空)的异或值从小到大排序得到序列 B,请问 Q 在  ...

  4. 【贪心】【线性基】bzoj2844 albus就是要第一个出场

    引用题解:http://blog.csdn.net/PoPoQQQ/article/details/39829237 注意评论区. #include<cstdio> using names ...

  5. 【线性基】bzoj2844: albus就是要第一个出场

    线性基求可重rank 题目描述 给定 n 个数 $\{ a_i \}$ ,以及数 $x$. 将 $\{ a_i \}$​ 的所有子集(包括空集)的异或值从小到大排序,得到 $\{ b_i \} $. ...

  6. 【BZOJ2844】albus就是要第一个出场 高斯消元求线性基

    [BZOJ2844]albus就是要第一个出场 Description 已知一个长度为n的正整数序列A(下标从1开始), 令 S = { x | 1 <= x <= n }, S 的幂集2 ...

  7. BZOJ 2844: albus就是要第一个出场 [高斯消元XOR 线性基]

    2844: albus就是要第一个出场 题意:给定一个n个数的集合S和一个数x,求x在S的$2^n$个子集从小到大的异或和序列中最早出现的位置 一开始看错题了...人家要求的是x第一次出现位置不是第x ...

  8. BZOJ 2844: albus就是要第一个出场

    2844: albus就是要第一个出场 Time Limit: 6 Sec  Memory Limit: 128 MBSubmit: 1134  Solved: 481[Submit][Status] ...

  9. 2844: albus就是要第一个出场

    2844: albus就是要第一个出场 链接 分析: 和HDU3949差不多互逆,这里需要加上相同的数. 结论:所有数任意异或,构成的数出现一样的次数,次数为$2^{n-cnt}$,cnt为线性基的大 ...

随机推荐

  1. 三、TortoiseGit之配置密钥

    TortoiseGit使用扩展名为ppk的密钥,而不是ssh-keygen生成的rsa密钥. 也就是说使用 ssh-keygen  -t rsa  -C "username@email.co ...

  2. JS判断是否是数组的四种做法

    一.前言 如何判断一个对象或一个值是否是一个数组,在面试或工作中我们常常会遇到这个问题,既然出现频率高,想着还是做个整理,那么本文主要基于几种判断方式,以及方式判断的原理,是否存在问题展开讨论. 二. ...

  3. ios --键盘监听JYKeyBoardListener

    没有前言,就是一个简单的键盘监听,自动调整输入框的位置不被键盘遮挡 .h // // JYKeyBoardListener.h // // Created by JianF.Sun on 17/9/2 ...

  4. MyBatis学习笔记(一) 概述

    一.什么是MyBatis? MyBatis 是一款优秀的持久层框架,它支持定制化 SQL.存储过程以及高级映射.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyBat ...

  5. HDU6205

    card card card Time Limit: 8000/4000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)To ...

  6. 10折交叉验证(10-fold Cross Validation)与留一法(Leave-One-Out)、分层采样(Stratification)

    10折交叉验证 我们构建一个分类器,输入为运动员的身高.体重,输出为其从事的体育项目-体操.田径或篮球. 一旦构建了分类器,我们就可能有兴趣回答类似下述的问题: . 该分类器的精确率怎么样? . 该分 ...

  7. blfs(systemv版本)学习笔记-wget的安装与配置

    我的邮箱地址:zytrenren@163.com欢迎大家交流学习纠错! blfs wget项目地址:http://www.linuxfromscratch.org/blfs/view/8.3/basi ...

  8. 2018-02-04 AppleScript类自然语言与非英语语法设计

    最早知晓是之前C#中文版的github讨论里提到了AppleScript有多语言版. 昨天想起, 觉得它毕竟是为数不多(仅有的?)大公司开发的非英语语法的编程语言, 不禁好奇它的前世今生. 于是作了一 ...

  9. 【代码笔记】Web-HTML-CSS

    一,效果图. 二,代码. <!DOCTYPE html> <html> <head> <meta charset="utf-8"> ...

  10. CSS盒模型的介绍

    CSS盒模型的概念与分类      CSS盒模型就是一个盒子,封装周围的HTML元素,它包括内容content.边框border.内边距padding.外边距margin. CSS盒模型分为标准模型和 ...