HDU - 6253 Knightmare (打表+拉格朗日插值)

题目链接

题意:一个马在无限大的棋盘中跳,问跳n步能跳到多少个不同的格子。

首先写个打表程序打一下n比较小的时候的表:

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 typedef long long ll;
 4 const int N=50+10,mod=998244353;
 5 const int dx[]= {-2,-1,1,2,2,1,-1,-2};
 6 const int dy[]= {-1,-2,-2,-1,1,2,2,1};
 7 typedef pair<int,int> P;
 8 set<P> st[2];
 9 int n=20,a[N];
10 int main() {
11     st[0].insert({0,0});
12     for(int i=0; i<n; ++i) {
13         for(P p:st[i&1]) {
14             st[i&1^1].insert(p);
15             for(int j=0; j<8; ++j)st[i&1^1].insert({p.first+dx[j],p.second+dy[j]});
16         }
17         a[i]=st[i&1].size();
18     }
19     for(int i=0; i<n; ++i)printf("%d ",a[i]);
20     return 0;
21 }

打印结果:

1 9 41 109 205 325 473 649 853 1085 1345 1633 1949 2293 2665 3065 3493 3949 4433 4945

把元素差分两次后,成了这个亚子:

1 7 24 36 28 24 28 28 28 28 28 28 28 28 28 28 28 28 28 28

发现了什么?当n比较大的时候,经过二次差分后的数组的每一项都是28!因此可以猜测答案是一个关于n的二次多项式,现在要做的是把这个多项式推出来。

手算当然可以,但有没有一个可以不用动脑子就算出来的代码吗?答案是肯定的。拉格朗日插值大法好!

核心代码:(只需要写三个函数,前两个函数的作用是封装多项式的加法和乘法,第三个函数的作用是插值)

 1 typedef double db;
 2 typedef vector<db> Poly;
 3 Poly operator*(Poly a,Poly b) {
 4     Poly c;
 5     c.resize(a.size()+b.size()-1);
 6     for(int i=0; i<a.size(); ++i)
 7         for(int j=0; j<b.size(); ++j)c[i+j]+=a[i]*b[j];
 8     return c;
 9 }
10 Poly operator+(Poly a,Poly b) {
11     Poly c;
12     c.resize(max(a.size(),b.size()));
13     for(int i=0; i<c.size(); ++i) {
14         if(i<a.size())c[i]+=a[i];
15         if(i<b.size())c[i]+=b[i];
16     }
17     return c;
18 }
19 Poly Lagrange(Poly X,Poly Y) {
20     Poly c;
21     int n=X.size();
22     for(int i=0; i<n; ++i) {
23         Poly x({1});
24         for(int j=0; j<n; ++j)if(j!=i) {
25                 x=x*Poly({-X[j],1});
26                 x=x*Poly({1.0/(X[i]-X[j])});
27             }
28         c=c+x*Poly({Y[i]});
29     }
30     return c;
31 }

这样一来,只要输入X向量和Y向量,就能直接求出原多项式了,非常方便。比如输入如下两个向量:

1     Poly a({1,2,3}),b({1,3,7});
2     Poly c=Lagrange(a,b);
3     for(db x:c)printf("%f ",x);

输出的结果为

1.000000 -1.000000 1.000000

也就是说,三个点$(1,2),(2,3),(3,7)$所确定的多项式为$f(x)=x^2-x+1$

现在我们在打印的结果中任取三个点比如$(10,1345),(11,1633),(12,1949)$,得到的结果为:

5.000000 -6.000000 14.000000

即答案关于n的多项式为$f(n)=14n^2-6n+5$。当n比较大时的答案就可以通过这个式子算出来了,n比较小的时候直接输出结果即可。最终提交上去的代码应该是这个亚子:

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 typedef unsigned long long ll;
 4 const int N=1000+10,inf=0x3f3f3f3f;
 5 const int a[]= {1,9,41,109,205,325,473,649,853,1085,1345};
 6 int ka,n;
 7 int main() {
 8     int T;
 9     for(scanf("%d",&T); T--;) {
10         printf("Case #%d: ",++ka);
11         scanf("%d",&n);
12         if(n<=10)printf("%d\n",a[n]);
13         else printf("%llu\n",5-(ll)6*n+(ll)14*n*n);
14     }
15     return 0;
16 }

注意要用unsigned long long,OK了~

ps:如果对精度要求高的话,也可以用分数版的:

 1 struct Frac {
 2     int x,y;
 3     Frac(int _x=0,int _y=1):x(_x),y(_y) {
 4         int g=__gcd(x,y);
 5         x/=g,y/=g;
 6         if(y<0)x=-x,y=-y;
 7     }
 8     Frac operator-() {return Frac(-x,y);}
 9     Frac operator+(Frac b) {return Frac(x*b.y+y*b.x,y*b.y);}
10     Frac operator+=(Frac b) {return *this=(*this)+b;}
11     Frac operator-(Frac b) {return Frac(x*b.y-y*b.x,y*b.y);}
12     Frac operator-=(Frac b) {return *this=(*this)-b;}
13     Frac operator*(Frac b) {return Frac(x*b.x,y*b.y);}
14     Frac operator*=(Frac b) {return *this=(*this)*b;}
15     Frac operator/(Frac b) {return Frac(x*b.y,y*b.x);}
16     Frac operator/=(Frac b) {return *this=(*this)/b;}
17 };
18 typedef Frac db;
19 typedef vector<db> Poly;
20 Poly operator*(Poly a,Poly b) {
21     Poly c;
22     c.resize(a.size()+b.size()-1);
23     for(int i=0; i<a.size(); ++i)
24         for(int j=0; j<b.size(); ++j)c[i+j]+=a[i]*b[j];
25     return c;
26 }
27 Poly operator+(Poly a,Poly b) {
28     Poly c;
29     c.resize(max(a.size(),b.size()));
30     for(int i=0; i<c.size(); ++i) {
31         if(i<a.size())c[i]+=a[i];
32         if(i<b.size())c[i]+=b[i];
33     }
34     return c;
35 }
36 Poly Lagrange(Poly X,Poly Y) {
37     Poly c;
38     int n=X.size();
39     for(int i=0; i<n; ++i) {
40         Poly x({Frac(1)});
41         for(int j=0; j<n; ++j)if(j!=i) {
42                 x=x*Poly({-X[j],Frac(1)});
43                 x=x*Poly({Frac(1)/(X[i]-X[j])});
44             }
45         c=c+x*Poly({Y[i]});
46     }
47     return c;
48 }

 

posted @ 2019-07-13 21:04  jrltx  阅读(367)  评论(0编辑  收藏  举报