@desription@

给定一个 R * C 表示高度的矩阵 A,另一个 H * W 的矩阵 B,与一个坐标 (x, y)。

这个 R * C 的矩阵中的一个 H * W 的子矩阵,记这个子矩阵中某一个格子的差异值为 = (该方格相对于该子矩阵中的 (x, y) 的高度 - B 中对应方格的数值差)^2。

该子矩阵的差异值为所有格子的差异值之和。

求差异值前 K 小的子矩阵。输出其左上角坐标。

input

第一行输入两个整数 R,C(1 <= R, C <= 666)。

接下来 R 行每行 C 个数字,描述了矩阵 A。

接下来输入两个整数 H,W(1 <= H <= (R+1)/2,1 <= W <= (C+1)/2)

接下来 H 行每行 W 个数字,描述了矩阵 B。

接下来输入一个坐标 (x, y)。

最后输入一个整数 K(1 <= K <= 1000)。

保证所有矩阵元素的绝对值 <= 30。

output

输出 K 行,每行三个整数 (br, bc, s),表示你选择的子矩阵左上角为 (br, bc),其差异值为 s。

如果 s 相同,取 br 较小的;如果 br 相同,取 bc 较小的。

sample input

5 8

6 1 8 6 0 0 8 1

6 5 8 6 8 6 8 1

6 5 8 6 8 6 8 2

6 5 8 6 8 6 8 2

6 5 8 6 8 6 0 2

3 4

-1 2 2 2

-1 2 0 2

-1 2 2 2

2 3

4

sample output

2 2 8

3 2 8

2 4 11

3 4 75

@solution@

不妨去求解每一个左上角对应的差异值。我们考虑以 (i, j) 作为左上角时,差异值的表达式。

\[\sum_{p=1}^{H}\sum_{q=1}^{W}(a[i+p-1][j+q-1] - a[i+x-1][j+y-1] - b[p][q])^2
\]

拆开可可以得到,它是由 6 部分组成:

第一部分:\(H*W*a[i+x-1][j+y-1]^2\)。

第二部分:\(-2*a[i+x-1][j+y-1]*\sum_{p=1}^{H}\sum_{q=1}^{W}a[i+p-1][j+q-1]\)

第三部分:\(2*a[i+x-1][j+y-1]*\sum_{p=1}^{H}\sum_{q=1}^{W}b[p][q]\)

第四部分:\(\sum_{p=1}^{H}\sum_{q=1}^{W}a[i+p-1][j+q-1]^2\)

第五部分:\(\sum_{p=1}^{H}\sum_{q=1}^{W}b[p][q]^2\)

第六部分:\(-2*\sum_{p=1}^{H}\sum_{q=1}^{W}a[i+p-1][j+q-1]*b[p][q]\)

第 1 个可以 O(1) 算。第 2, 3, 4, 5 个可以做个前缀和然后 O(1) 算。

考虑第 6 个。

令 \(c[i][j] = -2*\sum_{p=1}^{H}\sum_{q=1}^{W}a[i+p-1][j+q-1]*b[p][q]\)。

可以发现它非常的像卷积,但是不是 a 的下标与 b 的下标相加等于 c 的下标,而是 c 的下标与 b 的下标相加得到 a 的下标 - 1。

通过对 a 的翻转,移位,并在最后对 c 进行翻转,得到一个转换过后的问题:

\[c[i+p][j+q] = \sum_{p=1}^{H}\sum_{q=1}^{W}a[i][j]*b[p][q]
\]

令多项式 \(A = \sum a[i][j]*x^i*y^j\),同理得到 \(B, C\)。则 \(C = A*B\)。

这个问题虽然是卷积,但是有两个变元。不能用一般的 FFT 直接解决。

两种等价的方法,一个是插值时插若干二元组 (wi, wj),将 wi 相同的放在一起对 wj 插值,过后再将 wj 相同的放在一起对 wi 插值。逆变换同理。

这种方法具有一定的拓展性,因为它将一个两变元多项式完全转换为了点值表示。

第二个要直观一些:

我们令 \(A[i] = \sum a[i][j]*x^j\),同理得到 \(B[i], C[i]\)。

则有:\(C[i+p] = \sum A[i]*B[p]\)。

对 \(A[i], B[i], C[i]\) 作 FFT,得到 \(C[i+p][j] = \sum A[i][j]*B[p][j]\)。

交换横纵坐标的位置,得到 \(C[j][i+p] = \sum A[j][i]*B[j][p]\)。

这就是一个经典的卷积问题。

总时间复杂度 \(O(n^2\log n)\),其中 n 与 R,C 同阶。

@accepted code@

#include<cstdio>
#include<algorithm>
using namespace std;
const int G = 3;
const int MAXN = 666;
const int MAXM = 2048 + 5;
const int MOD = 998244353;
int pow_mod(int b, int p) {
int ret = 1;
while( p ) {
if( p & 1 ) ret = 1LL*ret*b%MOD;
b = 1LL*b*b%MOD;
p >>= 1;
}
return ret;
}
int pw[15], ipw[15];
void init() {
for(int i=1;i<15;i++)
pw[i] = pow_mod(G, (MOD - 1)/(1<<i)), ipw[i] = pow_mod(pw[i], MOD - 2);
}
void ntt(int *A, int n, int type) {
for(int i=0,j=0;i<n;i++) {
if( i < j ) swap(A[i], A[j]);
for(int l=(n>>1);(j^=l)<l;l>>=1);
}
for(int k=1;(1<<k)<=n;k++) {
int s = (1<<k), t = (s>>1);
int u = (type == 1) ? pw[k] : ipw[k];
for(int i=0;i<n;i+=s)
for(int j=0,p=1;j<t;j++,p=1LL*p*u%MOD) {
int x = A[i + j], y = 1LL*p*A[i + j + t]%MOD;
A[i + j] = (x + y)%MOD, A[i + j + t] = (x + MOD - y)%MOD;
}
}
if( type == -1 ) {
int inv = pow_mod(n, MOD - 2);
for(int i=0;i<n;i++)
A[i] = 1LL*A[i]*inv%MOD;
}
}
struct node{int x, y, ans; node(int _x=0, int _y=0, int _a=0):x(_x), y(_y), ans(_a){} }arr[MAXN*MAXN];
bool operator < (node a, node b) {
if( a.ans == b.ans ) return a.x == b.x ? a.y < b.y : a.x < b.x;
else return a.ans < b.ans;
}
int A[MAXM][MAXM], B[MAXM][MAXM], ans[MAXM][MAXM];
int M[MAXN][MAXN], s1[MAXN][MAXN], s2[MAXN][MAXN], S1, S2;
int getsum1(int x1, int y1, int x2, int y2) {
int ret = s1[x2][y2];
if( x1 > 0 ) ret -= s1[x1-1][y2];
if( y1 > 0 ) ret -= s1[x2][y1-1];
if( x1 > 0 && y1 > 0 ) ret += s1[x1-1][y1-1];
return ret;
}
int getsum2(int x1, int y1, int x2, int y2) {
int ret = s2[x2][y2];
if( x1 > 0 ) ret -= s2[x1-1][y2];
if( y1 > 0 ) ret -= s2[x2][y1-1];
if( x1 > 0 && y1 > 0 ) ret += s2[x1-1][y1-1];
return ret;
}
int R, C, H, W, K, x, y;
int main() {
init();
scanf("%d%d", &R, &C);
for(int i=0;i<R;i++)
for(int j=0;j<C;j++) {
scanf("%d", &M[i][j]), s1[i][j] = M[i][j], s2[i][j] = M[i][j]*M[i][j];
if( i > 0 ) s1[i][j] += s1[i-1][j], s2[i][j] += s2[i-1][j];
if( j > 0 ) s1[i][j] += s1[i][j-1], s2[i][j] += s2[i][j-1];
if( i > 0 && j > 0 ) s1[i][j] -= s1[i-1][j-1], s2[i][j] -= s2[i-1][j-1];
A[R-i-1][C-j-1] = M[i][j];
}
scanf("%d%d", &H, &W);
for(int i=0;i<H;i++)
for(int j=0;j<W;j++)
scanf("%d", &B[i][j]), S1 += B[i][j], S2 += B[i][j]*B[i][j], B[i][j] = -2*B[i][j];
scanf("%d%d", &x, &y); x--, y--;
int len; for(len = 1;len <= 2*R || len <= 2*C;len <<= 1);
for(int i=0;i<R;i++) ntt(A[i], len, 1);
for(int i=0;i<H;i++) ntt(B[i], len, 1);
for(int i=0;i<len;i++)
for(int j=0;j<i;j++)
swap(A[i][j], A[j][i]), swap(B[i][j], B[j][i]);
for(int i=0;i<len;i++) ntt(A[i], len, 1), ntt(B[i], len, 1);
for(int i=0;i<len;i++)
for(int j=0;j<len;j++)
ans[j][i] = 1LL*A[j][i]*B[j][i]%MOD;
for(int i=0;i<len;i++) ntt(ans[i], len, -1);
for(int i=0;i<len;i++)
for(int j=0;j<i;j++)
swap(ans[i][j], ans[j][i]);
for(int i=0;i<R;i++) ntt(ans[i], len, -1);
for(int i=0;i<=R-H;i++)
for(int j=0;j<=C-W;j++) {
ans[R-i-1][C-j-1] += S2;
ans[R-i-1][C-j-1] += getsum2(i, j, i+H-1, j+W-1);
ans[R-i-1][C-j-1] += M[i+x][j+y]*M[i+x][j+y]*H*W;
ans[R-i-1][C-j-1] += 2*M[i+x][j+y]*S1;
ans[R-i-1][C-j-1] -= 2*M[i+x][j+y]*getsum1(i, j, i+H-1, j+W-1);
}
int cnt = 0;
for(int i=0;i<=R-H;i++)
for(int j=0;j<=C-W;j++)
arr[cnt++] = node(i + 1, j + 1, ans[R-i-1][C-j-1]%MOD);
sort(arr, arr + cnt);
scanf("%d", &K);
for(int i=0;i<K;i++)
printf("%d %d %d\n", arr[i].x, arr[i].y, arr[i].ans);
}

@details@

双变元什么的……以前还真的没遇到过。

不过如果按第二种理解方法的话,也不过是一些代数变形的 trick 罢了。

本题还有一种常数较大的方法:把 (i, j) 这一个坐标视为一个 C 进制数 i + j*C,其中 i 是第 1 位,j 是第 2 位。然后再做单变元的卷积。

@codechef - BUYLAND@ Buying Land的更多相关文章

  1. @总结 - 1@ 多项式乘法 —— FFT

    目录 @0 - 参考资料@ @1 - 一些概念@ @2 - 傅里叶正变换@ @3 - 傅里叶逆变换@ @4 - 迭代实现 FFT@ @5 - 参考代码实现@ @6 - 快速数论变换 NTT@ @7 - ...

  2. Indian Scientists Design Device to Collect Solar Energy 印度科学家设计太阳能收集设备

      Indian scientists have designed a new device they hope will solve one of the biggest problems with ...

  3. From missionary to firebrand--Eisle Tu [20160102]

    From missionary to firebrand   杜叶锡恩(1913年(癸丑年)-2015年(乙未年),英文名字Elsie Hume Elliot Tu,丈夫是教育家杜学魁.她是香港著名的 ...

  4. Grass is Green

    Root    3719 - Grass is Green Time limit: 3.000 seconds This year exactly n <tex2html_verbatim_ma ...

  5. Codechef March Challenge 2014——The Street

    The Street Problem Code: STREETTA https://www.codechef.com/problems/STREETTA Submit Tweet All submis ...

  6. 『土地征用 Land Acquisition 斜率优化DP』

    斜率优化DP的综合运用,对斜率优化的新理解. 详细介绍见『玩具装箱TOY 斜率优化DP』 土地征用 Land Acquisition(USACO08MAR) Description Farmer Jo ...

  7. codechef February Challenge 2018 简要题解

    比赛链接:https://www.codechef.com/FEB18,题面和提交记录是公开的,这里就不再贴了 Chef And His Characters 模拟题 Chef And The Pat ...

  8. Codechef May Challenge 2020 Division 1 记录

    目录 Triple Sort Sorting Vases Buying a New String Chef and Bitwise Product Binary Land Not a Real Wor ...

  9. 【BZOJ-3514】Codechef MARCH14 GERALD07加强版 LinkCutTree + 主席树

    3514: Codechef MARCH14 GERALD07加强版 Time Limit: 60 Sec  Memory Limit: 256 MBSubmit: 1288  Solved: 490 ...

随机推荐

  1. Hibernate: insert into xxx (name) values (?)但是数据库中没有数据

    学习hibernate 控制台提示 但数据库中没有任何数据被插入 同样的代码,参考例程中就有数据被插入 比较无解,删除部分代码,红框中的部分,运行一下,再贴回去,就好了

  2. MVC的学习步骤

    (1)搭建环境(2)如何完成Controller和 Model的映射(3)如何把值传给Controller(4)Controller如何把值传给viewer(5)异常处理(6)页面标签(7)文件上传( ...

  3. IntelliJ IDEA 中如何查看一个类的所有继承关系(当前类的所有继承关系图)

    IntelliJ IDEA 中如何查看一个类的所有继承关系(当前类的所有继承关系图) .embody{ padding:10px 10px 10px; margin:0 -20px; border-b ...

  4. 插头DP智障操作合集

    今天一共四道插头DP[其实都差不多],智障错误出了不下五个:D 来,让我好好数落我自己一下 直接写代码注释里吧 Eat the Trees #include<iostream> #incl ...

  5. jsp页面_按回车键触发事件

    一般在列表页面中,都会带有查询按钮,当输入完查询条件后,如果需要通过鼠标点击"查询"按钮才发起查询,那么就感觉不够方便,那么我们就可以修改为按下回车键的时候发起查询. <sc ...

  6. bzoj 4198 [Noi2015]荷马史诗——哈夫曼树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4198 学习一下哈夫曼树.https://www.cnblogs.com/Zinn/p/940 ...

  7. Hdu 1299

    Diophantus of Alexandria Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java ...

  8. LintCode刷题笔记-- Maximal Square

    标签:动态规划 题目描述: Given a 2D binary matrix filled with 0's and 1's, find the largest square containing a ...

  9. 手机端点击键盘无法获取keyCode值的部分时隐藏键盘并执行事件

    用计时器监视window.innerHeight高度改变来判断.触发键盘其他地方也有事件反应 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 ...

  10. Python中的进程池与线程池(包含代码)

    Python中的进程池与线程池 引入进程池与线程池 使用ProcessPoolExecutor进程池,使用ThreadPoolExecutor 使用shutdown 使用submit同步调用 使用su ...