Problem Solving/문제풀이

NYPC 예선 후기

Ryute 2018. 8. 31. 22:51

세줄요약

1. 머리나쁘고

2. 게을러서

3. 조짐


 이번 NYPC 예선은 내가 공부를 엄청 게을리 하고 있다는 증거를 잔뜩 남겨두고 떠났다. 전체적으로 맞는 풀이인데 구현을 못하거나/반례가 있는 줄 알고 해결하지 못한 문제들이 너무 많았다. 해결하지 못한 문제는 16번 쉴드 생성기, 19번 종이접기, 21번 금 줄 게임, 22번 보라색 영역이다.

 점수를 계산해보면 2,135점으로 본선 가기는 그른듯.


 다음은 내가 해결한 문제들에 대한 간단한 해설과 소스코드이다.


3. 아이템 구매

간단한 일차방정식을 해결하는 문제이다. Naive하게 코딩할 경우 시간복잡도는 선형으로 나온다. 확장 유클리드 알고리즘을 이용해서 로그 시간복잡도로 해결할 수 있을 것 같다는 느낌이 들긴 하는데 그렇게 코딩할 이유는 없으니까 ^^7


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
#include <bits/stdc++.h>
 
using namespace std;
 
int main()
{
    int p,q,w;
    scanf("%d %d %d",&p,&q,&w);
    for(int i=0;;i++)
    {
        int left=w-p*i;
        if(left<0return -1;
        if(left%q==0)
        {
            printf("%d %d",i,left/q);
            return 0;
        }
    }
    return -1;
}
 
cs

4. 승리팀 찾기

문제에서 요구하는 대로 그대로 코딩해주면 된다.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
#include <bits/stdc++.h>
 
using namespace std;
 
typedef tuple<int,int,int> tp;
typedef pair<tp,int> player;
const int score[]={10,8,6,5,4,3,2,1};
 
bool tenSec(tp a, tp b)
{
    int ta=get<0>(b)-get<0>(a);
    int tb=get<1>(b)-get<1>(a);
    int tc=get<2>(b)-get<2>(a);
    if(tc<0) tb--;
    if(tb<0) tb+=60,ta--;
 
    if(ta>0return false;
    if(tb>=10return false;
    return true;
}
 
void solve()
{
    char sel[10];
 
    scanf("%s",sel);
 
    vector<player> V;
    for(int i=0;i<8;i++)
    {
        char team[10];
        int m,s,ms;
        scanf(" %s %d:%d.%d",team,&m,&s,&ms);
        V.push_back(make_pair(tie(m,s,ms),strcmp("red",team)!=0));
    }
 
    sort(V.begin(),V.end());
 
    if(strcmp("speed",sel)==0)
    {
        int sc[2]={};
        for(int i=0;i<8;i++)
            if(tenSec(V[0].first,V[i].first))
                sc[V[i].second]+=score[i];
        if(sc[0]==sc[1])
            printf("%s\n",V[0].second==0?"red":"blue");
        else
            printf("%s\n",sc[0]>sc[1]?"red":"blue");
    }
    else
    {
        printf("%s\n",V[0].second==0?"red":"blue");
    }
}
 
int main()
{
    int tc;
    scanf("%d",&tc);
    while(tc--) solve();
    return 0;
}
 
cs

5. 줄임말

단어에서 맨 앞글자만 따 주는 프로그램을 만드는 것인데, UTF-8 인코딩은 문자열 한글 처리가 굉장히 번거롭다. 유일한 아웃풋 only 문제라는 것을 잘 이용해서 그냥 상용 프로그램으로 유니코드로 바꿔준 다음(그럼 한글이 항상 2바이트) 그대로 코딩하면 된다. 제발 이런문제좀 안나왔으면 좋겠다. 


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
#include <bits/stdc++.h>
 
using namespace std;
 
int main()
{
    int n;
    freopen("input_test.txt","r",stdin);
    freopen("output_test.txt","w",stdout);
    scanf("%d\n",&n);
    char str[3000];
    while(n--)
    {
        bool pr=true;
        fgets(str,sizeof(str),stdin);
        int len=strlen(str);
        for(int i=0; i<len; i++)
        {
            if(str[i]==' ')
            {
                pr=true;
                continue;
            }
            if(!pr) continue;
            if(('A'<=str[i] && str[i]<='Z'|| ('a'<=str[i] && str[i]<='z'))
                printf("%c",str[i]);
            else
            {
                printf("%c%c",str[i],str[i+1]);
                i++;
            }
 
            pr=false;
        }
        printf("\n");
    }
    return 0;
}
 
cs

6. 우물왕 김배찌

x좌표와 y좌표의 분산을 각각 최소화하는 문제이다. 분산을 최소화하는 값은 모든 값의 평균이라는 사실이 매우 잘 알려져 있다. 끝!


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
#include <bits/stdc++.h>
 
using namespace std;
typedef long long ll;
 
int main()
{
    int n;
    ll xs=0, ys=0;
    scanf("%d",&n);
    for(int i=0;i<n;i++)
    {
        ll t1,t2;
        scanf("%lld %lld",&t1,&t2);
        xs+=t1;
        ys+=t2;
    }
    printf("%f %f",(double)xs/(double)n,(double)ys/(double)n);
    return 0;
}
 
cs

7. 최고의 동접 구간을 찾아라!

사람들이 접속할 수 있는 시간이 최대 1440분밖에 안된다는 것을 이용해서 Prefix Sum 비슷하게 해결하면 된다. 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
#include <bits/stdc++.h>
 
using namespace std;
 
int main()
{
    vector<int> v(1440);
    vector<int> pn(1440);
 
    int n,ma=0;
    scanf("%d",&n);
    for(int i=0;i<n;i++)
    {
        int t1,t2;
        scanf("%d:%d",&t1,&t2);
        v[t1*60+t2]++;
        scanf("%d:%d",&t1,&t2);
        v[t1*60+t2]--;
    }
    partial_sum(v.begin(),v.end(),pn.begin());
    int bf=-1,st,en,mst,men;
    for(int i=0;i<1440;i++)
    {
        if(bf!=pn[i])
        {
            en=i-1;
            if(ma<bf)
            {
                ma=bf;
                mst=st;
                men=en;
            }
            st=i;
        }
        bf=pn[i];
    }
 
    men++;
    printf("%d\n%02d:%02d %02d:%02d",ma,mst/60,mst%60,men/60,men%60);
    return 0;
}
 
cs

8. 버튼 게임

유클리드 알고리즘을 그대로 번형시킨 문제이다. a!=0이어야 한다는 사실에 주의하자.


1
2
3
4
5
6
7
8
9
10
11
12
13
#include <bits/stdc++.h>
 
using namespace std;
 
int main()
{
    int a,b;
    scanf("%d %d",&a,&b);
    if(__gcd(a,b)==&& a!=0printf("POSSIBLE");
    else printf("IMPOSSIBLE");
    return 0;
}
 
cs

9. 이진 트리

$DP(n,k)$를 $n$개의 노드로 높이가 $k$ 이하인 만들 수 있는 이진 트리 개수라고 정하자. 그러면 왼쪽 서브트리와 오른쪽 서브트리의 높이를 달리 하며 계산해줄 수 있다. 시간복잡도는 $O(nk^2)$. 답은 당연히 $DP(n,k)-DP(n,k-1)$이 된다. 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
#include <bits/stdc++.h>
 
using namespace std;
 
typedef long long ll;
 
const ll MOD=1e9+7;
 
ll cache[505][505];
ll n,k;
 
ll dp(ll re, ll h)
{
    if(h<0return 0;
    if(re==0return 1;
    ll& ret=cache[re][h];
    if(ret!=-1return ret;
 
    ll s=0;
    for(ll i=0;i<=re-1;i++)
        s=(s+(dp(i,h-1)*dp(re-i-1,h-1))%MOD)%MOD;
    return ret=s;
}
 
int main()
{
    memset(cache,-1,sizeof(cache));
    scanf("%lld %lld",&n,&k);
    ll ans=dp(n,k)-dp(n,k-1);
    if(ans<0) ans+=MOD;
    printf("%lld",ans);
    return 0;
}
 
cs

10. 캐릭터 경험치

역추적을 해야 하는 냅색이다. 더 이상의 설명이 필요한지?


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
#include <bits/stdc++.h>
 
using namespace std;
 
int s,n;
int cache[1005][10005];
int cost[1005];
int expt[1005];
 
int dp(int idx, int stm)
{
    if(idx==n) return 0;
    int& ret=cache[idx][stm];
    if(ret!=-1return ret;
 
    if(stm-cost[idx]<0return ret=dp(idx+1,stm);
    else return ret=max(dp(idx+1,stm),dp(idx+1,stm-cost[idx])+expt[idx]);
}
 
vector<int> ans;
void track(int idx, int stm)
{
    if(idx==n) return;
    if(dp(idx+1,stm-cost[idx])+expt[idx]==dp(idx,stm) && stm-cost[idx]>=0)
    {
        ans.push_back(idx);
        track(idx+1,stm-cost[idx]);
    }
    else
        track(idx+1,stm);
}
 
int main()
{
    memset(cache,-1,sizeof(cache));
    scanf("%d %d",&s,&n);
    for(int i=0;i<n;i++)
        scanf("%d %d",&cost[i],&expt[i]);
    printf("%d\n",dp(0,s));
    track(0,s);
    printf("%d\n",ans.size());
    for(int i=0;i<ans.size();i++)
        printf("%d ",ans[i]+1);
    return 0;
}
 
cs

11. 전염병 역학 조사

시간의 범위가 $10^6$으로 비교적 짧으니 매 시간마다 간선을 추가하고 지우는 것을 반복하면서 BFS를 돌려주면 된다. 감염되는 사람만을 큐에 넣으면 모든 인간은 단 한번 큐에 들어가므로 시간복잡도는 $O(10^6+V+E)$ 인것 같다. 아마도?


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
#include <bits/stdc++.h>
 
using namespace std;
typedef vector<int> vi;
typedef pair<int,int> pii;
 
vector<vector<pii> > meet(1000005,vector<pii>());
vi virus;
vector<vi> adj;
 
void add_edge(queue<int> &Q, int ti)
{
    for(int i=0;i<meet[ti].size();i++)
    {
        pii here=meet[ti][i];
        adj[here.first].push_back(here.second);
        adj[here.second].push_back(here.first);
        if(virus[here.first] && !virus[here.second])
        {
            virus[here.second]=1;
            Q.push(here.second);
        }
        if(virus[here.second] && !virus[here.first])
        {
            virus[here.first]=1;
            Q.push(here.first);
        }
    }
}
 
void erase_edge(int ti)
{
    for(int i=0;i<meet[ti].size();i++)
    {
        pii here=meet[ti][i];
        adj[here.first].pop_back();
        adj[here.second].pop_back();
    }
}
 
int main()
{
    int n,m;
    scanf("%d %d",&n,&m);
    adj.assign(n+1,vi());
    virus.assign(n+1,0);
    for(int i=0;i<m;i++)
    {
        int t1,t2,tt;
        scanf("%d %d %d",&t1,&t2,&tt);
        meet[tt].push_back(make_pair(t1,t2));
    }
    virus[1]=1;
    for(int i=1;i<=1e6;i++)
    {
        if(meet[i].size()==0continue;
        queue<int> Q;
        add_edge(Q,i);
        while(!Q.empty())
        {
            int here=Q.front(); Q.pop();
            for(int j=0;j<adj[here].size();j++)
            {
                int there=adj[here][j];
                if(virus[there]) continue;
                virus[there]=1;
                Q.push(there);
            }
        }
        erase_edge(i);
    }
    for(int i=1;i<=n;i++)
        if(virus[i]) printf("%d ",i);
    return 0;
}
 
cs

12. 강력한 한방, 필살기

강한필 조교님 사랑합니다


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
#include <bits/stdc++.h>
 
using namespace std;
 
typedef struct square
{
    char checker;
    int atk, def;
 
    void init_square(char _checker, int _atk, int _def)
    {
        checker=_checker;
        atk=_atk;
        def=_def;
    }
}Square;
 
Square board[5][5];
Square temp[5][5];
Square remain[15];
 
int cal_score()
{
    int s=0;
    for(int i=1;i<=4;i++)
    {
        bool check=true;
        for(int j=1;j<=3;j++)
            if(temp[i][j].checker!='M')
                check=false;
        if(check)
        {
            for(int j=1;j<=3;j++)
                s+=temp[i][j].atk;
        }
    }
 
    for(int j=1;j<=3;j++)
    {
        bool check=true;
        for(int i=1;i<=3;i++)
            if(temp[i][j].checker!='M')
                check=false;
        if(check)
        {
            for(int i=1;i<=3;i++)
                s+=temp[i][j].atk;
        }
        check=true;
        for(int i=2;i<=4;i++)
            if(temp[i][j].checker!='M')
                check=false;
        if(check)
        {
            for(int i=2;i<=4;i++)
                s+=temp[i][j].atk;
        }
    }
 
    for(int i=0;i<=1;i++)
    {
        bool check=true;
        for(int j=1;j<=3;j++)
        {
            if(temp[i+j][j].checker!='M')
                check=false;
        }
        if(check)
        {
            for(int j=1;j<=3;j++)
                s+=temp[i+j][j].atk;
        }
    }
 
    for(int i=0;i<=1;i++)
    {
        bool check=true;
        for(int j=1;j<=3;j++)
        {
            if(temp[i+j][4-j].checker!='M')
                check=false;
        }
        if(check)
        {
            for(int j=1;j<=3;j++)
                s+=temp[i+j][4-j].atk;
        }
    }
 
    return s;
}
 
bool canMove(Square a, Square b)
{
    if(b.checker=='X'return true;
    if(b.checker=='M'return false;
    return (a.atk>b.def) && (a.def>b.atk);
}
 
int main()
{
    for(int i=1;i<=4;i++)
    {
        for(int j=1;j<=3;j++)
        {
            char sel;
            int t1=0,t2=0;
            scanf(" %c",&sel);
            if(sel!='X')
                scanf("%d/%d",&t1,&t2);
            board[i][j].init_square(sel,t1,t2);
            temp[i][j].init_square(sel,t1,t2);
        }
    }
 
 
    int n;
    scanf("%d",&n);
    for(int i=0;i<n;i++)
    {
        char sel;
        int t1=0,t2=0;
        scanf(" %c",&sel);
        if(sel!='X')
            scanf("%d/%d",&t1,&t2);
        remain[i].init_square(sel,t1,t2);
    }
    int ma=0;
    for(int x=1;x<=4;x++)
    {
        for(int y=1;y<=3;y++)
        {
            if(board[x][y].checker=='O'continue;
            if(board[x][y].checker=='X')
            {
                for(int i=0;i<n;i++)
                {
                    temp[x][y]=remain[i];
                    ma=max(ma,cal_score());
                    temp[x][y]=board[x][y];
                }
            }
            if(board[x][y].checker=='M')
            {
                for(int i=-1;i<=1;i++)
                {
                    for(int j=-1;j<=1;j++)
                    {
                        int nx=x+i;
                        int ny=y+j;
                        if(nx<|| nx>4continue;
                        if(ny<|| ny>3continue;
 
                        if(!canMove(board[x][y],board[nx][ny]) && !(x==nx && y==ny)) continue;
                        temp[x][y].init_square('X',0,0);
                        temp[nx][ny]=board[x][y];
                        ma=max(ma,cal_score());
                        temp[x][y]=board[x][y];
                        temp[nx][ny]=board[nx][ny];
                    }
                }
            }
        }
    }
    printf("%d",ma);
    return 0;
}
 
cs

13. 회전 격자판

버블 정렬 비슷한걸 돌아가는 격자판에서 하는 문제다. 이유는 잘 모르겠는데 한번 정렬이 되면 4번 주기로 변하지 않는 상태가 되는것 같았다. 그래서 냈더니 맞았다. 덕분에 코드가 개판


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
#include <bits/stdc++.h>
 
using namespace std;
typedef vector<int> vi;
 
int n,k;
vector<vi> arr;
 
void print()
{
    for(int i=1;i<=n;i++)
    {
        for(int j=1;j<=n;j++)
            printf("%d ",arr[i][j]);
        printf("\n");
    }
    printf("\n");
}
 
void modify()
{
    vector<vi> a=arr;
    for(int i=1;i<=n;i++)
        for(int j=1;j<=n-1;j++)
            if(a[i][j] > a[i][j+1])
                swap(a[i][j],a[i][j+1]);
    for(int i=1;i<=n;i++)
        for(int j=1;j<=n;j++)
            arr[i][j]=a[j][n-i+1];
}
 
int main()
{
    scanf("%d %d",&n,&k);
    arr.assign(n+1,vi(n+1));
 
    for(int i=1;i<=n;i++)
        for(int j=1;j<=n;j++)
            scanf("%d",&arr[i][j]);
 
    //freopen("output.txt","w",stdout);
 
    //print();
    for(int i=1;i<=310;i++)
    {
        //printf("%d\n",i);
        modify();
        if(k<304 && i==k)
        {
            print();
            return 0;
        }
        if(k%4==i%&& i>=304)
        {
            print();
            return 0;
        }
    }
    return 0;
}
 
cs

14. 블록 숫자

BFS를 돌려주면 된다. 어떤 숫자가 결정되었을 때 새롭게 결정될 수 있는 숫자의 위치는 몇가지 되지 않는다. 모든 숫자는 최대 한번 결정되므로 시간복잡도는 $O(N^2)$.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
#include <bits/stdc++.h>
 
using namespace std;
typedef pair<int,int> pii;
 
int n;
int arr[1005][1005];
 
int da[]={1,1,-1,0,-1,0};
int db[]={0,1,-1,-1,0,1};
 
int mod(int x)
{
    if(x>=0return x%100;
    else return (x%100)+100;
}
 
bool update(int a, int b)
{
    if(a<|| a>n) return false;
    if(b<|| b>a) return false;
 
    if(arr[a][b]!=-1return false;
    if(arr[a][b-1]!=-&& arr[a-1][b-1]!=-1)
    {
        if(a!=&& b!=1)
        {
            arr[a][b]=mod(arr[a-1][b-1]-arr[a][b-1]);
            return true;
        }
    }
    if(arr[a][b+1]!=-&& arr[a-1][b]!=-1)
    {
        if((b+1)<=&& a!=1)
        {
            arr[a][b]=mod(arr[a-1][b]-arr[a][b+1]);
            return true;
        }
    }
    if(arr[a+1][b]!=-&& arr[a+1][b+1]!=-1)
    {
        if((a+1)<=n)
        {
            arr[a][b]=mod(arr[a+1][b]+arr[a+1][b+1]);
            return true;
        }
    }
    return false;
}
 
int main()
{
    scanf("%d",&n);
    for(int i=1;i<=n;i++)
        for(int j=1;j<=i;j++)
            scanf("%d",&arr[i][j]);
 
    queue<pii> Q;
    for(int i=1;i<=n;i++)
        for(int j=1;j<=i;j++)
            if(update(i,j))
                Q.push(make_pair(i,j));
    while(!Q.empty())
    {
        pii here=Q.front(); Q.pop();
        for(int i=0;i<6;i++)
            if(update(here.first+da[i],here.second+db[i]))
                Q.push(make_pair(here.first+da[i],here.second+db[i]));
    }
    for(int i=1;i<=n;i++)
        for(int j=1;j<=i;j++)
            printf("%d%c",arr[i][j]," \n"[i==j]);
    return 0;
}
 
cs

15. Flood-It

결국은 맨 마지막의 상태만 확인해 주면 되므로 그냥 다익스트라를 돌리면 된다. 가중치를 내가 지금 수행한 쿼리의 번호로부터 지금 내가 앞으로 이동할 위치에 있는 숫자의 가장 빠른 쿼리 번호가 뭔지 생각해주면서 설정하면 된다. 코드를 보자.


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
#include <bits/stdc++.h>
 
using namespace std;
 
typedef pair<int,int> pii;
typedef pair<int,pii> piii;
 
const int INF=100000;
 
int arr[1005][1005];
int query[1005];
int idx[1005][10];
int dist[1005][1005];
 
int n,k;
 
int dx[]={1,0,-1,0};
int dy[]={0,1,0,-1};
 
int main()
{
    scanf("%d %d",&n,&k);
    for(int i=1;i<=k;i++)
        scanf("%d",&query[i]);
    for(int i=1;i<=n;i++)
        for(int j=1;j<=n;j++)
            scanf("%d",&arr[i][j]);
    query[0]=arr[1][1];
 
    for(int i=1;i<=1000;i++)
        for(int j=1;j<=1000;j++)
            dist[i][j]=INF;
 
    memset(idx,-1,sizeof(idx));
    for(int i=0;i<=k;i++)
    {
        idx[i][query[i]]=i;
        for(int j=i-1;j>=0;j--)
        {
            if(query[j]==query[i]) break;
            idx[j][query[i]]=i;
        }
    }
 
    priority_queue<piii,vector<piii>,greater<piii> > Q;
    Q.push(make_pair(0,make_pair(1,1)));
    dist[1][1]=0;
    while(!Q.empty())
    {
        piii temp=Q.top(); Q.pop();
        int hidx=temp.first;
        int x=temp.second.first;
        int y=temp.second.second;
 
        if(hidx>dist[x][y]) continue;
 
        for(int i=0;i<4;i++)
        {
            int nx=x+dx[i];
            int ny=y+dy[i];
            if(nx<|| nx>n) continue;
            if(ny<|| ny>n) continue;
 
            int tidx=idx[hidx][arr[nx][ny]];
            if(tidx==-1continue;
            if(tidx>=dist[nx][ny]) continue;
            Q.push(make_pair(tidx,make_pair(nx,ny)));
            dist[nx][ny]=tidx;
        }
    }
    int s=0;
    for(int i=1;i<=n;i++)
        for(int j=1;j<=n;j++)
            if(dist[i][j]!=INF)
                s++;
    printf("%d",s);
    return 0;
}
 
cs

16. 실드 생성기

미리 전처리를 해 놓으면 $O(N^3)$에 된다는 듯 하다. 추후에 풀이 이해하고 다시 올리겠음.


17. 퍼즐 콤보

문자열을 (문자, 문자 개수) 형태로 나타낸 리스트로 바꾼 다음 잘 지워주면 된다. 리스트에서 원소를 삭제할 때 순서를 잘 지켜줘야 한다. 근데 이것보다 더 쉽게 코딩하는 방법이 분명 있을듯

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
#include <bits/stdc++.h>
 
using namespace std;
 
typedef pair<char,int> block;
typedef list<block>::iterator b_it;
 
int n;
char str[1000005];
 
int main()
{
    scanf("%d",&n);
    scanf("%s",str);
 
    list<block> L;
    vector<b_it> V;
 
    L.push_back(make_pair('@',0));
    char bf=str[0]; int sz=1;
    for(int i=1;i<=n;i++)
    {
        if(i<&& bf==str[i])
            sz++;
        else
        {
            L.push_back(make_pair(bf,sz));
            auto lb=L.end(); lb--;
            if(sz>=3) V.push_back(lb);
            sz=1;
        }
        bf=str[i];
    }
    L.push_back(make_pair('@',0));
 
    int combo=0;
    while(true)
    {
        if(V.empty()) break;
        vector<b_it> temp;
        vector<b_it> nextBlock;
 
        for(int i=0;i<V.size();i++)
        {
            b_it here=V[i];
            b_it l_here=V[i]; l_here--;
            L.erase(here);
            if(!temp.empty() && temp.back()==l_here) continue;
            temp.push_back(l_here);
        }
 
        for(int i=0;i<temp.size();i++)
        {
            b_it left=temp[i];
            b_it right=temp[i]; right++;
 
            if(left->first=='@' || right->first=='@')
                continue;
 
            if(left->first==right->first)
            {
                assert(right->second<3);
                if(left->second<&& left->second+right->second>=3)
                    nextBlock.push_back(right);
                else if(left->second>=3)
                {
                    nextBlock.pop_back();
                    nextBlock.push_back(right);
                }
                right->second+=left->second;
                L.erase(left);
            }
        }
        V=nextBlock;
        combo++;
    }
 
    printf("%d\n",combo);
    if(L.size()==2)
        printf("PERFECT!!!");
    else
    {
        for(b_it it=L.begin();it!=L.end();it++)
            for(int j=0;j<it->second;j++)
                printf("%c",it->first);
    }
    return 0;
}
 
cs

18. 피파 온라인 드리블 튜토리얼
공은 사실 언제든지 찰 수 있으니 그냥 장애물이랑 만났을 때 뚫고 지나가면서 공으로 장애물을 부순다고 생각해도 상관이 없다. 단 오른쪽이나 왼쪽으로 이동하면서 부수진 못한다. 그러면 간단하게 동적계획법으로 해결할 수 있다. 그러면 $O(NM)$이 된다.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
#include <bits/stdc++.h>
 
using namespace std;
 
const int INF=100000;
 
int n,m,k;
char arr[505][55];
int cache[505][55];
 
int dp(int a,int b)
{
    if(b<|| b>m) return INF;
    if(a==k) return 0;
 
    int &ret=cache[a][b];
    if(ret!=-1return ret;
 
    int mi=INF;
    for(int i=-1;i<=1;i++)
    {
        int temp=dp(a-1,b+i)+(arr[a-1][b+i]=='#'?1:0);
        if(i!=&& arr[a][b+i]=='#'continue;
        mi=min(mi,temp);
    }
 
    return ret=mi;
}
 
int main()
{
    memset(cache,-1,sizeof(cache));
    scanf("%d %d %d",&n,&m,&k);
    for(int i=1;i<=n;i++)
        for(int j=1;j<=m;j++)
            scanf(" %c",&arr[i][j]);
 
    printf("%d",dp(n,1));
    return 0;
}
 
cs

19. 종이접기

풀기 싫어서 안풀었다. 풀걸


20. 봇 탐지

아무래도 내가 푼 풀이가 정해가 아닌 것 같다. 단순히 해싱을 하면 시간이 모자라고, 길이에 대해서 map을 따로 선언해 주면 $N=2950$정도까지는 뚫을 수 있다. 여기서 최적화하려고 hash_map도 써보고 롤링 해시 곱하는 수도 바꿔보고 17 곱하는거 비트쉬프트로 최적화 한다고 뻘짓하고 반복문 변수 밖으로 빼고 맵 순회하는것도 그때그때 저장하는걸로 바꾸고 변수 접근시간도 줄이고 별별짓을 다했는데 그냥 메모리가 터지기 직전까지 map을 추가로 선언해 주고 해싱을 한번 더 해주면 바로 됐다. 가장 고생한 문제고, 정해를 진짜 알고싶다. 댓글로 제발 달아주세요


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
#include <bits/stdc++.h>
#include <hash_map>
 
using namespace std;
using namespace __gnu_cxx;
typedef pair<int,int> pii;
 
char str[3005];
int arr[3005];
int t[300];
hash_map<int,pii> M[1505][50];
 
int main()
{
    t['L']=1; t['R']=2; t['U']=3; t['D']=4;
 
    int n;
    int a,b;
    int i,j;
 
    scanf("%d %d %d",&n,&a,&b);
    scanf(" %s",str);
 
    for(i=0;i<n;i++)
        arr[i+1]=t[str[i]];
 
    int ma=0,tp,hv;
    for(i=1;i<=n;i++)
    {
        hv=0;
        for(j=i;j<=n;j++)
        {
            if((j-i+1)>=(n/2+1)) break;
            hv=((hv<<4)+hv+arr[j]);
            pii& p=M[j-i+1][abs(hv)%47][hv];
            if(p.first==|| (p.first+j-i)<i)
            {
                p.first=i;
                tp=++p.second;
                if(tp>=&& ma<(j-i+1)*tp) ma=(j-i+1)*tp;
            }
        }
    }
 
    if(ma>b) printf("1");
    else if(a<=ma && ma<=b) printf("2");
    else if(ma<a) printf("3");
    return 0;
}
 
cs


21. 금줄 게임

민컷인줄 알고 바로 손절했는데 처음 생각한 그리디 풀이가 맞았다는 이야기를 들었다. 빠른 자살


22. 보라색 영역

화성 지도에 포배하면 되겠지 하고 바로 시도했는데 코딩이 안돼서 실패했다. 이건 그냥 연습 부족... ㅠㅠ


총평)

 전체적으로 4~5회차에 시간이 부족하고 체력이 딸려서 시도를 많이 못해본게 아쉽다. 다음에 NYPC에 또 참가하게 되면 조금 더 열심히 해볼것 같음. 사실 NYPC를 하면서 원샷 엔딩을 봤다는 것부터가 그냥 열심히 안했다는 것을 증명하고 있다. 문제를 푸는 법을 알아도 실제로 문제를 많이 풀어보지 않아도 코딩에서 막히거나 안될거라고 생각하고 지레짐작해서 틀리는 경우가 4문제중 3문제였다. (실드 빼고) 본선은 못갈거같긴 한데, 티셔츠정도는 기대해도 좋을거라고 행복회로를 빡세게 돌리고 있다. 다음엔 진짜 진짜 열심히 해야지.