noip33
T1
第一个猎人死的轮数等于在1号猎人之前死的猎人数+1,如果当前这个人没死,那么他死在一号猎人之前的概率为 \(\frac{w_{i}}{w_{1}+w_{i}}\),因为每死一个就会造成1的贡献,所以概率就是期望。最后答案记得+1。
Code
#include<cstdio>
#define MAX 100010
#define re register
#define int long long
namespace OMA
{
int n,w[MAX];
const int p = 998244353;
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 quickpow(int a,int b)
{
int ans = 1;
while(b)
{
if(b&1)
{ ans = ans*a%p; }
a = a*a%p;
b >>= 1;
}
return ans;
}
int ans;
signed main()
{
cin >> n;
for(re int i=1; i<=n; i++)
{ cin >> w[i]; }
for(re int i=2; i<=n; i++)
{ (ans += w[i]*quickpow(w[1]+w[i],p-2)%p) %= p; }
printf("%lld\n",ans+1);
return 0;
}
}
signed main()
{ return OMA::main(); }
T2
再次是读题死亡,依旧是加粗了没看见。
注意符咒是对每个1进行操作,考试时看错题,样例还一下就过了QAQ。
正解:
线段树合并/启发式合并
首先,肯定是先把法术用了,再去用符咒,考虑用完法术后的序列。
设 \(l\) 为左边第一个1的位置,\(r\) 为右边第一个1的位置,\(xam\) 为中间最长的一段连续的0的长度,则答案为 \(\max(m-r+l-1,xam)\) 。 \(Push\;up\) 的时候维护一下就好。
那如何判断无解情况? 只需判断
注意一些坑点:比如一个城池可以掌握多种法术。
Code
#include<cstdio>
#define MAX 100010
#define re register
namespace OMA
{
int n,m,q;
int ans[MAX];
struct graph
{
int next;
int to;
}edge[MAX];
int cnt=1,head[MAX];
inline void add(int u,int v)
{ edge[++cnt] = (graph){head[u],v},head[u] = cnt; }
struct Segmnet_Tree
{
int tot;
struct TREE
{
int l,r;
int xam;
int size;
int ls,rs;
inline int len()
{ return m-r+l-1; }
}st[MAX<<5];
int root[MAX];
#define ls(p) st[p].ls
#define rs(p) st[p].rs
inline int max(int a,int b)
{ return a>b?a:b; }
inline void Push_up(int p)
{
st[p].size = st[ls(p)].size+st[rs(p)].size;
st[p].xam = max(st[ls(p)].xam,st[rs(p)].xam);
if(st[ls(p)].r&&st[rs(p)].l)
{ st[p].xam = max(st[p].xam,st[rs(p)].l-st[ls(p)].r-1); }
if(st[ls(p)].l)
{ st[p].l = st[ls(p)].l; }
else
{ st[p].l = st[rs(p)].l; }
if(st[rs(p)].r)
{ st[p].r = st[rs(p)].r; }
else
{ st[p].r = st[ls(p)].r; }
}
inline void insert(int &p,int l,int r,int x)
{
p = !p?++tot:p;
if(l==r)
{ st[p].size = 1,st[p].l = st[p].r = l; return ; }
int mid = (l+r)>>1;
if(x<=mid)
{ insert(ls(p),l,mid,x); }
else
{ insert(rs(p),mid+1,r,x); }
Push_up(p);
}
inline int merge(int p1,int p2,int l,int r)
{
if(!p1||!p2)
{ return p1|p2; }
if(l==r)
{ return p1; }
int mid = (l+r)>>1;
ls(p1) = merge(ls(p1),ls(p2),l,mid);
rs(p1) = merge(rs(p1),rs(p2),mid+1,r);
Push_up(p1);
return p1;
}
inline void dfs(int u)
{
for(re int i=head[u],v; i; i=edge[i].next)
{
v = edge[i].to; dfs(v);
root[u] = merge(root[u],root[v],1,m);
}
ans[u] = st[root[u]].size?max(st[root[u]].xam,st[root[u]].len()):-1;
}
}Tree;
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 >> n >> m >> q;
for(re int i=1,u,v; i<=n-1; i++)
{ cin >> u >> v; add(u,v); }
for(re int i=1,u,p; i<=q; i++)
{ cin >> u >> p; Tree.insert(Tree.root[u],1,m,p); }
Tree.dfs(1);
for(re int i=1; i<=n; i++)
{ printf("%d\n",ans[i]); }
return 0;
}
}
signed main()
{ return OMA::main(); }
T3
完全图骗分 \(ans=(n-2)\times w\),20pts。
再加上一些乱搞,+10pts。
然而大帝 "乱搞" 有80pts orz。
正解:
神仙状压。
设 \(dp_{i,j}\) 表示当前所选集合为 \(i\) ,当前链的结尾为 \(j\),状态转移考虑两种情况:
在当前链的结尾新加入节点。
给当前链新增个集合/链上去。
预处理出每一种链边的总权值,某个链向另外一个链连边的权值。
或者说叫联通块?
关于状压枚举子集
设 \(S1\) 表示当前枚举的子集,\(S2\cup S1=S\),\(S\) 为全集。
for(re int s1=s; s1; s1=(s1-1)&s)
{ s2 = s^s1; }
Code
#include<cstdio>
#include<cstring>
#define re register
const int MAX = 1<<16;
namespace OMA
{
int n,m,top;
int dp[MAX][16],sum[MAX];
int dis1[16][16],dis2[MAX][16];
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 max(int a,int b)
{ return a>b?a:b; }
signed main()
{
cin >> n >> m;
for(re int i=1,u,v,w; i<=m; i++)
{
cin >> u >> v >> w;
dis1[u][v] = w,dis1[v][u] = w;
}
top = (1<<n)-1;
for(re int i=1; i<=top; i++)
{
for(re int j=1; j<=n; j++)
{
for(re int k=j+1; k<=n; k++)
{
if(((i>>(j-1))&1) and ((i>>(k-1))&1))
{ sum[i] += dis1[j][k]; }
}
}
}
for(re int i=1; i<=top; i++)
{
for(re int j=1; j<=n; j++)
{
if((i>>(j-1))&1)
{
for(re int k=1; k<=n; k++)
{
if(not ((i>>(k-1))&1) and dis1[j][k])
{ dis2[i][k] += dis1[j][k]; }
}
}
}
}
memset(dp,128,sizeof(dp)); dp[1][1] = 0;
for(re int i=1; i<=top; i++)
{
for(re int j=1; j<=n; j++)
{
if(dp[i][j]<0)
{ continue ; }
for(re int k=1; k<=n; k++)
{
if((not (i>>(k-1))&1) and dis1[j][k])
{ dp[i|(1<<(k-1))][k] = max(dp[i|(1<<(k-1))][k],dp[i][j]+dis1[j][k]); }
}
int l = i^top;
for(re int k=l; k; k=(k-1)&l)
{ dp[i|k][j] = max(dp[i|k][j],dp[i][j]+sum[k]+dis2[k][j]); }
}
}
printf("%d\n",sum[top]-dp[top][n]);
return 0;
}
}
signed main()
{ return OMA::main(); }
noip33的更多相关文章
随机推荐
- buu 相册
一.拖入jeb,这个神器里面,感觉对jeb使用还是不熟悉,对我逆向产生了一些障碍. 抓住题目给的提示,邮箱,全局直接搜索,mail. 看下它的交叉引用 找到了发邮件的方法, C2的MAILFROME说 ...
- 一款不错的 Go Server/API boilerplate,使用 K8S+DDD+CQRS+ES+gRPC 最佳实践构建
Golang API Starter Kit 该项目的主要目的是使用最佳实践.DDD.CQRS.ES.gRPC 提供样板项目设置. 为开发和生产环境提供 kubernetes 配置.允许与反映生产的 ...
- 开源的负载测试/压力测试工具 NBomber
负载测试和压力测试对于确保 web 应用的性能和可缩放性非常重要. 尽管它们的某些测试是相同的,但目标不同. 负载测试:测试应用是否可以在特定情况下处理指定的用户负载,同时仍满足响应目标. 应用在正常 ...
- C语言:sizeof判断数据类型长度
#include <stdio.h> int main() { short a = 10; int b = 100; long c=100; int short_length = size ...
- navicat for sqlserver 注册过程
1.安装原软件,不要打开软件2.将Navicat_Keygen_Patch_v3.4_By_DFoX_URET复制到软件安装位置,运行3.选择navicat v12 products:SQL Serv ...
- [刘阳Java]_处理并发有哪些方法
1.HTML静态化 ,将活动页面上的所有可以静态的元素全部静态化,并尽量减少动态元素2.禁止重复提交:用户提交之后按钮置灰,禁止重复提交3.用户限流:在某一时间段内只允许用户提交一次请求,比如可以采取 ...
- [刘阳Java]_easyui-panel组件入门级_第3讲
EasyUI中的panel组件在前面一节中我们简单告诉了大家代码如何写.这一节我们会从panel的入门级开始讲起走,重点包括它的事件监听,属性tool介绍 1. 事件监听-通过data-options ...
- ORB随便记一记
论文摘取 (这部分看的是泡泡机器人的翻译) 基于特征点.单目.完全自动初始化,基于PTAM框架. 相关工作 A.位置识别(大概是用于回环检测) bags of words FAB-map DBOW2 ...
- lucene 类介绍
lucene中重要的类: IndexWriter:lucene中最重要的类之一,主要用于索引的创建 Analyzer(抽象类):分析器,主要用于分析文本,常用的有StandardAnalyzer分析器 ...
- PAT乙级:1069 微博转发抽奖 (20分)
PAT乙级:1069 微博转发抽奖 (20分) 题干 小明 PAT 考了满分,高兴之余决定发起微博转发抽奖活动,从转发的网友中按顺序每隔 N 个人就发出一个红包.请你编写程序帮助他确定中奖名单. 输入 ...