#include <cstdio>#include <cstring>#include <cstdlib>#include <bitset>usingnamespacestd;#define MX 210typedefunsignedshortUS;intnCase,N,M,i,j,k,c[MX][16],s,ns,l,mx,pq,cq,v,ix;USeq,t[MX][MX],cmp;intmain(){scanf("%d",&nCase);while(nCase--){scanf("%d%d",&N,&M);memset(t,0,sizeof(t));memset(c,0,sizeof(c));for(i=1;i<=N;++i){scanf("%hd",&t[i][i]);//비트 갯수 확인for(cmp=1,j=0;j<16;++j,cmp<<=1)c[i][j]=c[i-1][j]+((t[i][i]&cmp)?1:0);//OR연산 캐싱for(j=i-1;j>=1;j--)t[j][i]=t[j+1][i]|t[j][j];}//start algorithmeq=65535;mx=-1;s=1;//대략적인 알고리즘 :: 다음 강화까지 미리 구해서, 1의 값이 가장 많은 인덱스를 찾는 방식.//1. 현재 장비상수와 i번째 카드 그룹과 i+1번째 카드그룹을 AND 연산을 한 뒤에, 1의 비트 갯수를 센다.(갯수 : N)//2. 그리고 AND 연산을 한 장비상수의 비트를 순회하면서, 각 비트의 1의 갯수가 남은 강화횟수보다 적을 경우에는 N의 값을 하나 감소 시킨다.//3. N의 값이 가장 큰것을 다음 강화에 쓰일 장비상수로 결정.//1~3을 거치고나면 i번째 강화 후에 나오는 장비상수를 얻는다.for(ix=1;ix<=M-1;++ix){l=N-M+ix;for(i=s;i<=l;++i){for(j=i+1;j<=l+1;++j){cq=eq&t[s][i]&t[i+1][j];intcnt=bitset<16>(cq).count();for(cmp=1,k=0;k<16;++k,cmp<<=1)if((cmp&cq)&&(c[N][k]-c[j][k]<M-s-1))cnt--;if(cnt>mx){mx=cnt;pq=cq;ns=i+1;}}}eq=pq;s=ns;mx=-1;}intret=bitset<16>(eq&t[s][N]).count();printf("%d\n",ret);}return0;}
juhosung
문제가 도저히 풀리지 않아서... 약간의 조언을 얻고자 질문 올립니다.
코드에 대한 설명과 알고리즘은 주석으로 간략하게 달았습니다.
확인해주시면 감사하겠습니다!
수고하세요.
10년 전