일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |
Tags
- 2021
- python 강좌
- 프로그래밍
- 자작시
- 백준
- 철학
- python
- 2020
- 2024
- 파이썬 강의
- BOJ
- 계획
- 파이썬 강좌
- python 강의
- 알고리즘
- 라즈베리파이 모니터
- 강의
- it
- C++
- mmdetection
- 다이나믹프로그래밍
- mmcv
- 공부
- 인생
- 강좌
- 자살
- 파이썬
- 슬픔
- dp
- dynamic programming
Archives
- Today
- Total
Stargazer
[백준] 1007번: 벡터 매칭 (수학) c++ 풀이 본문
반응형
접근:
우선 모든 벡터의 조합을 계산 해보면,
최대 C(20*19, 10) 인데 기하급수적인 크기의 수이다.
여기서 접근은 벡터의 덧셈 뺄셈의 성질을 이해하는 것이 중요하다.
벡터는 시작점과 종점으로 이루어져 있는데, 예를 들어 두 점을 (a,b) (c,d) 라고 하면,
벡터 v는 v = (c-a, d-b) = (c,d) - (b,a) 라고 표현 할 수 있다.
모든 벡터의 합은 v1+v2+...+vn = {(c1,d1)+(c2,d2) + ... + (cn ,dn)} - {(b1,a1) + (b2,a2) + ... + (bn, an)} 라고 볼 수 있다.
모든 점을 한 번씩 사용해야 하므로, 점이 n개가 있다면
n/2개의 점을 선택해서 다 더한 것에 남은 점들을 다 뺀 것과 같다.
이렇게 된다면, 최악의 경우 C(20,10) 의 조합으로 줄일 수 있게 된다.
전략:
브루트 포스 하게 모든 경우의 수를 구해야한다.
모든 경우의 수를 dfs를 이용하여 n/2개를 선택하도록 하여(백트래킹), 선택한 것을 토대로 벡터를 구해 길이를 구하고,
각 경우마다 이전 최소값과 비교하여 최소값을 갱신한다.
구현:
#include <iostream>
#include <vector>
#include <cmath>
using namespace std;
int tc, n;
int x, y;
pair<int,int> dot[21];
pair<int,int> group1;
pair<int,int> group2;
double min_length =1000000000.0;
bool visit[21]{};
double length(){
group1 = group2 = {0,0};
for(int i=0;i<n;i++){
if(visit[i]){
group1.first += dot[i].first;
group1.second += dot[i].second;
}else{
group2.first += dot[i].first;
group2.second += dot[i].second;
}
}
return sqrt(pow(group1.first-group2.first,2) + pow(group1.second-group2.second,2));
}
void dfs(int remain, int startidx){
if(remain == 0){
double len = length();
if(min_length > len){
min_length = len;
}
}else{
for(int i=startidx;i<n;i++){
if(!visit[i]){
visit[i] = true;
dfs(remain-1,i);
visit[i] = false;
}
}
}
}
int main(){
ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
cin >> tc;
for(int test=0;test<tc;test++){
cin >> n;
min_length =1000000000.0;
for(int i=0;i<n;i++){
cin >> dot[i].first >> dot[i].second; // first: x, second:
}
dfs(n/2, 0);
cout << fixed;
cout.precision(7);
cout << min_length << '\n';
}
return 0;
}
반응형
'Undergraudate basics(학부생 기초) > 자료구조, 알고리즘' 카테고리의 다른 글
[백준] 1305번 : 광고 (KMP 알고리즘) c++ 풀이 (0) | 2022.05.29 |
---|---|
[백준] 1202번 : 보석 도둑 (그리디 알고리즘) C++ 풀이 (0) | 2022.05.20 |
[백준] 1197번 : 최소 스패닝 트리 c++ 풀이 (0) | 2022.05.17 |
[백준] 9465번: 스티커 C++ 풀이 (DP) (0) | 2022.05.13 |
[백준] 1991번: 트리순회 C++ 풀이 (0) | 2022.05.13 |
Comments