UVA11542 Square(高斯消元 异或方程组)
建立方程组消元,结果为2 ^(自由变元的个数) - 1
采用高斯消元求矩阵的秩
方法一:
#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<string>
#include<algorithm>
#include<map>
#include<queue>
#include<vector>
#include<cmath>
#include<utility>
using namespace std;
typedef long long LL;
const int N = 108, INF = 0x3F3F3F3F;
const double eps = 1e-8;
int a[N][N]; template<typename T>
int gauss_jordan(T A[N][N], int n, int m){
int i, c;
for(i = 0, c = 0; i < n && c < m; i++, c++){
int r = i;
for(int j = i + 1; j < n; j++){
if(A[j][c]){
r = j;
break;
}
}
if(A[r][c] == 0){
i--;
continue;
}
if(r != i){
for(int j = 0; j <= m; j++){
swap(A[r][j], A[i][j]);
}
}
for(int k = 0; k < n; k++){
if(k != i && A[k][c]){
for(int j = m; j >= c; j--){
A[k][j] ^= A[i][j];
}
}
}
}
return i;
} const int MAXN = 508;
int prime[MAXN];
bool vis[MAXN];
int getPrime(int n){//求1~n的素数
int tot=0;
memset(vis,0,sizeof(vis));
for(int i=2;i<=n;i++){
if(!vis[i]){
prime[tot++]=i;
}
for(int j=0;j<tot&&i*prime[j]<=n;j++){
vis[i*prime[j]]=true;
if(i%prime[j]==0){//让每个合数仅被其最小的质数筛去
break;
}
}
}
return tot;
} int main(){
int cnt = getPrime(500);
int t;
cin>>t;
while(t--){
memset(a, 0, sizeof(a));
int n;
cin>>n;
for(int j = 0; j < n; j++){
LL x;
cin>>x;
for(int i = 0; i < cnt && prime[i]<= x; i++){
while(x % prime[i] == 0){
a[i][j] ^= 1;
x /= prime[i];
}
}
}
LL ans = n - gauss_jordan(a, cnt, n);
//cout<<ans<<" ans\n";
cout<<((1ll << ans) - 1)<<'\n';
} return 0;
}
方法2:
消元后非0向量的行数即为矩阵的秩,但开始出现问题一直WA,后来在消元变成上三角矩阵后,从最后一行起,找出第一个非0元素,向上消元。
应该有更巧妙的写法避免这个问题。
#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<string>
#include<algorithm>
#include<map>
#include<queue>
#include<vector>
#include<cmath>
#include<utility>
using namespace std;
typedef long long LL;
const int N = 108, INF = 0x3F3F3F3F;
const double eps = 1e-8;
int a[N][N]; template<typename T>
void gauss_jordan(T A[N][N], int n, int m){
for(int i = 0; i < n; i++){
int r = i;
for(int j = i + 1; j < n; j++){
if(A[j][i]){
r = j;
break;
}
}
if(A[r][i] == 0){
continue;
}
if(r != i){
for(int j = 0; j <= m; j++){
swap(A[r][j], A[i][j]);
}
}
for(int k = i + 1; k < n; k++){
if(k != i && A[k][i]){
for(int j = m; j >= i; j--){
A[k][j] ^= A[i][j];
}
}
}
}
for(int i = n - 1; i > 0; i--){
for(int j = 0; j < m; j++){
if(A[i][j]){
for(int k = i - 1; k >= 0; k--){
if(A[k][j]){
for(int l = j; l <= m; l++){
A[k][l] ^= A[i][l];
}
}
}
break;
}
}
} } const int MAXN = 508;
int prime[MAXN];
bool vis[MAXN];
int getPrime(int n){//求1~n的素数
int tot=0;
memset(vis,0,sizeof(vis));
for(int i=2;i<=n;i++){
if(!vis[i]){
prime[tot++]=i;
}
for(int j=0;j<tot&&i*prime[j]<=n;j++){
vis[i*prime[j]]=true;
if(i%prime[j]==0){//让每个合数仅被其最小的质数筛去
break;
}
}
}
return tot;
} int main(){
int cnt = getPrime(500);
int t;
cin>>t;
while(t--){
memset(a, 0, sizeof(a));
int n;
int row = 0;
cin>>n;
for(int j = 0; j < n; j++){
LL x;
cin>>x;
for(int i = 0; i < cnt && prime[i]<= x; i++){
while(x % prime[i] == 0){
row = max(row, i);
a[i][j] ^= 1;
x /= prime[i];
}
}
}
row++;
gauss_jordan(a, row, n);
int rk = 0;
for(int i = 0; i < row; i++){
for(int j = 0; j < n; j++){
if(a[i][j]){
rk++;
break;
}
}
}
n -= rk; cout<<((1ll << n) - 1)<<'\n';
} return 0;
}
UVA11542 Square(高斯消元 异或方程组)的更多相关文章
- UVA 11542 Square 高斯消元 异或方程组求解
题目链接:点击打开链接 白书的例题练练手. . . P161 #include <cstdio> #include <iostream> #include <algori ...
- BZOJ.1923.[SDOI2010]外星千足虫(高斯消元 异或方程组 bitset)
题目链接 m个方程,n个未知量,求解异或方程组. 复杂度比较高,需要借助bitset压位. 感觉自己以前写的(异或)高斯消元是假的..而且黄学长的写法都不需要回代. //1100kb 324ms #i ...
- Tsinsen-A1488 : 魔法波【高斯消元+异或方程组】
高斯消元. 自己只能想出来把每一个点看成一个变量,用Xi表示其状态,这样必定TLE,n^2 个变量,再加上3次方的高斯消元(当然,可以用bitset压位). 正解如下: 我们把地图划分成一个个的横条和 ...
- UVa 11542 (高斯消元 异或方程组) Square
书上分析的太清楚,我都懒得写题解了.=_=|| #include <cstdio> #include <cstring> #include <cmath> #inc ...
- POJ.1830.开关问题(高斯消元 异或方程组)
题目链接 显然我们需要使每个i满足\[( ∑_{j} X[j]*A[i][j] ) mod\ 2 = B[i]\] 求这个方程自由元Xi的个数ans,那么方案数便是\(2^{ans}\) %2可以用^ ...
- 【高斯消元解xor方程组】BZOJ2466-[中山市选2009]树
[题目大意] 给出一棵树,初始状态均为0,每反转一个节点的状态,相邻的节点(父亲或儿子)也会反转,问要使状态均为1,至少操作几次? [思路] 一场大暴雨即将来临,白昼恍如黑夜!happy! 和POJ1 ...
- poj1830(高斯消元解mod2方程组)
题目链接:http://poj.org/problem?id=1830 题意:中文题诶- 思路:高斯消元解 mod2 方程组 有 n 个变元,根据给出的条件列 n 个方程组,初始状态和终止状态不同的位 ...
- POJ 1222 EXTENDED LIGHTS OUT(高斯消元解XOR方程组)
http://poj.org/problem?id=1222 题意:现在有5*6的开关,1表示亮,0表示灭,按下一个开关后,它上下左右的灯泡会改变亮灭状态,要怎么按使得灯泡全部处于灭状态,输出方案,1 ...
- UVA 11542 - Square(高斯消元)
UVA 11542 - Square 题目链接 题意:给定一些数字.保证这些数字质因子不会超过500,求这些数字中选出几个,乘积为全然平方数,问有几种选法 思路:对每一个数字分解成质因子后.发现假设要 ...
随机推荐
- python——tuple元组
>>> dir(tuple) ['__add__', '__class__', '__contains__', '__delattr__', '__doc__', '__eq__', ...
- idea让字体更圆滑
个人用myeclipse习惯了, 转战idea 觉得自己看题来很方很死板的样子. 然后更换了其字体,如果喜欢你可以试试. 点击file --> setting.. --> 把编辑的字体改成 ...
- NDK学习4: Eclipse HelloWorld
NDK学习4: Eclipse HelloWorld 1.配置Eclipse NDK环境 Window->preferences->android->ndk 2.新建Andro ...
- JQuery 遍历 - prev() 方法
http://www.w3school.com.cn/jquery/traversing_prev.asp http://www.w3school.com.cn/jquery/jquery_ref_t ...
- IPC---信号量
一.什么是信号量 信号量的使用主要是用来保护共享资源,使得资源在一个时刻只有一个进程(线程) 所拥有. 信号量的值为正的时候,说明它空闲.所测试的线程可以锁定而使用它.若为0,说明 它被占用,测试的线 ...
- BIOS开启虚拟化
启动时根据提示按del 键按 F10 键以配置 BIOS使用箭头键滚动到“System Configuration”选择“Virtualization Technology”,然后按 Enter 键选 ...
- POJ 2367 (裸拓扑排序)
http://poj.org/problem?id=2367 题意:给你n个数,从第一个数到第n个数,每一行的数字代表排在这个行数的后面的数字,直到0. 这是一个特别裸的拓扑排序的一个题目,拓扑排序我 ...
- Python之virtualenv安装
CentOS 7 yum install python-virtualenv virtualenv --no-site-packages testenv #不依赖真实环境的packages用 --no ...
- centos7安装docker并设置开机启动
版本要求:查看内核版本,需大于3.10 [root@localhost ~]# uname -r -.el7.x86_64 更新内核:如果是生产机器务必慎重更新内核,避免出现不必要的问题. sudo ...
- ios 在程序中使用iCloud
注意,这里说的使用icould不是用icloud进行系统备份,那个功能不需要我们写代码,备份到icloud的东西我们也不能操作.我们指的是以下这3种icloud使用方法: 这里有3中使用方法, Key ...