SHUOJ Arithmetic Sequence (FFT)
题意:求一个序列中,有多少三元组(其中元素不重复)在任意的排列下能构成等差数列。
分析:等差数列:\(A_j-A_i=A_k-A_j\),即\(2A_j=A_i+A_k\),枚举\(A_i+A_j\)的所有情况对应的个数,再扫一遍求解。
先统计出每个数对应的出现次数,FFT计算出和的组合情况。但是要减去\(A_i+A_i\)得到的结果以及\(A_i+A_j\)以及\(A_j+A_i\)重复的计算。
现在对于数\(A_j\),假设\(cnt=2*A_j\)的系数,当然cnt中要减去\(A_j\)本身和一个值与\(A_j\)相等的数组合而成的情况。枚举完这个数以后,把这个数从序列中抹去,因为这个数对结果做出的贡献已经计算,之后的统计中该数以及该数对结果的贡献不能重复计算。
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
const int MAXN = 1e5 + 10;
const double PI = acos(-1.0);
struct Complex{
    double x, y;
    inline Complex operator+(const Complex b) const {
        return (Complex){x +b.x,y + b.y};
    }
    inline Complex operator-(const Complex b) const {
        return (Complex){x -b.x,y - b.y};
    }
    inline Complex operator*(const Complex b) const {
        return (Complex){x *b.x -y * b.y,x * b.y + y * b.x};
    }
} va[MAXN * 2 + MAXN / 2], vb[MAXN * 2 + MAXN / 2];
int lenth = 1, rev[MAXN * 2 + MAXN / 2];
int N, M;   // f 和 g 的数量
    //f g和 的系数
    // 卷积结果
    // 大数乘积
int f[MAXN],g[MAXN];
vector<LL> conv;
vector<LL> multi;
void debug(){for(int i=0;i<conv.size();++i) cout<<conv[i]<<" ";cout<<endl;}
//f g
void init()
{
    int tim = 0;
    lenth = 1;
    conv.clear(), multi.clear();
    memset(va, 0, sizeof va);
    memset(vb, 0, sizeof vb);
    while (lenth <= N + M - 2)
        lenth <<= 1, tim++;
    for (int i = 0; i < lenth; i++)
        rev[i] = (rev[i >> 1] >> 1) + ((i & 1) << (tim - 1));
}
void FFT(Complex *A, const int fla)
{
    for (int i = 0; i < lenth; i++){
        if (i < rev[i]){
            swap(A[i], A[rev[i]]);
        }
    }
    for (int i = 1; i < lenth; i <<= 1){
        const Complex w = (Complex){cos(PI / i), fla * sin(PI / i)};
        for (int j = 0; j < lenth; j += (i << 1)){
            Complex K = (Complex){1, 0};
            for (int k = 0; k < i; k++, K = K * w){
                const Complex x = A[j + k], y = K * A[j + k + i];
                A[j + k] = x + y;
                A[j + k + i] = x - y;
            }
        }
    }
}
void getConv(){             //求多项式
    init();
    for (int i = 0; i < N; i++)
        va[i].x = f[i];
    for (int i = 0; i < M; i++)
        vb[i].x = g[i];
    FFT(va, 1), FFT(vb, 1);
    for (int i = 0; i < lenth; i++)
        va[i] = va[i] * vb[i];
    FFT(va, -1);
    for (int i = 0; i <= N + M - 2; i++)
        conv.push_back((LL)(va[i].x / lenth + 0.5));
}
void getMulti()             //求A*B
{
    getConv();
    multi = conv;
    reverse(multi.begin(), multi.end());
    multi.push_back(0);
    int sz = multi.size();
    for (int i = 0; i < sz - 1; i++){
        multi[i + 1] += multi[i] / 10;
        multi[i] %= 10;
    }
    while (!multi.back() && multi.size() > 1)
        multi.pop_back();
    reverse(multi.begin(), multi.end());
}
int a[MAXN];
int cnt[MAXN];
int main()
{
    #ifndef ONLINE_JUDGE
        freopen("in.txt","r",stdin);
        freopen("out.txt","w",stdout);
    #endif
    int T; scanf("%d",&T);
    while(T--){
        int n; scanf("%d",&n);
        int mx = -1;
        memset(cnt,0,sizeof(cnt));
        for(int i =1;i<=n;++i){
            scanf("%d",&a[i]);
            mx = max(mx,a[i]);
            cnt[a[i]]++;
        }
        N = M = mx+1;
        for(int i=0;i<N;++i){
            f[i] = g[i] = cnt[i];
        }
        getConv();
        int sz = conv.size();
        for(int i=1;i<=n;++i){
            conv[a[i]*2]--;
        }
        for(int i=0;i<sz;++i){
            conv[i]>>=1;
        }
        LL res=0;
        //debug();
        //sort(a+1,a+n+1);
        for(int i=1;i<=n;++i){
            if(2*a[i]>=sz) continue;
            LL tmp = conv[2*a[i]];
            tmp -= cnt[a[i]]-1;           //减去由自己构成的
            conv[2*a[i]] -= cnt[a[i]]-1;    //将Ai对结果的贡献抹去
            cnt[a[i]]--;                    //将Ai从原序列中抹去
            res += tmp;
        }
        printf("%lld\n",res);
    }
    return 0;
}
SHUOJ Arithmetic Sequence (FFT)的更多相关文章
- hdu 5400 Arithmetic Sequence
		http://acm.hdu.edu.cn/showproblem.php?pid=5400 Arithmetic Sequence Time Limit: 4000/2000 MS (Java/Ot ... 
- hdu 5400 Arithmetic Sequence(模拟)
		Problem Description A sequence b1,b2,⋯,bn are called (d1,d2)-arithmetic sequence ≤i≤n) such that ≤j& ... 
- Arithmetic Sequence(dp)
		Arithmetic Sequence Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 51 Solved: 19[Submit][Status][We ... 
- [Swift]LeetCode1027. 最长等差数列 | Longest Arithmetic Sequence
		Given an array A of integers, return the length of the longest arithmetic subsequence in A. Recall t ... 
- (模拟)Arithmetic Sequence -- HDU -- 5400
		链接: http://acm.hdu.edu.cn/showproblem.php?pid=5400 Time Limit: 4000/2000 MS (Java/Others) Memory ... 
- HZAU 21——Arithmetic Sequence——————【暴力 or dp】
		Arithmetic Sequence Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 1810 Solved: 311[Submit][Status] ... 
- 华中农业大学第四届程序设计大赛网络同步赛-1020: Arithmetic Sequence,题挺好的,考思路;
		1020: Arithmetic Sequence Time Limit: 1 Sec Memory Limit: 128 MB Submit: ->打开链接<- Descriptio ... 
- LeetCode 1027. Longest Arithmetic Sequence
		原题链接在这里:https://leetcode.com/problems/longest-arithmetic-sequence/ 题目: Given an array A of integers, ... 
- 【leetcode】1027. Longest Arithmetic Sequence
		题目如下: Given an array A of integers, return the length of the longest arithmetic subsequence in A. Re ... 
随机推荐
- Object和Function谁先被创建
			http://bbs.csdn.net/topics/390772104#post-397284029 
- 【C++缺省函数】  空类默认产生的6个类成员函数
			1.缺省构造函数. 2.缺省拷贝构造函数. 3. 缺省析构函数. 4.缺省赋值运算符. 5.缺省取址运算符. 6. 缺省取址运算符 const. <span style="font-s ... 
- 论坛模块__发帖时使用FCKeditor
			论坛模块__发帖时使用FCKeditor 测试 <html> <head> <meta http-equiv="content-type" conte ... 
- iOS开发之-- 从当前隐藏导航界面push到下一个显示导航界面出现闪一下的问题
			在修改项目代码的过程中,遇到一个问题,就是比如主页面的导航栏是隐藏的,但是需要push到别的页面,这个时候,会出现导航栏闪一下的情况, 下面是我写的一种方案,也就是在loadView这个生命周期函数中 ... 
- Hadoop1.2.1 的 “Hello world!”
			下图是大概步骤: 下面是详细步骤,但我的代码跟上面有点不一样,但都是一个道理: 第一个程序测试 wordcount 先创建目录 hadoop fs -mkdir /wc hadoop fs -mkdi ... 
- RF内建的变量
			${CURDIR} 提供当前测试文件存放的绝对路径.该变量是大小写敏感的.${TEMPDIR} 获取操作系统临时文件夹的绝对路径. 在UNIX系统是在/tmp, 在windows系统是在c:\Docu ... 
- 第十五篇:使用 FP-growth 算法高效挖掘海量数据中的频繁项集
			前言 对于如何发现一个数据集中的频繁项集,前文讲解的经典 Apriori 算法能够做到. 然而,对于每个潜在的频繁项,它都要检索一遍数据集,这是比较低效的.在实际的大数据应用中,这么做就更不好了. 本 ... 
- 透过Nim游戏浅谈博弈
			452. Nim游戏! ★ 输入文件:nim!.in 输出文件:nim!.out 简单对比时间限制:1 s 内存限制:128 MB 甲,乙两个人玩Nim取石子游戏. nim游戏的规则是 ... 
- vue+node+mongoDB 火车票H5(五)---城市选择
			选择城市,支持搜索,储存选择过城市的历史记录,点击索引能跳到相应位置 1. 父子组件传值,trainCity城市选择组件,选择城市后改变父组件的值 2. 把城市数组按照字母A到Z排序 3.点击字母索引 ... 
- JS:ES5数组基本操作
			一.添加删除 push(): 尾部添加,返回数组 pop(): 尾部删除,返回删除项 unshift() : 头部添加,返回数组 shift() : 头部删除,返回删除项 二.插入.替换 万能spli ... 
