题目链接点这儿
这题本来不难的。。。但是太悲伤了。。。没看到是当有输入的n是小于3时就结束程序T^T
如果设an表示有1~n个线时,可以组成的三角形的数目,那么如果扩展到an+1,多出的三角形全是包含边n+1的,所以只要保证你在1~n挑选的两数之和大于n+1就行。
这时,我们可以发现an和an+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; }