質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Q&A

解決済

1回答

1983閲覧

AtCoderのABC134「Sequence Decomposing」がTLEになってしまう

yotayokuto

総合スコア10

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

0グッド

0クリップ

投稿2019/07/20 17:38

概要

AtCoderのABC134「Sequence Decomposing」で9/35件がTLEになってしまいます。(その他はACです。)
勉強不足で恐縮なのですが、計算量を減らす方法がわからず、ご教示いただけましたら幸いです。
何卒宜しくお願い致します。

問題

https://atcoder.jp/contests/abc134/submissions/6481452

コード

c++

1#include <iostream> 2#include <tuple> 3 4std::tuple<int, int> get_smaller_max(int a, int l[], int l_count) { 5 int max = -1, index = -1; 6 for (size_t i=0; i < l_count; i++) { 7 if (l[i] < a && l[i] > max) { 8 max = l[i]; 9 index = i; 10 } 11 } 12 return std::forward_as_tuple(max, index); 13} 14 15int main() { 16 int n; std::cin >> n; 17 18 int l[n], l_count = 0, max, index; 19 for (size_t i = 0; i < n; i++) { 20 int a; std::cin >> a; 21 std::tie(max, index) = get_smaller_max(a, l, l_count); 22 if (max == -1) { 23 l[l_count] = a; 24 l_count++; 25 } else { 26 l[index] = a; 27 } 28 } 29 30 std::cout << l_count << std::endl; 31 return 0; 32}

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

get_smaller_max 中で O(l_count) のループを回してますが、ここで二分探索を使って O(log(l_count)) に短縮するのがコツです。
これで全体でみると O(N*N) から O(N log N) に改善して想定解の計算量になります。

以下、変更例です。 vector 使っといたほうが見通し良くなる気はします。

cpp

1#include <iostream> 2#include <tuple> 3#include <algorithm> 4 5int get_smaller_max(int a, int l[], int l_count) { 6 int* ptr = std::upper_bound(l, l + l_count, a, std::greater<int>()); 7 int index = ptr - l; 8 if (index == l_count) return -1; 9 return index; 10} 11 12int main() { 13 int n; std::cin >> n; 14 15 int l[n], l_count = 0, max, index; 16 for (size_t i = 0; i < n; i++) { 17 int a; std::cin >> a; 18 int index = get_smaller_max(a, l, l_count); 19 if (index == -1) { 20 l[l_count] = a; 21 l_count++; 22 } else { 23 l[index] = a; 24 } 25 } 26 27 std::cout << l_count << std::endl; 28 return 0; 29}

投稿2019/07/20 18:32

set0gut1

総合スコア2413

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

yotayokuto

2019/07/21 23:14

ありがとうございます!! 色々無駄な記述もありましたね、、 初c++での挑戦だったのですが、次回もっと上手くできるように頑張ります
set0gut1

2019/07/22 03:47

ベストアンサーありがとうございます!自分は競プロはc++なんですが、競プロ以外では一切c++を使う機会がないタイプです。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問