UOJ#121. 【NOI2013】向量内积 随机化算法,矩阵
原文链接www.cnblogs.com/zhouzhendong/UOJ121.html
前言
完蛋了我越来越菜了贺题都不会了。
题解
\(O(n ^ 2 d)\) 暴力送 60 分。
Bitset 优化一下说不定更稳。可能有 85 分。
来讲正解。
注意下文中的 "p" 表示原题中的 "k"。
首先我们来解决一个问题:
如何在较低的复杂度下判定矩阵 A,B,C 是否满足 \(A\times B = C\) 。
做法是:随机 O(1) 个行向量 \(x\),判定 \(x \times A \times B = x\times C\) 是否成立。本题中,行向量大概只需要 10 个左右。
回到本题。
考虑 p = 2 的情况。设矩阵 A 为 \(n \times d\) 的矩阵,且满足 \(A_{i,j} = x_{i,j}\) ,设矩阵 \(B = A ^ {T}\) 。
令
\]
那么,如果无解,则
\begin{bmatrix}
self_1 & 1 & 1 & 1 & \cdots & 1 \\
1 & self_2 & 1 & 1 & \cdots & 1 \\
1 & 1 & self_3 & 1 & \cdots & 1 \\
1 & 1 & 1 & self_4 & \cdots & 1 \\
\vdots & \vdots &\vdots &\vdots &\ddots & \vdots\\
1 & 1 & 1 & 1 & \cdots & self_n
\end{bmatrix}
\]
所以,我们可以利用之前提到的方法在 \(O(nd)\) 的时间复杂度内判定是否有解。如果算出来的行向量的第 \(i\) 项前后不等,那么第 \(i\) 个向量和某一个向量的内积一定模p为0。
当 p = 3 时,考虑到最终矩阵很难确定,因为模 3 意义下可能是 1 或 2。
但是由于 \(1 ^ 2 \equiv 2 ^ 2 \equiv 1 \pmod 3\),所以我们可以考虑取平方。
可以发现
\]
所以我们可以改写矩阵的定义,设矩阵 \(A'\) 为一个由 \([1\cdots n]\) 映射到 \((k_1,k_2) | k_1\in [1\cdots d] , k_2 \in [1\cdots d]\) 的矩阵,矩阵 \(B'\) 为一个由 \((k_1,k_2) | k_1\in [1\cdots d] , k_2 \in [1\cdots d]\) 映射到 \([1\cdots n]\) 的矩阵。
剩下的做法和原先一样。
时间复杂度为 \(O(n d ^ {p-1})\) 。
代码
#include <bits/stdc++.h>
#define clr(x) memset(x,0,sizeof x)
#define For(i,a,b) for (int i=(a);i<=(b);i++)
#define Fod(i,b,a) for (int i=(b);i>=(a);i--)
#define fi first
#define se second
#define pb(x) push_back(x)
#define mp(x,y) make_pair(x,y)
#define outval(x) cerr<<#x" = "<<x<<endl
#define outtag(x) cerr<<"---------------"#x"---------------"<<endl
#define outarr(a,L,R) cerr<<#a"["<<L<<".."<<R<<"] = ";\
For(_x,L,R)cerr<<a[_x]<<" ";cerr<<endl;
using namespace std;
typedef long long LL;
LL read(){
LL x=0,f=0;
char ch=getchar();
while (!isdigit(ch))
f|=ch=='-',ch=getchar();
while (isdigit(ch))
x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
return f?-x:x;
}
const int N=100005,D=105;
int n,d,p;
int x[N][D],y[N];
int a[N][D],b[D][N];
int s[N],t[N],c[D][D];
int self[N];
int Solve(int k){
For(i,1,n)
if (i!=k){
int s=0;
For(j,1,d)
s+=x[i][j]*x[k][j];
s%=p;
if (s==0){
printf("%d %d\n",min(i,k),max(i,k));
return 1;
}
}
return 0;
}
int main(){
n=read(),d=read(),p=read();
For(i,1,n){
For(j,1,d)
x[i][j]=read()%p;
self[i]=0;
For(j,1,d)
self[i]+=x[i][j]*x[i][j];
self[i]%=p;
if (p==3)
self[i]=self[i]*self[i]%p;
}
For(i,1,n)
For(j,1,d)
a[i][j]=b[j][i]=x[i][j];
srand(233);
For(_,1,10){
int sum=0;
For(i,1,n)
y[i]=rand()%5?1:0,sum+=y[i];
if (p==2){
For(i,1,n)
s[i]=y[i];
clr(t);
For(i,1,n)
For(j,1,d)
t[j]+=s[i]*a[i][j];
For(i,1,d)
t[i]%=p;
clr(s);
For(i,1,n)
For(j,1,d)
s[i]+=t[j]*b[j][i];
For(i,1,n)
s[i]%=p;
}
else {
For(i,1,n)
s[i]=y[i];
clr(c);
For(i,1,n)
For(j1,1,d)
For(j2,1,d)
c[j1][j2]+=s[i]*a[i][j1]*a[i][j2];
For(j1,1,d)
For(j2,1,d)
c[j1][j2]%=p;
clr(s);
For(i,1,n)
For(j1,1,d)
For(j2,1,d)
s[i]+=c[j1][j2]*b[j1][i]*b[j2][i];
For(i,1,n)
s[i]%=p;
}
For(i,1,n)
t[i]=(sum-(1-self[i])*y[i])%p;
For(i,1,n)
if (s[i]!=t[i])
return assert(Solve(i)),0;
}
puts("-1 -1");
return 0;
}
UOJ#121. 【NOI2013】向量内积 随机化算法,矩阵的更多相关文章
- 【BZOJ-3243】向量内积 随机化 + 矩阵
3243: [Noi2013]向量内积 Time Limit: 10 Sec Memory Limit: 256 MBSec Special JudgeSubmit: 1249 Solved: ...
- 【fake题解】[NOI2013]向量内积
[fake题解][NOI2013]向量内积 做法1 大暴力.哪里不会T哪里. 做法2 所有数都%=k不影响结果.(废话 k的取值只有2和3,所以肯定是要分类讨论的.k=2肯定简单些啦. k=2 出现的 ...
- [Noi2013]向量内积
来自FallDream的博客,未经允许,请勿转载,谢谢. 两个d 维向量A=[a1,a2,...,ad]与B=[b1,b2,...,bd]的内积为其相对应维度的权值的乘积和,即: $\sum_{i=1 ...
- P1224 [NOI2013]向量内积
传送门 发现这个内积和矩乘有点像,考虑构造一个 $n$ 行 $m$ 列的矩阵 $A$,每一行都是一个题目给定的 $m$ 维向量 设 $B=AA^T$ ,其中 $A^T$ 为 $A$ 的转置矩阵,那么对 ...
- luogu P1224 [NOI2013]向量内积
传送门 挺有意思的一道题 暴力60就是枚举每个向量暴力check,随机选向量就能多骗一些分 然后两个向量内积要模\(k\)为\(0\),那么如果全部不为\(0\)就不合法.先考虑\(k=2\),对于向 ...
- 洛谷 P1224 - [NOI2013] 向量内积(随机化)
洛谷题面传送门 一道很神的随机化. 首先由于我们要求向量点乘 \(\bmod k\) 的值,因此我们可以将所有 \(x_{i,j}\) 都模上 \(k\),显然该操作不影响结果正确性. 注意到这里的 ...
- BZOJ3243 NOI2013向量内积(随机化)
考虑奇技淫巧. 首先是k=2.对向量维护一个前缀和,每次将当前向量与前缀和点乘.如果点乘结果不等于i-1&1,说明当前向量至少和之前的某个向量的数量积是2的倍数,暴力找就可以了.当然等于i-1 ...
- UOJ#75. 【UR #6】智商锁 随机化算法 矩阵树定理
原文链接www.cnblogs.com/zhouzhendong/p/UOJ75.html 前言 根本没想到. 题解 首先我们可以考虑一种做法: 找一些图,使得他们各自的生成树个数乘起来等于 k. 那 ...
- 【uoj121】 NOI2013—向量内积
http://uoj.ac/problem/121 (题目链接) 题意 给出${n}$个${d}$维向量,问是否有两个不同的向量的内积是${k}$的倍数. Solution 又卡了一上午常数,我弃了T ...
随机推荐
- Linux实现MYSQl数据库的定时备份
今天给大家分享一下如何在Linux下实现MYSQl数据库的定时备份. 前提需要保证你的Linux服务器已经安装了MYSQl数据库服务. 1.创建shell脚本 vim backupdb.sh 创建脚本 ...
- java之hibernate之单向的一对多关联映射
这篇主要讲hiberante中的 单向一对多关联映射 1.在应用中,有时候需要从一的一端获取多的一端的数据.比如:查看某个分类下的所有书籍信息:查看某个订单下的所有商品等. 2.在一对多的关联关系中, ...
- java之struts2之文件下载
1.在实际应用开发中,文件下载功能也非常常见. 2.最简单的文件下载方式是通过超链接来进行文件下载: <body> <a href="download/s.txt" ...
- 12.1 Mapping手动创建
只能在index里的field不存在的时候,才能指定新field的数据类型,field有数据后,就不能再修改field的类型了 可创建的类型如下: integer double date text/s ...
- iOS - WWDC18 iOS 自动生成强密码和自动填充验证码/密码
本文将介绍WWDC18 Automatic Strong Passwords and Security Code Autofill和WWDC17 Introducing Password AutoFi ...
- angular异步获取数据后在ngOnInit中无法获取,显示undefined解决办法
两种方法 1 通过*ngif动态加载要数据渲染的dom 2 通过路由导航resolve 第一种感觉太麻烦了,要是一个页面请求多个接口,那就不得不写多个*ngif,本人还是更倾向与第二种发法 具体步骤: ...
- 导航条按钮的设置UIBarButtonItem
1.目的 2.代码 // 设置导航栏的按钮 UIButton *leftNavBtn = [[UIButton alloc] initWithFrame:CGRectMake(0, 0, 35, 35 ...
- IDEA设置项目文件自动Add到Svn/Git
1)配置自动Add 2)将未添加的文件添加到本地 3)取消已经添加的文件
- 修改bash终端命令行颜色
要修改linux终端命令行颜色,我们需要用到PS1,PS1是Linux终端用户的一个环境变量,用来说明命令行提示符的设置.在终端输入命令:#set,即可在输出中找到关于PS1的定义如下: PS1='[ ...
- 解决pycharm新建项目后按钮灰色问题
解决pycharm新建项目后按钮灰色问题 出现过多次该问题了, 在此记录一下 同样适用于导入别人的新项目后无法运行问题 原因一: pycharm没有设置系统解析器 解决方法一: 打开pycharm-& ...