UVA 11401 - Triangle Counting 组合

题目链接点这儿

这题本来不难的。。。但是太悲伤了。。。没看到是当有输入的n是小于3时就结束程序T^T

如果设an表示有1~n个线时,可以组成的三角形的数目,那么如果扩展到an+1,多出的三角形全是包含边n+1的,所以只要保证你在1~n挑选的两数之和大于n+1就行。

这时,我们可以发现an和an+1的差值是一个三角形

                 UVA 11401 - Triangle Counting 组合_第1张图片

……

差值就第n行的值,规律很明显。所以式子也很容易找出来。

——————————————————————————————————————————————————————

以上是网上的做法。。。我在训练赛中也想出来了。。。但是。。。由于开头提到的错误。。。我程序是判断输入为0的时候结束程序。所以当程序是以1或2结尾时。。。它会TLE。。。我还不知道。。。我还真以为是程序太慢了。。。所以。。。我只好把通项求了出来。。。就是上面那个三角形的前n行和。。。递推的式子都出来了,求通项是很容易的。。。而且可以轻松的得到一个线性齐次的差分方程,不想化简sigma的可以直接去解那个差分方程得到通项的表达式。。。总之。。。我是在求出通项后发现仍然TLE才意识到我看错题了。。。←这是我求的通项,你可以在OEIS上找到这个数列来验证我的答案。总之。。这题做的太伤了。。。T^T

下面是代码。。。其实也不用放了。。。直接把上面的公式复制粘贴一下就行了。。。

#include <vector>
#include <list>
#include <map>
#include <set>
#include <deque>
#include <queue>
#include <stack>
#include <bitset>
#include <algorithm>
#include <functional>
#include <numeric>
#include <utility>
#include <sstream>
#include <iostream>
#include <iomanip>
#include <cstdio>
#include <cmath>
#include <cstdlib>
#include <cctype>
#include <string>
#include <cstring>
#include <cstdio>
#include <cmath>
#include <cstdlib>
#include <ctime>
#include <climits>
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)>(b)?(b):(a))
#define rep(i,initial_n,end_n) for(int (i)=(initial_n);(i)<(end_n);i++)
#define repp(i,initial_n,end_n) for(int (i)=(initial_n);(i)<=(end_n);(i)++)
#define reep(i,initial_n,end_n) for((i)=(initial_n);(i)<(end_n);i++)
#define reepp(i,initial_n,end_n) for((i)=(initial_n);(i)<=(end_n);(i)++)
#define eps 1.0e-9
#define MAX_N 1010

using namespace std;
typedef pair<int, int> pii;
typedef pair<double, double> pdd;
typedef long long ll;
typedef unsigned long long ull;

int main() {
    int n;
    while(scanf("%d", &n)) {
        if(n<3) break;
        n-=3;
        printf("%lld\n", (ll)(((ll)(2*n+3)*(n+2)*(n+1)/6-(ll)((n+2)/2))/4));
    }
    return 0;

}


你可能感兴趣的:(算法,数学,ACM,组合数学)