[Noi2013]向量内积
来自FallDream的博客,未经允许,请勿转载,谢谢。
两个d 维向量A=[a1,a2,...,ad]与B=[b1,b2,...,bd]的内积为其相对应维度的权值的乘积和,即:
$\sum_{i=1}^{d}ai*bi$
现有 n 个d 维向量x1,...,xn ,小喵喵想知道是否存在两个向量的内积为k的倍数。请帮助她解决这个问题
k=2时 n<=20000 d<=100 k=3时n<=1000,d<=100 或者n<=100000 d<=30
把两个向量内积看作矩阵一个1*d的矩阵和d*1的矩阵相乘
那么k=2的时候,就是一个n*d的矩阵和一个d*n的矩阵相乘,问得到的n*n的矩阵除了主对角线之外有没有0(mod k)
直接计算复杂度n^2d 所以不考虑计算它,而是转而判断得到的矩阵是否等于我想要的矩阵
假设n*d的矩阵为A,把它倒过来之后得到的d*n的矩阵为B,C=A*B,我期望的矩阵为D
矩阵D的主对角线的值可以O(nd)计算,其他值显然是1。
要判断C是否等于D,我可以转而随机一个n*1的向量T,判断C*T是否等于D*T即可
因为T只存在0和1,所以D*T可快速计算 而计算A*B*T时,我们可以先计算B*T,再计算A*(B*T),这两次计算都是O(nd)的
那么就做完了。复杂度O(nd)
但是k=3的时候,无法直接知道期望的矩阵,但是发现2^2=1(mod 3)
于是考虑计算E,Eij=Cij^2;矩阵D主对角线也对应平方一下
发现$Eij=(\sum Aik*Bkj)^{2}$ 展开后得到$Eij=\sum_{k1=1}^{d}\sum_{k2=1}^{d}Aik1*Bk1j*Aik2*Bk2j$
这个可以看作$n*d^{2}$的矩阵和一个$d^{2}*n$的矩阵相乘 之后就按照k=2的做法就行了 复杂度$O(nd^{2})$
但是这个办法并不能保证正确 如果直接让矩阵T都是1的话 在bzoj会被叉...
所以我们多随机几次就行了 一般都能过的吧
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define rint register int
#define getchar() (*S++)
char BB[<<],*S=BB;
using namespace std;
inline int read()
{
int x = ; char ch = getchar();
while(ch < '' || ch > '') ch = getchar();
while(ch >= '' && ch <= ''){x = x * + ch - '';ch = getchar();}
return x;
} int A[][],B[],C[],D[],E[],n,d,K,cnt=,sum=,id[][]; inline bool check(int a,int b)
{
int Sum=;
for(rint i=;i<=d;++i)
Sum+=A[a][i]*A[b][i];
return Sum%K==;
} inline void Solve(int x)
{
for(rint i=;i<=n;++i)
if(i!=x&&check(i,x))
{printf("%d %d\n",min(x,i),max(x,i));return;}
} int main()
{
fread(BB,,<<,stdin);
srand(23333U);
n=read();d=read();K=read();
for(rint i=;i<=n;++i)
for(rint j=;j<=d;++j)
A[i][j]=read()%K;
if(K==) for(int It=;It<=;++It)
{
for(rint i=;i<=n;++i) C[i]=rand()&,sum+=C[i];
for(rint i=;i<=n;++i)
for(rint j=;j<=d;++j)
B[j]+=A[i][j]*C[i];
for(rint i=;i<=d;++i) B[i]%=K;
for(rint i=;i<=n;++i)
for(rint j=;j<=d;++j)
D[i]+=A[i][j]*B[j];
for(rint i=;i<=n;++i)
for(rint j=;j<=d;++j)
E[i]+=A[i][j];
for(rint i=;i<=n;++i) E[i]=(E[i]*C[i]+sum-C[i])%K;
for(rint i=;i<=n;++i) if(E[i]!=(D[i]%K)) return Solve(i),;
sum=;
for(rint i=;i<=n;++i) E[i]=D[i]=B[i]=;
}
else for(int It=;It<=;++It)
{
for(rint i=;i<=n;++i) C[i]=rand()%,sum+=C[i];
for(rint i=;i<=d;++i)
for(rint j=;j<=d;++j)
id[i][j]=++cnt;
for(rint k=;k<=n;++k)
for(rint i=;i<=d;++i)
for(rint j=;j<=d;++j)
B[id[i][j]]+=A[k][i]*A[k][j]*C[k];
for(rint i=;i<=cnt;i++) B[i]%=K;
for(rint i=;i<=n;++i)
for(rint j=;j<=d;++j)
for(rint k=;k<=d;++k)
D[i]+=A[i][j]*A[i][k]*B[id[j][k]];
for(rint i=;i<=n;++i)
for(rint j=;j<=d;++j)
E[i]+=A[i][j]*A[i][j];
for(rint i=;i<=n;++i) E[i]%=K,E[i]=(E[i]*E[i]*C[i]+sum-C[i])%K;
for(rint i=;i<=n;++i) if(E[i]!=(D[i]%K)) return Solve(i),;
sum=;
for(rint i=;i<=n;++i) E[i]=D[i]=B[i]=;
}
return *puts("-1 -1");
}
[Noi2013]向量内积的更多相关文章
- 【fake题解】[NOI2013]向量内积
[fake题解][NOI2013]向量内积 做法1 大暴力.哪里不会T哪里. 做法2 所有数都%=k不影响结果.(废话 k的取值只有2和3,所以肯定是要分类讨论的.k=2肯定简单些啦. k=2 出现的 ...
- P1224 [NOI2013]向量内积
传送门 发现这个内积和矩乘有点像,考虑构造一个 $n$ 行 $m$ 列的矩阵 $A$,每一行都是一个题目给定的 $m$ 维向量 设 $B=AA^T$ ,其中 $A^T$ 为 $A$ 的转置矩阵,那么对 ...
- luogu P1224 [NOI2013]向量内积
传送门 挺有意思的一道题 暴力60就是枚举每个向量暴力check,随机选向量就能多骗一些分 然后两个向量内积要模\(k\)为\(0\),那么如果全部不为\(0\)就不合法.先考虑\(k=2\),对于向 ...
- 3243: [Noi2013]向量内积 - BZOJ
Description 两个d 维向量A=[a1,a2,...,ad]与B=[b1,b2,...,bd]的内积为其相对应维度的权值的乘积和,即: 现有 n 个d 维向量x1,...,xn ,小喵喵想知 ...
- 【uoj121】 NOI2013—向量内积
http://uoj.ac/problem/121 (题目链接) 题意 给出${n}$个${d}$维向量,问是否有两个不同的向量的内积是${k}$的倍数. Solution 又卡了一上午常数,我弃了T ...
- bzoj 3243: [Noi2013]向量内积
Description 两个d 维向量A=[a1,a2,...,ad]与B=[b1,b2,...,bd]的内积为其相对应维度的权值的乘积和,即: 现有 n 个d 维向量x1,...,xn ,小喵喵想知 ...
- BZOJ3243/UOJ121 [Noi2013]向量内积
本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/ ...
- BZOJ3243 NOI2013向量内积(随机化)
考虑奇技淫巧. 首先是k=2.对向量维护一个前缀和,每次将当前向量与前缀和点乘.如果点乘结果不等于i-1&1,说明当前向量至少和之前的某个向量的数量积是2的倍数,暴力找就可以了.当然等于i-1 ...
- BZOJ3243 [Noi2013]向量内积 【乱搞】
题目链接 BZOJ3243 题解 模数只有\(2\)或\(3\),可以大力讨论 如果模数为\(2\),乘积结果只有\(1\)或\(0\) 如果一个向量和前面所有向量乘积都为\(1\),那么其和前面向量 ...
随机推荐
- 201621123031 《Java程序设计》第2周学习总结
Week02-Java基本语法与类库 1. 本周学习总结 本周讲了Java的基本数据类型,主要分为八类(byte,short,int,long,double,float,char,boolean),其 ...
- sys模块的使用
import sys,time ''' if sys.argv[1]=='sleepy': print('nongsi') else: print('....')''' #进度条 for i in r ...
- DOM相关知识
一.查找元素 间接查找 parentNode // 父节点 childNodes // 所有子节点 firstChild // 第一个子节点 lastChild // 最后一个子节点 nextSibl ...
- salesforce零基础学习(八十七)Apex 中Picklist类型通过Control 字段值获取Dependent List 值
注:本篇解决方案内容实现转自:http://mysalesforceescapade.blogspot.com/2015/03/getting-dependent-picklist-values-fr ...
- Python内置函数(51)——hasattr
英文文档: hasattr(object, name) The arguments are an object and a string. The result is True if the stri ...
- 【原创】公司各个阶段 CTO 需要做什么?(上篇)
CTO 是企业内技术最高负责人,对企业的发展起到至关重要的作用.但随着公司的不断发展,CTO 的工作重心也会不断变化.只有在正确的阶段做正确的事,才能更好地为公司做出贡献.我是空中金融 CTO ,TG ...
- gradle入门(1-7)eclipse和gradle集成插件的安装和使用
一.安装gradle插件:buildship 1.安装插件 gradle默认的本地缓存库在c盘user目录下的.gradle文件夹下,安装好gradle后,可以添加环境变量GRADLE_USER_HO ...
- Django之views系统
Django的View(视图)简介 一个视图函数(类),简称视图,是一个简单的Python 函数(类),它接受Web请求并且返回Web响应. 响应可以是一张网页的HTML内容,一个重定向,一个404错 ...
- CWMP开源代码研究番外篇——博通方案
声明:本篇文章来自于某公司Cable Modem产品的文档资料,源码来自于博通公司,只提供参考(为保护产权,本人没有源码). 前文曾提到会写一篇关于博通的tr069,那么福利来了.福利,福利,福利,重 ...
- 小技巧-ASP.Net session保存在数据库服务器
引用博客:http://www.cnblogs.com/lykbk/archive/2013/01/13/hf576856868.html web Form 网页是基于HTTP的,它们没有状态, 这意 ...