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

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

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

Javaは、1995年にサン・マイクロシステムズが開発したプログラミング言語です。表記法はC言語に似ていますが、既存のプログラミング言語の短所を踏まえていちから設計されており、最初からオブジェクト指向性を備えてデザインされています。セキュリティ面が強力であることや、ネットワーク環境での利用に向いていることが特徴です。Javaで作られたソフトウェアは基本的にいかなるプラットフォームでも作動します。

Q&A

解決済

3回答

8012閲覧

Listから値を取得するときに順番がばらばらになる。

flumchaso

総合スコア15

Java

Javaは、1995年にサン・マイクロシステムズが開発したプログラミング言語です。表記法はC言語に似ていますが、既存のプログラミング言語の短所を踏まえていちから設計されており、最初からオブジェクト指向性を備えてデザインされています。セキュリティ面が強力であることや、ネットワーク環境での利用に向いていることが特徴です。Javaで作られたソフトウェアは基本的にいかなるプラットフォームでも作動します。

0グッド

0クリップ

投稿2018/06/29 01:22

前提・実現したいこと

DBからデータを取得したときに設定した順番で値を取得したい。

該当のソースコード

SQL

1SELECT areaCd, areaNm, areaENm 2FROM areaMaster 3ORDER BY areaCd

と、このSQLで日本の各地方をList化して取得します。
DBには以下の通りに入っています。設定時はareaCdの小さい順に入っています。デバッグで確認済み。

areaCdareaNmareaENm
01北海道hokkaido
02東北tohoku
・・・
08九州kyushu

TestMasterInterceptor.java

Java

1LinkedList<areaMaster> areaMasterList = getAreaMaster(); 2areaMasterListは実際使用時にアノテーションAutowiredを使用し呼び出しています。

TestInterceptor.java

Java

1@Autowired TestMasterInterceptor master; 2・・・ 3for (areaMaster am : master.getAreaMasterList()) { //ここで取得したときに設定したときの順番と違う。 4 //処理実施 5}

補足情報(FW/ツールのバージョンなど)

java8,spring

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

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

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

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

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

m.ts10806

2018/06/29 01:30

実行結果、どのようになっているのでしょうか。 データ取得処理部分と出力結果もご提示ください。あとできればそれぞれの型も含めたテーブル定義も。
mather

2018/06/29 02:12

DBからデータを取得してリストに変換する部分が気になります。質問には ORDER BY の付いたクエリを表示していますが、実際そのとおりにクエリが発行されているんですか?DBに格納されている順番というのは保証できない(追記型など)のでクエリをきちんと確認すべきです。
Wind

2018/06/29 03:25

「設定時はareaCdの小さい順に入っています。デバッグで確認済み。」この確認した画面をキャプチャして貰えないでしょうか?
guest

回答3

0

コードを見る限りでは、特に問題なさそうですので、あとは推測になります。

DBへのアクセスは何を使用しているのでしょうか? JDBCそのまま?なんらかのO/R Mapper?
O/R Mapperを使用しているならば、そこで何かが起きている可能性があります。
そのO/R Mapperのドキュメントを確認してみると良いと思います。

また、もう一点気になったこととして、areaCdが最大二桁の数字の0詰め文字列として扱われているので、100件を超えたときに文字列によるソートになってしまい

areaCdareaNmareaENm
100沖ノ鳥島okinotorishima
11東京都toukyoto

のようなことになっていたりしないでしょうか?この場合なら、areaCdをintにする、または、0詰めを十分な桁数にすることで解決すると思います。

余談ですが、テーブル名(とクラス名)がlowerCamelになっているのが気になります。社内の規約なら仕方ないと思いますが治せるならぜひクラス名だけでもUpperCamelにっ

投稿2018/06/29 03:41

takezoux2

総合スコア3

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

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

0

自己解決

すみません。
他の方が作成したソースで順番を変えたところがあり、Autowiredしているのですが
そのまま参照して値を変えていたため元のListの順番が変わってしまっていたようです。

順番を入れ替えたところで新しいListに詰め直すようにして元のListには影響がないように対応しました。

投稿2018/06/29 05:19

flumchaso

総合スコア15

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

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

0

私は趣味でC++やっています。
なので妄想入りですが。


たぶん仕様です。最初、「適当にセットしたからじゃね?」と思ったのですが、

DBには以下の通りに入っています。設定時はareaCdの小さい順に入っています。デバッグで確認済み。

とあることから、除外。

ほかの可能性を考えて見ました。

でも思いつかなかったのでコードを見ると...

~~リスト構造は LinkedList で生成していますね。これが原因なのでは?と思いました。
~~
で、気になったので「LinkedList ArrayList 違い」及び「LinkedList 順番が狂う」と検索すると、

根拠1: LinkedList

がヒットしました。

ArrayListは「次のオブジェクトへのポインタ」を持っているような状態なのに対し、LinkedListは「前と後~~~~ろのオブジェクトへのポインタ」を持っている状態です。 ( C言語風にいうと。 )

で、

~~> LinkedListは自分で順序を持ってないためインデックスを指定してダイレクトにオブジェクトを読み出す場合は、インデックスが全体の前半にある場合は先頭から、後半にある場合は後から順番を数えて読み出します。
( from 根拠1のページ )
~~
~~とあります。 ~~

~~ということは、C++のiteratorに相当するもの ( Javaにも iterator自体はあるようですが使ったことない... ) (根拠2: [Java] Iteratorのメモ)
が後ろにある状態でそれを元に戻していないから後ろから読み込まれているため、ソート済みでない感じになっていると思います。
~~
どうしてもLinkedListを使いたいなら、**iteratorに相当するやつで先頭に持ってきて...**とするか、
**ArrayListの方を使うか...**だと思います。
( iteratorを使わなくても先頭にiteratorに相当するやつを前に持ってくることができればそれでいいし。 )

[追記1]
どうやら、takezoux2さんのご指摘によると

ArrayListはリスト構造ではなく、内部に配列を持った実装です。

また、LinkedListも追加された要素の順番が変化することはありません。
なので、違うみたいです。取り消し線で取り下げますね。

たぶん、私が最初に挙げた内容が違うなら、「そもそもの前提であるデータ設定がなんらかの理由(ライブラリがバグを含んでいる等)によっての操作ミス」か「(〃)によってのソートミス」が考えられますね。
それかDBから引っ張ってくるときのデータとかが壊れているか、使い方が間違っているか。

投稿2018/06/29 01:50

編集2018/06/29 04:42
BeatStar

総合スコア4958

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

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

takezoux2

2018/06/29 03:46

ArrayListはリスト構造ではなく、内部に配列を持った実装です。 また、LinkedListも追加された要素の順番が変化することはありません。説明に書かれていることは、get(int index)によるindexアクセスの際の最適化の話なので、iteratorには関係しません。 なので、基本的にLinkedListをArrayListに変更しても治る可能性は低いかと思います。(使っているDB周りのライブラリがバグや仕様でLinkedListの時に特殊処理をしている場合は治るかもしれませんが)
BeatStar

2018/06/29 04:32 編集

takezoux2さん、ありがとうございます。 マジですか!? だったらArrayListって名前不適切だと思いますね。Vectorみたいにすべきだと思います。 ( まぁ、Javaの開発者辺りの趣味だと思いますし、ここで言ってもどーしよーもないですが。 ) じゃあ、設定する前の問題かな?
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問