T1

考场乱搞出锅了...

正解:

把原序列按k往左和往右看成两个序列,求个前缀和,找下一个更新的位置,直接暴跳。

Code
#include<cstdio>
#include<cstring>
#define MAX 100100
#define re register
#define int long long
namespace OMA
{
int t,n,k;
int c[MAX],a[MAX],b[MAX];
int sum[MAX][2],next[MAX][2];
struct stream
{
template<typename type>inline stream &operator >>(type &s)
{
int w=1; s=0; char ch=getchar();
while(ch<'0'||ch>'9'){ if(ch=='-')w=-1; ch=getchar(); }
while(ch>='0'&&ch<='9'){ s=s*10+ch-'0'; ch=getchar(); }
return s*=w,*this;
}
}cin;
signed main()
{
cin >> t;
while(t--)
{
memset(next,0,sizeof(next));
cin >> n >> k;
for(re int i=1; i<=n; i++)
{ cin >> c[i]; }
int len1 = 1,len2 = 1;
a[1] = b[1] = 0;
for(re int i=k; i>=2; i--)
{ a[++len1] = c[i]; }
for(re int i=k+1; i<=n; i++)
{ b[++len2] = c[i]; }
for(re int i=1; i<=len1; i++)
{ sum[i][0] = sum[i-1][0]+a[i]; }
for(re int i=1; i<=len2; i++)
{ sum[i][1] = sum[i-1][1]+b[i]; }
int nim = sum[1][0],pos = 1;
for(re int i=2; i<=len1; i++)
{ if(sum[i][0]<nim){ nim = sum[i][0],next[pos][0] = i,pos = i; } }
nim = sum[len1][0],pos = len1;
for(re int i=len1-1; i>=1; i--)
{ if(sum[i][0]<nim){ nim = sum[i][0],next[pos][0] = i,pos = i; } }
nim = sum[1][1],pos = 1;
for(re int i=2; i<=len2; i++)
{ if(sum[i][1]<nim){ nim = sum[i][1],next[pos][1] = i,pos = i; } }
nim = sum[len2][1],pos = len2;
for(re int i=len2-1; i>=1; i--)
{ if(sum[i][1]<nim){ nim = sum[i][1],next[pos][1] = i,pos = i; } }
if(sum[len1][0]+sum[len2][1]>0)
{ printf("No\n"); continue ; }
int l = 1,r = 1;
bool flag = true;
while(next[l][0]||next[r][1])
{
flag = !next[l][0]?false:true;
for(re int i=l+1; i<=next[l][0]; i++)
{
if(sum[i][0]+sum[r][1]>0)
{ flag = false; break ; }
}
if(!flag)
{
flag = !next[r][1]?false:true;
for(re int i=r+1; i<=next[r][1]; i++)
{
if(sum[i][1]+sum[l][0]>0)
{ flag = false; break ; }
}
if(!flag)
{ break ; }
r = next[r][1];
continue ;
}
l = next[l][0];
}
if(!flag)
{ printf("No\n"); continue ; }
l = len1,r = len2; flag = true;
while(next[l][0]||next[r][1])
{
flag = !next[l][0]?false:true;
for(re int i=l-1; i>=next[l][0]; i--)
{
if(sum[i][0]+sum[r][1]>0)
{ flag = false; break ; }
}
if(!flag)
{
flag = !next[r][1]?false:true;
for(re int i=r-1; i>=next[r][1]; i--)
{
if(sum[i][1]+sum[l][0]>0)
{ flag = false; break ; }
}
if(!flag)
{ break ; }
r = next[r][1];
continue ;
}
l = next[l][0];
}
if(!flag)
{ printf("No\n"); }
else
{ printf("Yes\n"); }
}
return 0;
}
}
signed main()
{ return OMA::main(); }

T2

\(next\_permutation\)+\(2^{n-1}\) 50pts。

正解:

不会...

T3

直接跑两遍最短路,20pts。

乱搞:

显然刚刚的做法是假的,画个图就知道。

考虑如何让它不那么假,我们可以枚举 \(i,i\in[2,n-1]\) ,先删去 \(i\) ,即将其值赋成0,然后再正反跑两遍最短路,如果两遍最短路中都经过了当前的 \(i\) ,说明 \(i\) 是我必须要加上的点,再把当前的答案加上 \(i\) 的值即可。

最后再跑两遍不删点的最短路,比较一下答案即可。

然而好像也是假的

Code
//应该是假做法,数据太水了
#include<queue>
#include<cstdio>
#include<cstring>
#define MAX 62500
#define re register
using std::queue;
const int N = 255;
int n,m,p1[N],p2[N];
struct graph
{
int next;
int to;
}edge[MAX];
int cnt=1,head[N];
inline void add(int u,int v)
{ edge[++cnt] = (graph){head[u],v},head[u] = cnt; }
namespace SSSP
{
bool vis[MAX];
int dis[MAX],pre[MAX];
inline void spfa(int u)
{
memset(dis,0x3f,sizeof(dis));
memset(pre,0,sizeof(pre));
memset(vis,0,sizeof(vis));
queue<int>q; q.push(u);
dis[u] = p2[u]; vis[u] = true;
while(!q.empty())
{
int v = q.front(); q.pop();
vis[v] = false;
for(re int i=head[v],k; i; i=edge[i].next)
{
k = edge[i].to;
if(dis[k]>dis[v]+p2[k])
{
dis[k] = dis[v]+p2[k];
pre[k] = v;
if(!vis[k])
{ q.push(k); vis[k] = true; }
}
}
}
}
}using namespace SSSP;
namespace OMA
{
struct stream
{
template<typename type>inline stream operator >>(type &s)
{
int w=1; s=0; char ch=getchar();
while(ch<'0'||ch>'9'){ if(ch=='-')w=-1; ch=getchar(); }
while(ch>='0'&&ch<='9'){ s=s*10+ch-'0'; ch=getchar(); }
return s*=w,*this;
}
}cin;
inline int min(int a,int b)
{ return a<b?a:b; }
signed main()
{
cin >> n >> m;
for(re int i=1; i<=n; i++)
{ cin >> p1[i]; }
for(re int i=1,u,v; i<=m; i++)
{ cin >> u >> v; add(u,v); }
int ans = MAX;
for(re int i=2; i<=n-1; i++)
{
int tmp = 0;
for(re int j=1; j<=n; j++)
{ p2[j] = p1[j]; }
p2[i] = 0;
spfa(1);
tmp += dis[n];
int u = n;
bool flag = true,jud = false;
while(pre[u])
{ if(pre[u]==i){ flag = false; } p2[u] = 0; u = pre[u]; }
p2[1] = 0;
spfa(n);
while(pre[u])
{ if(pre[u]==i&&!flag){ jud = true; tmp += p1[i]; } u = pre[u]; }
tmp += dis[1];
//ans = min(ans,tmp);
if(jud)
{ ans = min(ans,tmp); /*printf("ans=%d tmp=%d\n",ans,tmp);*/ }
}
int tmp = 0;
for(re int i=1; i<=n; i++)
{ p2[i] = p1[i]; }
spfa(1);
tmp += dis[n];
int u = n;
while(pre[u])
{ p2[u] = 0; u = pre[u]; }
p2[1] = 0;
spfa(n);
tmp += dis[1];
ans = min(ans,tmp);
printf("%d\n",ans>=MAX?-1:ans);
return 0;
}
}
signed main()
{ return OMA::main(); }

T4

不会...

noip35的更多相关文章

随机推荐

  1. Linux学习之路第四天(运行级别)

    linux 实用指令 指定运行级别 运行级别说明 0 :关机 1:单用户(找回丢失密码) 2.多用户状态没有网络服务 3.多用户状态有网络服务 4.系统未保留给用户 5.图形界面 6.系统重启 常用的 ...

  2. 分库分表框架ShardingSphere入门学习1

    背景 传统的将数据集中存储至单一数据节点的解决方案,在性能.可用性和运维成本这三方面已经难于满足互联网的海量数据场景. 从性能方面来说,由于关系型数据库大多采用 B+ 树类型的索引,在数据量超过阈值的 ...

  3. android实现计时器(转)

    新建布局文件activity_main.xml   <?xml version="1.0" encoding="utf-8"?> <Linea ...

  4. chage 修改用户密码时间限制

    chage [options] LOGIN chage针对用户的密码过期时间.过期提前多少天警示等功能实现,passwd也可以实现,但是passwd --expire参数是直接用户密码过期,强制用户下 ...

  5. 高校表白App-团队冲刺第六天

    今天要做什么 在引导页的基础上添加小红点,并且在滑动时进行增强用户体验的修饰 做了什么 在布局中成功添加小红点,并在activity中得到实现;滑动在3/4时发生渐变,增强用户体验;滑动可回退;在最后 ...

  6. IO编程之IO流

    Java的IO流是实现输入.输出的基础,它可以方便的实现数据的输入输出操作.在Java中把不同的输入输出源(键盘.文件.网络连接)抽象表述为流(stream).通过流可以使java程序使用相同的方式访 ...

  7. 10、Java——内部类

    ​  1.类中定义类 (1)当一类中的成员,作为另外一种事物的时候,这个成员就可以定义为内部类. (2)分类:①成员内部类 ②静态内部类 ③私有内部类 ④局部内部类 ⑤匿名内部类 ⑥Lambda表达式 ...

  8. Spring常见问题(五)

    1.静态资源访问配置 绝对路径:访问静态资源. <mvc:resources location="/js/" mapping="/js/**">&l ...

  9. .net 5+ 知新:【2】 .Net Framework 、.Net 、 .NET Standard的概念与区别

    作为了解历史和眼睛或者过程,我们需要将 .Net Framwork ..Net. .Net Stander几个概念进行下理解. .net 代表跨平台框架,从.net 5开始就统一叫.net,废弃原来的 ...

  10. Redis挂了,流量把数据库也打挂了,怎么办?

    你好呀,我是歪歪. 是这样的,前几天有个读者给我发消息,说面试的时候遇到一个场景题: 他说他当时,一时间竟然找不到回答问题的角度,感觉自己没有回答到点子上. 我仔细想了一下,确实是感到这个问题有一丝丝 ...