codeforces 781 D. Axel and Marston in Bitland (DP+bitset)

D. Axel and Marston in Bitland
time limit per test
5 seconds
memory limit per test
256 megabytes
input
standard input
output
standard output

A couple of friends, Axel and Marston are travelling across the country of Bitland. There are n towns in Bitland, with some pairs of towns connected by one-directional roads. Each road in Bitland is either a pedestrian road or a bike road. There can be multiple roads between any pair of towns, and may even be a road from a town to itself. However, no pair of roads shares the starting and the destination towns along with their types simultaneously.

The friends are now located in the town 1 and are planning the travel route. Axel enjoys walking, while Marston prefers biking. In order to choose a route diverse and equally interesting for both friends, they have agreed upon the following procedure for choosing the road types during the travel:

  • The route starts with a pedestrian route.
  • Suppose that a beginning of the route is written in a string s of letters P (pedestrain road) and B (biking road). Then, the string  is appended to s, where  stands for the string s with each character changed to opposite (that is, all pedestrian roads changed to bike roads, and vice versa).

In the first few steps the route will look as follows: P, PB, PBBP, PBBPBPPB, PBBPBPPBBPPBPBBP, and so on.

After that the friends start travelling from the town 1 via Bitlandian roads, choosing the next road according to the next character of their route type each time. If it is impossible to choose the next road, the friends terminate their travel and fly home instead.

Help the friends to find the longest possible route that can be travelled along roads of Bitland according to the road types choosing procedure described above. If there is such a route with more than 1018 roads in it, print -1 instead.

Input

The first line contains two integers n and m (1 ≤ n ≤ 5000 ≤ m ≤ 2n2) — the number of towns and roads in Bitland respectively.

Next m lines describe the roads. i-th of these lines contains three integers viui and ti (1 ≤ vi, ui ≤ n0 ≤ ti ≤ 1), where vi and uidenote start and destination towns indices of the i-th road, and ti decribes the type of i-th road (0 for a pedestrian road, 1 for a bike road). It is guaranteed that for each pair of distinct indices ij such that 1 ≤ i, j ≤ m, either vi ≠ vj, or ui ≠ uj, or ti ≠ tj holds.

Output

If it is possible to find a route with length strictly greater than 1018, print -1. Otherwise, print the maximum length of a suitable path.

Examples
input
2 2
1 2 0
2 2 1
output
3
input
2 3
1 2 0
2 2 1
2 2 0
output
-1
Note

In the first sample we can obtain a route of length 3 by travelling along the road 1 from town 1 to town 2, and then following the road 2 twice from town 2 to itself.

In the second sample we can obtain an arbitrarily long route by travelling the road 1 first, and then choosing road 2 or 3 depending on the necessary type.



题目大意:给出一个有向图,边分成两类0/1.每两个点之间不会存在重复的同类边。要求以一条0边开始走,要求每次走的边满足都是前面的取反。简单说就是0->01->0110->01101001->0110100110010110....

题解:DP+bitset

一道不错的DP题。。刚开始被拉过来做,结果想了半天被告知tag读错了。。。。。

f[0/1][i][x][y] 0表示已0打头的串,1表示已1打头的串。i表示长度是2^i,从x出发是否可以到达y

那么转移的时候

if (f[p][i-1][x][y]) f[p][i][x]|=f[p^1][i-1][y]

判断的时候如果f[0][60][1][i]=1,那么答案就是-1

如果不行的话,我们需要求构造一个最长的长度,倒序枚举长度,再枚举上次可到达的点,那么当前可以达到的位置就是now|=f[p][i][x] if (pre[x]) ,每个长度只要有可以到达的点就计入答案。需要注意的是p每到达一个长度都要取反。

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<bitset>
#define N 533
#define LL long long 
using namespace std;
bitset<N> f[2][63][N],pre,now;
int n,m;
LL ans;
int main()
{
	freopen("a.in","r",stdin);
	scanf("%d%d",&n,&m);
	for (int i=1;i<=m;i++) {
		int x,y,opt; scanf("%d%d%d",&x,&y,&opt);
		f[opt][0][x][y]=1;
	}
	for (int i=1;i<=60;i++)
	 for (int p=0;p<=1;p++)
	  for (int x=1;x<=n;x++) 
	   for (int y=1;y<=n;y++) 
	    if (f[p][i-1][x][y]) f[p][i][x]|=f[p^1][i-1][y];
	for (int i=1;i<=n;i++)
	 if (f[0][60][1][i]) {
	 	printf("-1");
	 	return 0;
	 }
	int p=0; pre[1]=1;
	for (LL i=59;i>=0;i--) {
	    now=0;
	    for (int j=1;j<=n;j++)
	     if (pre[j]) now|=f[p][i][j];
	    if (now.count()) p^=1,ans+=(1LL<<i),pre=now;
	}
	if (ans>(1e18)) printf("-1\n");
	else printf("%I64d\n",ans);
}


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
大学生参加学科竞赛有着诸多好处,不仅有助于个人综合素质的提升,还能为未来职业发展奠定良好基础。以下是一些分析: 首先,学科竞赛是提高专业知识和技能水平的有效途径。通过参与竞赛,学生不仅能够深入学习相关专业知识,还能够接触到最新的科研成果和技术发展趋势。这有助于拓展学生的学科视野,使其对专业领域有更深刻的理解。在竞赛过程中,学生通常需要解决实际问题,这锻炼了他们独立思考和解决问题的能力。 其次,学科竞赛培养了学生的团队合作精神。许多竞赛项目需要团队协作来完成,这促使学生学会有效地与他人合作、协调分工。在团队合作中,学生们能够学到如何有效沟通、共同制定目标和分工合作,这对于日后进入职场具有重要意义。 此外,学科竞赛是提高学生综合能力的一种途径。竞赛项目通常会涉及到理论知识、实际操作和创新思维等多个方面,要求参赛者具备全面的素质。在竞赛过程中,学生不仅需要展现自己的专业知识,还需要具备创新意识和解决问题的能力。这种全面的综合能力培养对于未来从事各类职业都具有积极作用。 此外,学科竞赛可以为学生提供展示自我、树立信心的机会。通过比赛的舞台,学生有机会展现自己在专业领域的优势,得到他人的认可和赞誉。这对于培养学生的自信心和自我价值感非常重要,有助于他们更加积极主动地投入学习和未来的职业生涯。 最后,学科竞赛对于个人职业发展具有积极的助推作用。在竞赛中脱颖而出的学生通常能够引起企业、研究机构等用人单位的关注。获得竞赛奖项不仅可以作为个人履历的亮点,还可以为进入理想的工作岗位提供有力的支持。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值