$SCOJ4427 Miss Zhao's Graph$

\(problem\)

给定一个包含n个顶点m条边的带权有向图,找一条边数最多的路径,且路径上的边的权值严格递增。
图中可能有重边和自环。

\(题意非常简单:n个点 m个带权边 最多能连成多少条边\)

\(看起来像搜索 就拿搜索水了一波\)
\(然后吧评测都卡爆了(而且只有9pts)\)
总是觉得好亏
\(打了一只前向星+DFS瞎搜\)(大雾)

\(~~恶心的DP题~~\)
\(9pts -> 0pts -> 64pts -> 73pts - >91pts ->100pts\)

数据范围很大 显然\(O(N)\) \(or\) $O(M) $
这题用搜索可以达到 \(O(N*玄学)\) 玄学 = 一次搜索时间 >\(如果带上记忆化 应该快点\)
\(那么DP的解法是O(M)的 每条边跑一次 记录一次\)
重点 ->“递增”
首先 要按权值给边排序
\(所以 应该注重权值相同的边\)
\(if(i < m and edge[i].dis == edge[i+1].dis) continue ;\)
\(加不加等于都没有关系 因为初始值是0\)
\(权值相同就直接跳过 否则会Wrong Answer\)
\(这是特判权值相同的边\)
cnt数组用来储存边权相同的情况
这句\(cnt[edge[k].to] = max ( cnt[edge[k].to] , dp[edge[k].from] + 1 ) ;\) 不要手欠把dp和cnt打错 否则喜提WA数量不等(滑稽
还有一句话 \(十年OI一场空 不开LL见祖宗orz\)
实测int WA了一个点


\(code\)

#include <bits/stdc++.h>
using namespace std;
typedef long long LL ;
inline LL In() { LL res(0),f(1); register char c = getchar() ;
	while(!isdigit(c)) { if(c == '-') f = -1 ; c = getchar() ; }
	while(isdigit(c)) res = (res << 1) + (res << 3) + (c & 15) , c = getchar() ; return res * f ;
}
int n , m ;
const int N = 300000 ;
struct node{
    int from ;
    int to ;
    int dis ;
}edge[N] ;
LL cnt[N] ;
LL dp[N] ;
bool cmp(node x,node y) { return x.dis < y.dis ; }
signed main() {
	memset(edge,0,sizeof(edge)) ; memset(dp,0,sizeof(dp)) ;
    n = In() ; m = In() ;
    for(register int i =1 ; i <= m ; i ++) {//读入
    	int u = In() , v = In() , w = In() ;
		edge[i] = (node){u,v,w} ;
    }
    sort(edge+1 , edge+m+1 , cmp) ;//排序
	int pos = 1 ;
	for(register int i = 1 ;i <= m ;i ++) {//DP
        if(i < m and edge[i].dis == edge[i+1].dis) continue ;//特判权值相同
        for(register int k = pos ; k <= i ; k ++) cnt[edge[k].to] = max ( cnt[edge[k].to] , dp[edge[k].from] + 1 ) ;
		for(register int k = pos ; k <= i ; k ++) dp[edge[k].to] = cnt[edge[k].to] ;
        pos = i + 1 ;
    }
	cout << * max_element( dp + 1 , dp + n + 1 ) << endl ;//最大值函数
    return 0;
}

posted @ 2019-03-17 14:27  Isaunoya  阅读(221)  评论(3编辑  收藏  举报
TOP