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

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

新規登録して質問してみよう
ただいま回答率
85.48%
データ構造

データ構造とは、データの集まりをコンピュータの中で効果的に扱うために、一定の形式に系統立てて格納する形式を指します。(配列/連想配列/木構造など)

C++

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

Q&A

解決済

2回答

1574閲覧

C++の参照渡しについて

cinnamoroll

総合スコア14

データ構造

データ構造とは、データの集まりをコンピュータの中で効果的に扱うために、一定の形式に系統立てて格納する形式を指します。(配列/連想配列/木構造など)

C++

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

0グッド

0クリップ

投稿2018/11/02 09:39

編集2018/11/02 10:03

前提・実現したいこと

UnionFind木を構築したのですが、参照渡しがうまくいっていないのか関数の外の配列の値が更新されていません。

テストケースでは頂点数をn、辺の数をkとして、k組の頂点対を与えることを想定しています。

UnionFindの実行後のtree1の値は任意の連結頂点間で等しい値であり、任意の非連結頂点間で異なる値になるはずなのですが、初期化したtree1[i] = iの値のままでした。

### 想定される実行結果
入力
5 3
0 2
1 4
2 4

出力
0 0 0 3 0

該当のソースコード

C++

1#include "bits/stdc++.h" 2using namespace std; 3 4int root(vector<int>& tree, vector<int>& child, int i){ 5 if(i == tree[i]){ 6 for(int j = 0; j < child.size(); j++){ 7 tree[child[j]] = i; 8 child.clear(); 9 return i; 10 } 11 } 12 else{ 13 child.push_back(i); 14 return i = root(tree,child,tree[i]); 15 } 16} 17 18int unit(vector<int>& tree, vector<int>& child, int x,int y){ 19 x = root(tree,child,x); 20 y = root(tree,child,y); 21 tree[max(x,y)] = min(x,y); 22} 23 24bool isUnit(vector<int>& tree, vector<int>& child, int x,int y){ 25 return root(tree,child,x) == root(tree,child,y); 26} 27 28signed main(){ 29 ios::sync_with_stdio(false); 30 cin.tie(0); 31 32 int n,k; 33 cin >> n >> k; 34 35 vector<int> tree1(n),child1; 36 for(int i = 0, i < n; i++) tree1[i] = i; // 初期化 37 38 for(int i = 0; i < k; i++){ 39 int x,y; 40 cin >> x >> y; 41 unit(tree1,child1,x,y); 42 } 43 44 for(int i = 0; i < n; i++) cout << tree[i] << " "; 45 46}

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

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

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

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

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

tiitoi

2018/11/02 09:55 編集

マクロが多くて読みづらいです。C++ でマクロは極力使わないほうがよいです。
cinnamoroll

2018/11/02 09:58

競技プログラミングで使用しているコードなのでマクロを使用する前提となっています。実務で使う場合には可読性を重視するつもりです。
cateye

2018/11/02 10:50 編集

>#define int long long・・・intは予約語ですが?まぁ、マクロだからいいっちゃいいんですが・・・
cinnamoroll

2018/11/02 11:03

競技プログラミングでは条件によって32bitに収まらない値を扱うので、オーバーフローの対策としてintをlong longにしています。その関係でint main()がlong long main()となってしまいエラーが発生するので、signed main()に置き換えてあります。
guest

回答2

0

ベストアンサー

これコンパイラ通りますか?
root関数で値を返せないパターンがあります。
で、そのパターンに最初に当たります。

C++

1int root(vector<int>& tree, vector<int>& child, int i){ 2 if(i == tree[i]){              ←ここが最初成り立つが、childのサイズは0なので 3 for(int j = 0; j < child.size(); j++){ 4 tree[child[j]] = i; 5 child.clear(); 6 return i; 7 } 8 } 9 else{ 10 child.push_back(i); 11 return i = root(tree,child,tree[i]); 12 } 13}

if(i == tree[i]){

ここが最初成り立つが、childのサイズは0なのでループせずに終わる。
しかし、returnがない。

投稿2018/11/02 10:29

ardin

総合スコア544

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

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

cinnamoroll

2018/11/02 10:59

もともと想定していたコードではfor文に{}をつけずにtree[child[j]] = i;のみをループするつもりだったところで下の2文もループに含んでいたことが原因だったようです。 c++では返り値を用意しなくてもeaxレジスタの値を自動で返すためコンパイルを通ってしまい発見できなかったようです。 ありがとうございました。
guest

0

スース読んでないですが・・・
下記、warningの説明が出来ますか? 出来ない場合はwarningが起きないソースにしましょう。

cpp

1usr~/test/cpp % g++ t.cpp 2t.cpp: In function ‘long long int root(std::vector<long long int>&, std::vector<long long int>&, long long int): 3t.cpp:10:35: warning: comparison between signed and unsigned integer expressions [-Wsign-compare] 4 #define FOR(i,a,b) for(int i=(a);i<(b);i++) 5 ^ 6t.cpp:12:19: note: in expansion of macro ‘FOR’ 7 #define REP(i,n) FOR(i,0,n) 8 ^ 9t.cpp:23:5: note: in expansion of macro ‘REP’ 10 REP(j,child.size()){ 11 ^ 12t.cpp: In function ‘long long int unit(std::vector<long long int>&, std::vector<long long int>&, long long int, long long int): 13t.cpp:39:1: warning: no return statement in function returning non-void [-Wreturn-type] 14 } 15 ^ 16t.cpp: In function ‘long long int root(std::vector<long long int>&, std::vector<long long int>&, long long int): 17t.cpp:33:1: warning: control reaches end of non-void function [-Wreturn-type] 18 } 19 ^ 20usr~/test/cpp % ./a.out 215 3 220 2 23Segmentation fault (core dumped) 24usr~/test/cpp % ./a.out 25

投稿2018/11/02 09:59

編集2018/11/02 10:01
cateye

総合スコア6851

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

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

cateye

2018/11/02 10:08 編集

ちなみに、clang++ 8.0でコンパイルすると‘48 warnings generated.’と言われます^^;
cinnamoroll

2018/11/02 10:12

intとunsigned intを比較していますが、マクロの性質上intが負になることがないので問題ありません。rootは戻り値をvoidにしていないことが原因ですが本質的な問題点では無いと思われます。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問