[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\),那么其和前面向量 ...
随机推荐
- Python多线程案例
from time import ctime,sleep import threading def music(): for i in range(2): print ("I was lis ...
- 利用python实现简单随机验证码
#!/usr/bin/env python # -*- coding:utf-8 -*- import random temp ='' for i in range(6): num = random. ...
- Andrew Ng机器学习第一章——单变量线性回归
监督学习算法工作流程 h代表假设函数,h是一个引导x得到y的函数 如何表示h函数是监督学习的关键问题 线性回归:h函数是一个线性函数 代价函数 在线性回归问题中,常常需要解决最小化问题.代价函数常用平 ...
- C语言学习(二)
今天在程序员面试题中,碰到一个有意思的题目:数组a[N],存放了1至N-1个数,其中某个数重复一次,现在要求找出重复的数字且程序时间复杂度必须为O(N).乍一看,如果不计时间复杂度和空间复杂度程序比较 ...
- Dojo API中文 Dojo内容模块概览,初学者
官网:http://dojotoolkit.org/reference-guide/1.10/dojo/index.html#dojo-dojo的翻译 dojo 内容: dojo dojo/dojo ...
- Spring知识点回顾(05)bean的初始化和销毁
Java配置方式:@Bean @InitMethod @destroyMethod xml配置方式:init-method,destroy-method 注解方式:@PostConstruct,@Pr ...
- maven编译时出现读取XXX时出错invalid LOC header (bad signature)
问题原因 该包没有下载正确. 解决办法 找到该包的目录,删除该包重新下载即可. 重新下载后用maven test一下,红叉消失.
- shuffle和sort分析
MapReduce中的Shuffle和Sort分析 MapReduce 是现今一个非常流行的分布式计算框架,它被设计用于并行计算海量数据.第一个提出该技术框架的是Google 公司,而Google 的 ...
- React-redux使用中有关Provider问题
先上错误: Warning: Failed prop type: Invalid prop `children` of type `array` supplied to `Provider`, exp ...
- [论文阅读] A Discriminative Feature Learning Approach for Deep Face Recognition (Center Loss)
原文: A Discriminative Feature Learning Approach for Deep Face Recognition 用于人脸识别的center loss. 1)同时学习每 ...