330 lines
10 KiB
HTML
330 lines
10 KiB
HTML
<!DOCTYPE html>
|
||
<html lang="ko">
|
||
<head>
|
||
<meta charset="UTF-8" />
|
||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||
<title>BOJ 30917 - Offline</title>
|
||
<style>
|
||
:root {
|
||
--bg: #fafaf8;
|
||
--paper: #ffffff;
|
||
--ink: #1e1f24;
|
||
--muted: #6a6d75;
|
||
--line: #d8dce3;
|
||
--accent: #0d6e6e;
|
||
--code-bg: #f4f6fb;
|
||
}
|
||
* { box-sizing: border-box; }
|
||
body {
|
||
margin: 0;
|
||
background:
|
||
radial-gradient(circle at 15% 0%, #f0efe9 0%, transparent 42%),
|
||
radial-gradient(circle at 85% 20%, #e7f1f2 0%, transparent 38%),
|
||
var(--bg);
|
||
color: var(--ink);
|
||
font-family: "Noto Sans KR", "Pretendard", "Apple SD Gothic Neo", sans-serif;
|
||
line-height: 1.65;
|
||
}
|
||
main {
|
||
max-width: 980px;
|
||
margin: 0 auto;
|
||
padding: 24px 16px 56px;
|
||
}
|
||
.header {
|
||
background: var(--paper);
|
||
border: 1px solid var(--line);
|
||
border-radius: 14px;
|
||
padding: 18px 20px;
|
||
margin-bottom: 18px;
|
||
}
|
||
.header h1 { margin: 0 0 6px; font-size: 1.5rem; }
|
||
.header p { margin: 0; color: var(--muted); font-size: 0.95rem; }
|
||
.header a { color: var(--accent); text-decoration: none; }
|
||
.section {
|
||
background: var(--paper);
|
||
border: 1px solid var(--line);
|
||
border-radius: 14px;
|
||
padding: 16px 18px;
|
||
margin-bottom: 14px;
|
||
overflow-x: auto;
|
||
}
|
||
h2 {
|
||
margin: 0 0 10px;
|
||
font-size: 1.05rem;
|
||
color: var(--accent);
|
||
border-bottom: 1px solid var(--line);
|
||
padding-bottom: 8px;
|
||
}
|
||
pre, code {
|
||
font-family: "JetBrains Mono", "Fira Code", monospace;
|
||
background: var(--code-bg);
|
||
}
|
||
pre {
|
||
padding: 12px;
|
||
border-radius: 10px;
|
||
border: 1px solid #e7ebf2;
|
||
overflow: auto;
|
||
}
|
||
blockquote {
|
||
margin: 14px 0;
|
||
padding: 16px 16px 14px 22px;
|
||
border-left: 4px solid var(--accent);
|
||
border-radius: 10px;
|
||
background: linear-gradient(90deg, #eef8f8 0%, #f9fdfd 100%);
|
||
color: #24313a;
|
||
font-weight: 600;
|
||
position: relative;
|
||
}
|
||
blockquote::before {
|
||
content: "“";
|
||
position: absolute;
|
||
left: 8px;
|
||
top: 2px;
|
||
font-size: 1.35rem;
|
||
line-height: 1;
|
||
color: #0b5f5f;
|
||
opacity: 0.7;
|
||
}
|
||
blockquote > :first-child { margin-top: 0; }
|
||
blockquote > :last-child { margin-bottom: 0; }
|
||
q {
|
||
color: #114f50;
|
||
font-weight: 700;
|
||
background: #edf8f8;
|
||
border-radius: 6px;
|
||
padding: 0 4px;
|
||
}
|
||
.math-inline math {
|
||
font-size: 1em;
|
||
vertical-align: middle;
|
||
}
|
||
.math-block {
|
||
margin: 10px 0;
|
||
padding: 8px 10px;
|
||
overflow-x: auto;
|
||
background: #f8fbff;
|
||
border: 1px solid #e2ecf8;
|
||
border-radius: 8px;
|
||
}
|
||
.math-block math {
|
||
font-size: 1.04em;
|
||
display: block;
|
||
}
|
||
table { border-collapse: collapse; width: 100%; }
|
||
th, td { border: 1px solid var(--line); padding: 6px 8px; }
|
||
img { max-width: 100%; height: auto; }
|
||
</style>
|
||
</head>
|
||
<body>
|
||
<main>
|
||
<header class="header">
|
||
<h1>A+B - 10 (제1편)</h1>
|
||
</header>
|
||
<article class="section">
|
||
<h2>문제</h2>
|
||
<p>1 이상 9 이하의 정수 A, B에 대해 A+B의 값을 출력해야 한다. 단, 이 문제는 <strong>인터랙티브 (상호작용)</strong> 문제이다. 이 문제에서는 A와 B의 값이 바로 주어지지 않고, 채점기와의 상호작용을 통해 그 값을 알아내야 한다.</p>
|
||
</article>
|
||
<article class="section">
|
||
<h2>출력</h2>
|
||
<p>여러분은 채점기에게 <strong>최대 18번</strong> 질문할 수 있다. 질문의 형태는 다음 중 하나이다.</p>
|
||
|
||
<ul>
|
||
<li><code>? A x</code> - A의 값이 x인지 물어본다.</li>
|
||
<li><code>? B x</code> - B의 값이 x인지 물어본다.</li>
|
||
</ul>
|
||
|
||
<p>예를 들어 A의 값이 4인지 물어보려면 <code>? A 4</code>를 출력하면 된다. 이때 x는 1 이상 9 이하의 정수여야 한다.</p>
|
||
|
||
<p>질문을 출력한 뒤 실제로 질문을 채점기에게 보내려면 <strong>표준 출력을 flush</strong>해야 한다. 자세한 내용은 아래의 <strong>노트</strong> 란을 참조하자.</p>
|
||
|
||
<p>질문을 보낸 뒤에는 채점기로부터 정수 하나를 입력받는다. 그 수는 질문의 답이 "예"인 경우 1, "아니요"인 경우 0이다.</p>
|
||
|
||
<p>A와 B의 값을 알아냈으면 A+B를 계산하여 다음과 같은 형태로 출력한다.</p>
|
||
|
||
<ul>
|
||
<li><code>! x</code> - A+B의 값은 x이다.</li>
|
||
</ul>
|
||
|
||
<p>예를 들어 A+B가 10인 경우 <code>! 10</code>을 출력하면 된다. 이는 질문 횟수에 포함되지 않는다.</p>
|
||
|
||
<p>아래의 "노트" 란에 여러 언어에 대한 예시 코드가 작성되어 있다. 단, 완성되지 않은 코드이므로 빈 공간은 직접 채워 넣어야 한다.</p>
|
||
</article>
|
||
<article class="section">
|
||
<h2>힌트</h2>
|
||
<p>상호작용 문제는 채점 방식이 복잡하기 때문에, 문제에 주어진 규약을 지키지 않았을 때 "틀렸습니다"가 아닌 다른 결과가 나올 수도 있음에 유의하자.</p>
|
||
|
||
<p><strong>예시 코드</strong></p>
|
||
|
||
<p>C</p>
|
||
|
||
<pre>
|
||
<code>#include <stdio.h>
|
||
int main() {
|
||
int resp;
|
||
for(int a=1; a<=9; a++){
|
||
// A가 a인지 물어보고 flush를 한다.
|
||
printf("? A %d\n", a);
|
||
fflush(stdout);
|
||
|
||
// 채점기의 답변을 입력받는다.
|
||
scanf("%d", &resp);
|
||
|
||
if(resp == 1){
|
||
// 답변이 "예"이므로 A의 값은 a이다.
|
||
// B의 값도 알아내야 하는데, 이 부분은 직접 채워보자.
|
||
int b = 0;
|
||
printf("! %d", a + b);
|
||
break;
|
||
}
|
||
}
|
||
}</code>
|
||
</pre>
|
||
|
||
<p>C++</p>
|
||
|
||
<pre>
|
||
<code>#include <iostream>
|
||
int main() {
|
||
int resp;
|
||
for(int a=1; a<=9; a++){
|
||
// A가 a인지 물어보고 flush를 한다.
|
||
// endl은 자동으로 flush도 해준다.
|
||
std::cout << "? A " << a << std::endl;
|
||
|
||
// 채점기의 답변을 입력받는다.
|
||
std::cin >> resp;
|
||
|
||
if(resp == 1){
|
||
// 답변이 "예"이므로 A의 값은 a이다.
|
||
// B의 값도 알아내야 하는데, 이 부분은 직접 채워보자.
|
||
int b = 0;
|
||
std::cout << "! " << a + b;
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
</code></pre>
|
||
|
||
<p>Python</p>
|
||
|
||
<pre>
|
||
<code>for a in range(1, 10):
|
||
# A가 a인지 물어보고 flush를 한다.
|
||
# print에 flush 파라미터를 넣으면 된다.
|
||
print("? A", a, flush=True)
|
||
|
||
# 채점기의 답변을 입력받는다.
|
||
resp = int(input())
|
||
|
||
if resp == 1:
|
||
# 답변이 "예"이므로 A의 값은 a이다.
|
||
# B의 값도 알아내야 하는데, 이 부분은 직접 채워보자.
|
||
b = 0
|
||
print("!", a + b)
|
||
break</code>
|
||
</pre>
|
||
|
||
<p>Rust</p>
|
||
|
||
<pre>
|
||
<code>fn read_i32() -> i32 {
|
||
let mut resp_s = String::new();
|
||
std::io::stdin().read_line(&mut resp_s).unwrap();
|
||
resp_s.trim().parse().unwrap()
|
||
}
|
||
|
||
fn main() {
|
||
for a in 1..=9 {
|
||
// A가 a인지 물어보고 flush를 한다.
|
||
// println!은 자동으로 flush도 해준다.
|
||
println!("? A {a}");
|
||
|
||
// 채점기의 답변을 입력받는다.
|
||
let resp = read_i32();
|
||
|
||
if resp == 1 {
|
||
// 답변이 "예"이므로 A의 값은 a이다.
|
||
// B의 값도 알아내야 하는데, 이 부분은 직접 채워보자.
|
||
let b = 0;
|
||
println!("! {}", a + b);
|
||
break;
|
||
}
|
||
}
|
||
}</code>
|
||
</pre>
|
||
|
||
<p>Java</p>
|
||
|
||
<pre>
|
||
<code>import java.io.*;
|
||
import java.util.*;
|
||
|
||
public class Main {
|
||
public static void main(String[] args) throws IOException{
|
||
Scanner sc = new Scanner(System.in);
|
||
for(int a=1; a<=9; a++){
|
||
// A가 a인지 물어보고 flush를 한다.
|
||
// System.out.println은 자동으로 flush도 해준다.
|
||
System.out.println("? A " + a);
|
||
|
||
// 채점기의 답변을 입력받는다.
|
||
int resp = sc.nextInt();
|
||
|
||
if(resp == 1){
|
||
// 답변이 "예"이므로 A의 값은 a이다.
|
||
// B의 값도 알아내야 하는데, 이 부분은 직접 채워보자.
|
||
int b = 0;
|
||
System.out.println("! " + (a + b));
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
}</code>
|
||
</pre>
|
||
|
||
<p><strong>표준 출력을 flush해야 하는 이유</strong></p>
|
||
|
||
<p>콘솔 혹은 파일 입출력은 사칙연산이나 값을 대입하는 등의 기본적인 연산에 비해 상대적으로 느린 작업이다. 따라서 많은 프로그래밍 언어의 기본 출력 기능은 출력을 요청받은 값들을 한 공간에 쌓아두고, 적당한 때에 한꺼번에 출력되도록 한다. 이 공간을 버퍼(buffer)라고 부른다.</p>
|
||
|
||
<p>그러나 이 문제에서는 채점기가 프로그램의 출력을 실시간으로 확인해야 답변을 줄 수 있다. 따라서 버퍼를 직접 비우는 작업이 필요하고, 이 작업을 flush라고 한다.</p>
|
||
|
||
<p><strong>예고편</strong></p>
|
||
|
||
<p><a href="/problem/30924">제2편</a>에서는 "적응하는 채점기"와 "적응하지 않는 채점기"의 차이를 다룬다. 제1편에 비해 어렵다는 점을 유의하자.</p>
|
||
|
||
<p>이 문제의 채점기는 적응하지 않는다.</p>
|
||
</article>
|
||
<article class="section">
|
||
<h2>예제 입력 1 복사</h2>
|
||
<pre class="sampledata" id="sample-input-1">
|
||
|
||
0
|
||
|
||
0
|
||
|
||
1
|
||
|
||
0
|
||
|
||
1
|
||
</pre>
|
||
</article>
|
||
<article class="section">
|
||
<h2>예제 출력 1 복사</h2>
|
||
<pre class="sampledata" id="sample-output-1">? A 1
|
||
|
||
? A 2
|
||
|
||
? A 3
|
||
|
||
? B 1
|
||
|
||
? B 2
|
||
|
||
! 5
|
||
</pre>
|
||
</article>
|
||
</main>
|
||
</body>
|
||
</html>
|