NOIP2017Day1题解
Day1
T1.小学奥数。。。
代码:
#include<iostream>
#include<cstring>
#include<string>
#include<algorithm>
#include<cstdio>
#include<cmath>
using namespace std;
long long a,b;
int main()
{
scanf("%lld%lld",&a,&b);
long long ans=a*b-a-b;
printf("%lld",ans);
return 0;
}
T2:大模拟题,没什么好讲的(我还写的特别麻烦,估计最傻逼的代码)
代码:
#include<iostream>
#include<cstring>
#include<string>
#include<algorithm>
#include<cstdio>
#include<cmath>
#define INF 1000
using namespace std;
int T,l;
bool vis[1005],vis1[10005];
char s[10005],ch[105][10];
bool c[10005],e[1005];
int d[1005];
char ssss[1005],sss[10005],ss[105];
int a[1005],b[1005];
int main()
{
scanf("%d",&T);
while (T--){
bool flag=true;
int sum=0; bool zong=true;
memset(a,0,sizeof(a));
memset(b,0,sizeof(b));
scanf("%d%s",&l,s+1);
int n=strlen(s+1);
if (s[3]=='n') flag=true;
else flag=false;
if (flag){
int ckr=0;
for (int i=1;i<=n;i++){
if (s[i]=='^'){
ckr=i;
break;
}
}
for (int i=ckr+1;i<=n;i++){
if (s[i]>='0'&&s[i]<='9') sum=sum*10+s[i]-'0';
else break;
}
}
else {
int ckr=0;
for (int i=1;i<=n;i++){
if (s[i]=='('){
ckr=i;
break;
}
}
for (int i=ckr+1;i<=n;i++){
if (s[i]>='0'&&s[i]<='9') sum=sum*10+s[i]-'0';
else break;
}
}
if (!flag) sum=0;
for (int i=1;i<=l;i++){
scanf("%s",ss);
if (ss[0]=='F'){
cin>>ch[i][0];
//scanf("%s",ch[i][0]);
scanf("%s%s",sss+1,ssss+1);
if (sss[1]=='n') a[i]=INF;
else {
int n=strlen(sss+1);
for (int j=1;j<=n;j++) a[i]=a[i]*10+sss[j]-'0';
}
if (ssss[1]=='n') b[i]=INF;
else {
int n=strlen(ssss+1);
for (int j=1;j<=n;j++) b[i]=b[i]*10+(ssss[j]-'0');
}
if (a[i]==INF&&b[i]==INF) d[i]=0;
else if (a[i]>b[i]) d[i]=-1;
else if (a[i]==INF&&b[i]<INF) d[i]=-1;
else if (a[i]<INF&&b[i]<INF) d[i]=0;
else if (a[i]<INF&&b[i]==INF) d[i]=1;
c[i]=true;
}
else c[i]=false;
}
memset(vis,false,sizeof(vis));
memset(vis1,false,sizeof(vis1));
int i=1; int j=1; int tot=0; int bianlitot=0;
int Max1=0;
int flag1=0;
while (j<=l){
if (c[j]==true){
if (vis[ch[j][0]-'a']){
if (zong) puts("ERR");
zong=false;
}
vis[ch[j][0]-'a']=true;
if (flag1==0){
if (d[j]==1) tot++,e[j]=true;
else if (d[j]==0) tot=tot,e[j]=true;
else if (d[j]==-1) flag1++,e[j]=false;
}
else{
tot=tot;
e[j]=false;
if (d[j]==-1) flag1++;
}
bianlitot++;
Max1=max(Max1,tot);
}
else {
bianlitot--;
if (bianlitot<0){
if (zong) puts("ERR");
zong=false;
}
for (int k=j;k>=1;k--){
if (c[k]==true&&vis1[k]==false){
vis[ch[k][0]-'a']=false;
vis1[k]=true;
if (e[k]) tot=tot-d[k];
else if (e[k]==false){
tot=tot;
if (d[k]==-1) flag1--;
}
break;
}
}
Max1=max(Max1,tot);
}
j++;
}
if (!zong) continue;
if (bianlitot!=0) puts("ERR");
else{
if (Max1==sum) puts("Yes");
else puts("No");
}
}
return 0;
}
T3.一看k<=50,很显然的状态就有了
我们考虑用dp[i][j]表示到i这个点浪费了的代价为j,这里浪费的代价可以这么理解:就是到i相比于最短路dis[i]来说长了j,也就是说走了dis[i]+j的路
考虑转移dp[i][j]->dp[v][j+dis[i]+e[i].step-dis[v]](当前走得路径+到v的长度和到v的最短路进行比较,就知道了浪费了多少),其中dis[i]表示到i的最短路
注意更新顺序,关于dis排序来更新
另外DP的时候要先枚举差量,一定是差量小的来更新差量大的
这样就有70分了
我们考虑怎么拿满分。
注意到一个套路,就是如果拓扑排序后有点剩余,那么剩下的点一定构成环
然后对于-1的情况就是有一个0环在路径上
我们用拓扑排序判一下就行了
同一个 0 环上任意一个点到1的最短路和到n的最短路都一样
所以当这个点 i 满足 dis1[i]+disn[i]<=dis1[n]+K 时,就可以输出 -1了
代码:
#include<bits/stdc++.h>
#define N 200005
#define M 100005
using namespace std;
int T,n,m,K,Mod,kk,kkk,kkkk,x,y,z;
int head[M],head1[M],Head[N],dis1[M],dis2[N],ru[M],dp[M][55],tmp[M],q[N*100];
struct Edge{int nxt,to,step;}e[N];
struct Edge1{int nxt,to,step;}e1[N];
struct Edge2{int nxt,to;}E[N];
inline void link(int x,int y,int z){e[++kk].nxt=head[x];e[kk].to=y;e[kk].step=z;head[x]=kk;}
inline void link1(int x,int y,int z){e1[++kkk].nxt=head1[x];e1[kkk].to=y;e1[kkk].step=z;head1[x]=kkk;}
inline void Link(int x,int y){E[++kkkk].nxt=Head[x];E[kkkk].to=y;Head[x]=kkkk;}
inline void spfa1(){
memset(dis1,127,sizeof(dis1));dis1[1]=0;
int left1=1;int right1=1;q[1]=1;
while (left1<=right1){
int u=q[left1++];
for (int i=head[u];i;i=e[i].nxt){
int v=e[i].to;
if (dis1[v]>dis1[u]+e[i].step){
dis1[v]=dis1[u]+e[i].step;
q[++right1]=v;
}
}
}
}
inline void spfa2(){
memset(dis2,127,sizeof(dis2));dis2[n]=0;
int left1=1;int right1=1;q[1]=n;
while (left1<=right1){
int u=q[left1++];
for (int i=head1[u];i;i=e1[i].nxt){
int v=e1[i].to;
if (dis2[v]>dis2[u]+e1[i].step){
dis2[v]=dis2[u]+e1[i].step;
q[++right1]=v;
}
}
}
}
int id[N];
inline bool cmp(int x,int y){if(dis1[x]==dis1[y])return id[x]<id[y];else return dis1[x]<dis1[y];}
inline void solve(){
memset(dp,0,sizeof(dp));dp[1][0]=1;
for (int i=1;i<=n;i++) tmp[i]=i;
sort(tmp+1,tmp+n+1,cmp);
for (int k=0;k<=K;k++){//注意先枚举差量,一定是差量小的先更新差量大的
for (int i=1;i<=n;i++){
int u=tmp[i];
for (int j=head[u];j;j=e[j].nxt){
int v=e[j].to;
if (k+dis1[u]+e[j].step-dis1[v]>K) continue;
dp[v][k+(dis1[u]+e[j].step-dis1[v])]=1ll*(dp[v][k+(dis1[u]+e[j].step-dis1[v])]+dp[u][k])%Mod;
}
}
}
int ans=0;
for (int i=0;i<=K;i++) ans=(ans+dp[n][i])%Mod;
if (!ans) puts("-1");else printf("%d\n",ans);
}
bool check(){//拓扑排序找环,如果存在一个环,那么这个环中的点都不会加入到拓扑序中,显然不可能会入度为0
int left1=1;int right1=0;//right1=0打成right1=1了
for (int i=1;i<=n;i++) if (!ru[i]) q[++right1]=i,id[i]=n;
while (left1<=right1){
int u=q[left1++];
for (int i=Head[u];i;i=E[i].nxt){
int v=E[i].to;
if (!--ru[v]) q[++right1]=v;
}
}
for (int i=1;i<=right1;i++) id[q[i]]=i;
for (int i=1;i<=n;i++)
if (ru[i]&&dis1[i]+dis2[i]<=dis1[n]+K) return true;//如果i这个环在可能的路上那么就gg
return false;
}
inline void init(){
memset(Head,0,sizeof(Head));kkkk=0;
memset(head,0,sizeof(head));kk=0;
memset(head1,0,sizeof(head1));kkk=0;
memset(ru,0,sizeof(ru));
}
int main(){
scanf("%d",&T);
while (T--){
init();
scanf("%d%d%d%d",&n,&m,&K,&Mod);
for (int i=1;i<=m;i++){
scanf("%d%d%d",&x,&y,&z);
link(x,y,z);link1(y,x,z);
if (!z) Link(x,y),ru[y]++;
}
spfa1();spfa2();
if (check()) puts("-1");
else solve();
}
return 0;
}
/*1
2 2 6 5
1 2 2
2 1 2*/
NOIP2017Day1题解的更多相关文章
- 2016 华南师大ACM校赛 SCNUCPC 非官方题解
我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...
- noip2016十连测题解
以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...
- BZOJ-2561-最小生成树 题解(最小割)
2561: 最小生成树(题解) Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1628 Solved: 786 传送门:http://www.lyd ...
- Codeforces Round #353 (Div. 2) ABCDE 题解 python
Problems # Name A Infinite Sequence standard input/output 1 s, 256 MB x3509 B Restoring P ...
- 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解
题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...
- 2016ACM青岛区域赛题解
A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Jav ...
- poj1399 hoj1037 Direct Visibility 题解 (宽搜)
http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...
- 网络流n题 题解
学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ...
- CF100965C题解..
求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ...
随机推荐
- 【翻译】在Sencha应用程序中使用插件和混入
原文:Using Plugins and Mixins in Your Sencha Apps 概述 当扩展一个框架类的功能的时候,通常都会直接将新功能写入派生类,然而,如果所需的同一功能存在于多个组 ...
- Shell中read的常用方式
read命令的语法: read -p "Prompt" variable1 variable2 variableN -p "Prompt": 显示提示信息(和用 ...
- 跨平台移动APP开发进阶(四)AngularJS简介
AngularJS 是一个为动态WEB应用设计的结构框架.它能让你使用HTML作为模板语言,通过扩展HTML的语法,让你能更清楚.简洁地构建你的应用组件. 它的创新点在于,利用 数据绑定 和 依赖注入 ...
- 关于oracle表名区分大小写的问题
oracle不是区分大小写的,是建表的时候是没有去掉双引号. CREATE TABLE TableName(id number); //虽然写的时候是有大写和小写,但是在数据库里面是不区分的. ...
- tomcat整合apache
历时4个多小时,终于把tomcat与apache整合起来了. 中间出了各种各样的问题,现记录一下,也希望能对后来者有点帮助. 背景 apache与tomcat的区别联系大家都知道: tomcat能处理 ...
- 网站论坛同步用户,整合api,实现…
在网上参考了很多资料后,终于完美实现了网站和discuz!nt论坛的双向整合,整合后网站和论坛之间可以同步注册.登录.退出和修改登录密码操作. 本系统的实现形式是新云CMS网站(ASP)和Discuz ...
- STL - vector容器
1Vector容器简介 vector是将元素置于一个动态数组中加以管理的容器. vector可以随机存取元素(支持索引值直接存取, 用[]操作符或at()方法,这个等下会详讲). vector尾部添加 ...
- LeetCode之“动态规划”:Maximal Square && Largest Rectangle in Histogram && Maximal Rectangle
1. Maximal Square 题目链接 题目要求: Given a 2D binary matrix filled with 0's and 1's, find the largest squa ...
- nfc近场通信
NFC简介: Near Field Communication 近场通信,是一种数据传输技术. 与wifi.蓝牙.红外线等数据传输技术的一个主要差异就是有效距离一般不能超过4cm. NFC支持3种工作 ...
- OpenCV——颜色运算(二)
#ifndef PS_ALGORITHM_H_INCLUDED #define PS_ALGORITHM_H_INCLUDED #include <iostream> #include & ...