102 条题解
-
5
PowderHan LV 10 @ 7 年前
-
27 年前@
貌似就我一个人傻傻的用dfs。贴一下吧,虽然速度不咋地- -
//位运算+dfs -
27 年前@
状态
耗时
内存占用
#1 Accepted 4ms 2.469MiB
#2 Accepted 3ms 2.5MiB
#3 Accepted 3ms 2.375MiB
#4 Accepted 3ms 2.5MiB
#5 Accepted 4ms 2.375MiB
#6 Accepted 3ms 2.375MiB
#7 Accepted 3ms 2.379MiB
#8 Accepted 4ms 2.375MiB
#9 Accepted 6ms 6.367MiB
#10 Accepted 4ms 2.5MiB -
17 年前@
-
17 年前@
此题由于n<=15,而2^15<35000因此可以使用状态压缩
那么可以想到一个简单的搜索,每一次判断可以用的补丁就补上。
这样的搜索加上hash判重以及最优性剪枝速度是比较快的。
代码如下:仔细思考,这样的搜索本质其实与最短路没有区别。
将状态看作点,状态转移看作边,时间是边权。
那么我们的hash判重其实就是spfa的inque数组,因此可以跑一遍最短路。
但实际上最短路更慢。 -
17 年前@
状态
耗时
内存占用
#1 Accepted 4ms 2.469MiB
#2 Accepted 3ms 2.5MiB
#3 Accepted 3ms 2.375MiB
#4 Accepted 3ms 2.5MiB
#5 Accepted 4ms 2.375MiB
#6 Accepted 3ms 2.375MiB
#7 Accepted 3ms 2.379MiB
#8 Accepted 4ms 2.375MiB
#9 Accepted 6ms 6.367MiB
#10 Accepted 4ms 2.5MiB -
03 年前@
#include<cstdio>
#include<iostream>
#include<sstream>
#include<algorithm>
#include<string>
#include<cstring>
#include<vector>
#include<queue>
#include<stack>
#include<set>
#include<map>
using namespace std;
const int inf=0x3f3f3f3f;
struct node
{
int tim;
vector<int>b1,b2,k1,k2;
}patch[110];
int n,m,ans=inf;
int vis[20];
int mi[25];
map<int,int>mapp;
bool check()
{
for(int i=1;i<=n;i++)
if(vis[i]) return false;
return true;
}
int Hash()
{
int sum=0;
for(int i=1;i<=n;i++)
sum=sum+mi[i]*vis[i];
return sum;
}
void dfs(int cost)
{
if(cost>=ans) return;
if(check())
{
ans=min(ans,cost);
return;
}
int temp[20];
for(int i=1;i<=n;i++)
temp[i]=vis[i];
for(int i=1;i<=m;i++)
{
int len=patch[i].b1.size(),flag=1;
for(int j=0;j<len;j++)
if(!vis[patch[i].b1[j]])
{
flag=0;
break;
}
if(flag)
{
len=patch[i].b2.size();
for(int j=0;j<len;j++)
if(vis[patch[i].b2[j]])
{
flag=0;
break;
}
}
if(!flag) continue;
len=patch[i].k2.size();
for(int j=0;j<len;j++)
vis[patch[i].k2[j]]=0;
len=patch[i].k1.size();
for(int j=0;j<len;j++)
vis[patch[i].k1[j]]=1;
int num=Hash();
if(!mapp.count(num))
{
mapp[num]=cost+patch[i].tim;
dfs(cost+patch[i].tim);
}
else if(mapp[num]>cost+patch[i].tim)
{
mapp[num]=cost+patch[i].tim;
dfs(cost+patch[i].tim);
}
for(int j=1;j<=n;j++)
vis[j]=temp[j];
}
}
int main()
{
int temp=1;
for(int i=1;i<=20;i++)
{
mi[i]=temp;
temp*=2;
}
cin>>n>>m;
string b,k;
for(int i=1;i<=m;i++)
{
cin>>patch[i].tim>>b>>k;
for(int j=0;j<n;j++)
if(b[j]=='+') patch[i].b1.push_back(j+1);
else if(b[j]=='-') patch[i].b2.push_back(j+1);
for(int j=0;j<n;j++)
if(k[j]=='+') patch[i].k1.push_back(j+1);
else if(k[j]=='-') patch[i].k2.push_back(j+1);
}
for(int i=1;i<=n;i++)
vis[i]=1;
mapp[Hash()]=0;
dfs(0);
if(ans==inf) cout<<"0";
else cout<<ans;
return 0;
} -
08 年前@
位运算+SPFA,每个状态为一个点
#include <cstdio>
#include <cstring>
#include <queue>
#define add(A,K) A|=1<<Kint st,end,time[200];
int n,m;
int B[200]={0},B_[200]={0};
int F[200]={0},F_[200]={0};void build(){
scanf("%d%d",&n,&m);
st=(1<<n)-1;
end=0;
char s1[100],s2[100];
for(int i=1;i<=m;i++){
scanf("%d",&time[i]);
scanf("%s%s",s1,s2);
for(int x=0;x<strlen(s1);x++){
if(s1[x]=='+')
add(B[i],x);
if(s1[x]=='-')
add(B_[i],x);
}
for(int x=0;x<strlen(s2);x++){
if(s2[x]=='+')
add(F[i],x);
if(s2[x]=='-')
add(F_[i],x);
}
}
}int isok(int x,int way){
int flag=1;
flag*=(B[way]&x)==B[way];
flag*=(B_[way]&x)==0;
return flag;
}int update(int x,int way){
x|=F[way];
x&=(~F_[way]);
return x;
}//SPFA
std::queue<int> q;
int inque[1<<15+10]={0};
int dist[1<<15+10];void spfa(){
dist[st]=0;
q.push(st);
inque[st]=1;
int t,temp;
while(!q.empty()){
t=q.front(),q.pop();
inque[t]=0;
for(int i=1;i<=m;i++){
if(!isok(t,i))
continue;
temp=update(t,i);
if(dist[t]+time[i]<dist[temp]){
dist[temp]=dist[t]+time[i];
if(!inque[temp]){
q.push(temp);
inque[temp]=1;
}
}
}
}
}int main(){
for(int i=0;i<=(1<<15)+5;i++)
dist[i]=999999999;
build();
spfa();
if(dist[end]==999999999)
printf("0");
else
printf("%d",dist[end]);return 0;
} -
08 年前@
-
08 年前@
数据真弱……
-
09 年前@
位运算+BFS,扩展时加判断,只有合法并且答案更优才加到队列里去,一直做到队列为空才输出答案。
//vijos p1019
#include <stdio.h>#define INF 2000000000
#define STATUS 0
#define STEP 1char function[120][20];
char condition[120][20];
int cost[120];int answer[1<<20];
int queue[1<<20][2];
int head, tail;void addToQueue(int status, int step){
if(answer[status] > step){
answer[status] = step;
queue[tail][STATUS] = status;
queue[tail][STEP] = step;
tail++;
}
}
int bfs(int numBug, int numPatch){
int status, step, i, j, tmp, target;
head = tail = 0;
target = (1<<numBug) - 1; //all bugs have been fixed
addToQueue(0, 0); //no bug has been fixed
while(head < tail){
status = queue[head][STATUS];
step = queue[head][STEP];
for(i=0; i<numPatch; i++){
//judge if the move is legal
tmp = 1;
for(j=0; j<numBug; j++){
switch(condition[i][j]){
case '+':
if((status>>j)&1)
tmp = 0;
break;
case '-':
if(!((status>>j)&1))
tmp = 0;
break;
}
if(tmp == 0)
break;
}
if(tmp == 0) //illegal
continue;//move
tmp = status;
for(j=0; j<numBug; j++){
switch(function[i][j]){
case '-':
tmp |= (1<<j);
break;
case '+':
tmp &= (target^(1<<j));
break;
}
}
addToQueue(tmp, step+cost[i]);
}
head++;
}
return answer[target]==INF ? 0:answer[target];
}
int main(){
int numBug, numPatch, i;scanf("%d %d", &numBug, &numPatch);
for(i=0; i<numPatch; i++)
scanf("%d %s %s", &cost[i], condition[i], function[i]);for(i=0; i<(1<<numBug); i++)
answer[i] = INF;printf("%d\n", bfs(numBug, numPatch));
return 0;
} -
09 年前@
program asddasf;
type po=^node;
node=record
a,b:longint;
c:po;
end;
var i,j,k,l,m,n,sum:longint;
s2,s1:array[0..100] of string;
a:array[0..100] of longint;
pa:array[0..400000] of po;
dis:array[-1..400000] of longint;
h:array[0..800000] of longint;
v:array[0..400000] of boolean;
p:po;
s,s0:ansistring;
ch:boolean;
function ff(x:string):longint;
var k,l:longint;
begin
ff:=0;
l:=1;
for k:=length(x) downto 1 do
begin
if x[k]='1' then
ff:=ff+l;
l:=l*2;
end;
exit(ff);
end;procedure chu(x1,y1:string;z1:longint);
var k,l,k1,k2:longint;
sn:ansistring;
begin
sn:=x1;
for j:=1 to n do
begin
if y1[j]='+' then
sn[j]:='1';
if y1[j]='-' then
sn[j]:='0';
end;
k1:=ff(x1);
k2:=ff(sn);
new(p);
p^.a:=k2;
p^.b:=z1;
p^.c:=pa[k1];
pa[k1]:=p;
end;
procedure sou(x,y:longint;z:string);
var j:longint;
begin
if y=n+1 then
begin
chu(z,s2[x],a[x]);
exit;
end;
if s1[x][y]='0' then
begin
sou(x,y+1,z+'1');
sou(x,y+1,z+'0');
end;
if s1[x][y]='-' then
sou(x,y+1,z+'0');
if s1[x][y]='+' then
sou(x,y+1,z+'1');
end;procedure spfa(x:longint);
var i,j,t,u,ans:longint;
begin
fillchar(dis,sizeof(dis),$7f div 3);
dis[x]:=0;
h[1]:=x;
v[x]:=true;
t:=0; u:=1;
repeat
inc(t);
ans:=h[t];
v[ans]:=false;
p:=pa[ans];
while p<>nil do
begin
if dis[p^.a]>dis[ans]+p^.b then
begin
dis[p^.a]:=dis[ans]+p^.b;
if v[p^.a]=false then
begin
inc(u);
h[u]:=p^.a;
v[h[u]]:=true;
end;
end;
p:=p^.c;
end;
until t=u;
end;begin
readln(n,m);
for i:=1 to m do
begin
readln(s);
j:=pos(' ',s);
s0:=copy(s,1,j-1);
val(s0,a[i]);
delete(s,1,j);
j:=pos(' ',s);
s1[i]:=copy(s,1,j-1);
delete(s,1,j);
s2[i]:=s;
end;
for i:=1 to m do
sou(i,1,'');
s:='';
for i:=1 to n do
s:=s+'1';
spfa(ff(s));
if dis[0]=dis[-1] then
writeln(0)
else
writeln(dis[0]);end.
-
010 年前@
按照补丁个数进行BFS,先不考虑时间问题。
对每个被修复的状态剪枝——如果BFS到当前状态耗时大于被记录的最小时间,就不继续BFS。
好像数据比较弱,这样就能最多15MS过了int bfs(){
int ret=iinf;
tmp.now=0,tmp.lv=0;
q.push(tmp);
while(!q.empty()){
tmp=q.front();
q.pop();
rep(i,0,m){
if(!( (bb2[i] == (tmp.now|bb2[i])) && (bb1[i] == (tmp.now&bb1[i])) ))
continue;ne.now=((tmp.now&ff2[i])|ff1[i]);
ne.lv=tmp.lv+w[i];
if(vst[ne.now]&&ne.lv>=cost[ne.now])
continue;vst[ne.now]=1;
cost[ne.now]=ne.lv;if(ne.now==ed)
ret=ne.lv;
else
q.push(ne);
}
}
return ret;
} -
011 年前@
高兴啊,提交了9次终于TM还是让我AC了!
测试数据 #0: Accepted, time = 0 ms, mem = 1904 KiB, score = 10
测试数据 #1: Accepted, time = 15 ms, mem = 1904 KiB, score = 10
测试数据 #2: Accepted, time = 0 ms, mem = 1904 KiB, score = 10
测试数据 #3: Accepted, time = 0 ms, mem = 1904 KiB, score = 10
测试数据 #4: Accepted, time = 0 ms, mem = 1908 KiB, score = 10
测试数据 #5: Accepted, time = 15 ms, mem = 1904 KiB, score = 10
测试数据 #6: Accepted, time = 15 ms, mem = 1904 KiB, score = 10
测试数据 #7: Accepted, time = 15 ms, mem = 1904 KiB, score = 10
测试数据 #8: Accepted, time = 0 ms, mem = 1908 KiB, score = 10
测试数据 #9: Accepted, time = 0 ms, mem = 1904 KiB, score = 10
Accepted, time = 60 ms, mem = 1908 KiB, score = 100 -
011 年前@
我擦,不想说了,发程序的人都TM二笔,全是编译错误
-
011 年前@
编译成功
测试数据 #0: Accepted, time = 0 ms, mem = 1264 KiB, score = 10
测试数据 #1: Accepted, time = 0 ms, mem = 1264 KiB, score = 10
测试数据 #2: Accepted, time = 0 ms, mem = 1260 KiB, score = 10
测试数据 #3: Accepted, time = 0 ms, mem = 1268 KiB, score = 10
测试数据 #4: Accepted, time = 0 ms, mem = 1264 KiB, score = 10
测试数据 #5: Accepted, time = 0 ms, mem = 1264 KiB, score = 10
测试数据 #6: Accepted, time = 0 ms, mem = 1264 KiB, score = 10
测试数据 #7: Accepted, time = 0 ms, mem = 1264 KiB, score = 10
测试数据 #8: Accepted, time = 0 ms, mem = 1264 KiB, score = 10
测试数据 #9: Accepted, time = 0 ms, mem = 1260 KiB, score = 10
Accepted, time = 0 ms, mem = 1268 KiB, score = 100
纪念一下 思路和毒药类似 但没有构图觉得会超空间
因为有使用条件 直接枚举即可 为不构图提供了条件
BFS+位运算 -
012 年前@
-
012 年前@
├ 测试数据 01:答案正确... (0ms, 7260KB)
├ 测试数据 02:答案正确... (0ms, 7260KB)
├ 测试数据 03:答案正确... (0ms, 7260KB)
├ 测试数据 04:答案正确... (0ms, 7260KB)
├ 测试数据 05:答案正确... (0ms, 7260KB)
├ 测试数据 06:答案正确... (0ms, 7260KB)
├ 测试数据 07:答案正确... (0ms, 7260KB)
├ 测试数据 08:答案正确... (0ms, 7260KB)
├ 测试数据 09:答案正确... (0ms, 7260KB)
├ 测试数据 10:答案正确... (0ms, 7260KB)VIJOS这碉堡了的评测机,同样的程序测了至少6遍才AC了。。。口口声声说我的0ms是TLE。。。
-
012 年前@
#01: Accepted (39ms, 504KB)
#02: Accepted (78ms, 504KB)
#03: Accepted (12ms, 504KB)
#04: Accepted (16ms, 504KB)
#05: Accepted (47ms, 504KB)
#06: Accepted (67ms, 504KB)
#07: Accepted (59ms, 504KB)
#08: Accepted (153ms, 504KB)
#09: Accepted (35ms, 508KB)
#10: Accepted (24ms, 508KB) -
012 年前@
编译通过...
├ 测试数据 01:答案正确... 0ms
├ 测试数据 02:答案正确... 0ms
├ 测试数据 03:答案正确... 0ms
├ 测试数据 04:答案正确... 0ms
├ 测试数据 05:答案正确... 0ms
├ 测试数据 06:答案正确... 0ms
├ 测试数据 07:答案正确... 0ms
├ 测试数据 08:答案正确... 0ms
├ 测试数据 09:答案正确... 0ms
├ 测试数据 10:答案正确... 0ms
---|---|---|---|---|---|---|---|-
Accepted 有效得分:100 有效耗时:0ms
/*
spfa 秒过 && 1A
用二进制表示状态,用数组记录第i个补丁是否可以用在状态j上(小优化)
话说vijos上pascal是王道么?还是c++用着顺手。
70行代码。spfa+位运算
*/