Codeforces Round #274 (Div. 2)-E. Riding in a Lift

原题链接

E. Riding in a Lift
time limit per test
2 seconds
memory limit per test
256 megabytes
input
standard input
output
standard output

Imagine that you are in a building that has exactly n floors. You can move between the floors in a lift. Let's number the floors from bottom to top with integers from 1 to n. Now you're on the floor number a. You are very bored, so you want to take the lift. Floor numberb has a secret lab, the entry is forbidden. However, you already are in the mood and decide to make k consecutive trips in the lift.

Let us suppose that at the moment you are on the floor number x (initially, you were on floor a). For another trip between floors you choose some floor with number y (y ≠ x) and the lift travels to this floor. As you cannot visit floor b with the secret lab, you decided that the distance from the current floor x to the chosen y must be strictly less than the distance from the current floor x to floor b with the secret lab. Formally, it means that the following inequation must fulfill: |x - y| < |x - b|. After the lift successfully transports you to floor y, you write down number y in your notepad.

Your task is to find the number of distinct number sequences that you could have written in the notebook as the result of k trips in the lift. As the sought number of trips can be rather large, find the remainder after dividing the number by 1000000007 (109 + 7).

Input

The first line of the input contains four space-separated integers nabk (2 ≤ n ≤ 50001 ≤ k ≤ 50001 ≤ a, b ≤ na ≠ b).

Output

Print a single integer — the remainder after dividing the sought number of sequences by 1000000007 (109 + 7).

Examples
input
5 2 4 1
output
2
input
5 2 4 2
output
2
input
5 3 4 1
output
0
假设dp[0][i]表示第j(j <k)次时到达的i层的路径数,那么通过i,第j+1次可以到达的范围设为l, r那么l到r的数要加上dp[0][i],区间更新,可以用前缀和思想,l处加上dp[0][i], r+1处减去dp[0][i]

#include <bits/stdc++.h>
#define maxn 5005
#define MOD 1000000007
using namespace std;
typedef long long ll;

ll dp[2][maxn];
int n, a, b, k;
int main(){
	
//	freopen("in.txt", "r", stdin);
	scanf("%d%d%d%d", &n, &a, &b, &k);
	int e = 0;
	dp[0][a] = 1;
	while(k--){
		memset(dp[e^1], 0, sizeof(dp[e^1]));
		for(int i = 1; i <= n; i++){
			if(dp[e][i] && abs(i-b) != 1){
				int l, r;
				if(i > b){
					l = b + 1;
					r = min(n,(i-b)+i-1);
					if(i == n)
					 r = n - 1;
				}
				else{
					r = b - 1;
					l = max(1, i-(b-i)+1);
					if(i == 1)
					 l = i + 1;
				}
				if(i < l || i > r){
				(dp[e^1][l] += dp[e][i]) %= MOD;
				((dp[e^1][r+1] -= dp[e][i]) += MOD) %= MOD;
			   }
			   else{
			   	(dp[e^1][l] += dp[e][i]) %= MOD;
			   	((dp[e^1][i] -= dp[e][i]) += MOD) %= MOD;
			   	((dp[e^1][i+1] += dp[e][i]) += MOD) %= MOD;
			   	((dp[e^1][r+1] -= dp[e][i]) += MOD) %= MOD;
			   }
			}
		}
		e ^= 1;
		for(int i = 1; i <= n; i++)
		 (dp[e][i] += dp[e][i-1]) %= MOD;
	}
	ll ans = 0;
	for(int i = 1; i <= n; i++){
		(ans += dp[e][i]) %= MOD;
	}
	printf("%I64d\n", ans);
	return 0;
}


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值