4
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

ややこしい条件分岐-良いコード書くきっかけを与えたい。1[C#リファクタリングサンプル]

Last updated at Posted at 2017-02-05

#ややこしい条件分岐
前記事
に記載した具体例は以下のとおり。(ソースコードはC#)
仕様)
掲示板のコメント管理を考える。
コメント一覧に、投稿されたコメントが表示され、1コメント毎に編集、削除、承認、差し戻し、詳細表示、各ボタンを以下の条件で制御する。

  1. ログイン者が投稿者であればコメントの編集および削除、詳細表示が可能。承認後は編集、削除は不可。
  2. 第一管理者は投稿後のコメントを承認(1次承認)、あるいは差し戻し、詳細表示が可能。承認後は詳細表示のみ可能。
  3. 第二管理者は第一管理者が承認したコメントのみ承認(2次承認)、差し戻し、詳細表示が可能。承認後は詳細表示のみ可能。

ソースコード)※条件分岐だけ見たいので、ボタン活性制御は省略した。


        private const string AuthorId = "投稿者ID";
        private const string FirstManagerId = "第一管理者ID";
        private const string SecondManagerId = "第二管理者ID";

        /// <summary>
        /// ボタンの活性制御を行う。
        /// </summary>
        /// <param name="loginUserId">ログイン者ID</param>
        /// <param name="firstApproverId">1次承認者ID(未承認ならnull)</param>
        /// <param name="secondApproverId">2次承認者ID(未承認ならnull)</param>
        private void SetButtonControls(string loginUserId, string firstApproverId, string secondApproverId)
        {
            // 各ボタンを非活性で初期化済み。

            if (loginUserId == AuthorId && firstApproverId == null && secondApproverId == null)
            {
                // 編集ボタン、削除ボタン、詳細表示ボタン、活性
            }
            else if (loginUserId == FirstManagerId && firstApproverId == null)
            {
                // 承認ボタン、差し戻しボタン、詳細表示ボタン活性
            }
            else if (loginUserId == FirstManagerId && firstApproverId != null)
            {
                // 詳細表示ボタン活性
            }
            else if (loginUserId == SecondManagerId && firstApproverId != null && secondApproverId == null)
            {
                // 承認ボタン、差し戻しボタン、詳細表示ボタン活性
            }
            else if (loginUserId == SecondManagerId && firstApproverId != null && secondApproverId != null)
            {
                // 詳細表示ボタン活性
            }
        }
java
    private static final String AUTHOR_ID = "投稿者ID";
    private static final String FIRST_MANAGER_ID = "第一管理者ID";
    private static final String SECOND_MANAGER_ID = "第二管理者ID";

    protected void setButtonControls(String loginUserId, String firstApproverId, String secondApproverId) {
        // 各ボタンを非活性で初期化済み。

        if (AUTHOR_ID.equals(loginUserId) && firstApproverId == null && secondApproverId == null) {
            // 編集ボタン、削除ボタン、詳細表示ボタン、活性
        } else if (FIRST_MANAGER_ID.equals(loginUserId) && firstApproverId == null) {
            // 承認ボタン、差し戻しボタン、詳細表示ボタン活性
        } else if (FIRST_MANAGER_ID.equals(loginUserId) && firstApproverId != null) {
            // 詳細表示ボタン活性
        } else if (SECOND_MANAGER_ID.equals(loginUserId) && firstApproverId != null && secondApproverId == null) {
            // 承認ボタン、差し戻しボタン、詳細表示ボタン活性
        } else if (SECOND_MANAGER_ID.equals(loginUserId) && firstApproverId == null && secondApproverId != null) {
            // 詳細表示ボタン活性
        }

    }

よく言えば仕様のとおりだけど、全てelse ifにしているせいで、読みずらい。また、同じ条件文の否定文をif文内に書いているため、冗長になっている。
同列のif文同士を並べたほうが良い。


        private void SetButtonControls(string loginUserId, string firstApproverId, string secondApproverId)
        {
            // 各ボタンを非活性で初期化済み。

            // この書き方を知らない人が私の周りでは多かった。
            bool firstApproved = firstApproverId != null;
            bool secondApproved = secondApproverId != null;

            if (loginUserId == AuthorId)
            {
                if (!firstApproved && !secondApproved)
                {
                    // 編集ボタン、削除ボタン、活性
                }

                // 詳細表示ボタン 活性
            }
            else if (loginUserId == FirstManagerId)
            {

                if (!firstApproved)
                {
                    // 承認ボタン、差し戻しボタン、活性
                }

                // 承認詳細ボタン活性
            }
            else if (loginUserId == SecondManagerId)
            {
                if (firstApproved && !secondApproved)
                {
                    // 承認ボタン、差し戻しボタン
                }

                // 詳細表示ボタン活性
            }
        }
java
    private static final String AUTHOR_ID = "投稿者ID";
    private static final String FIRST_MANAGER_ID = "第一管理者ID";
    private static final String SECOND_MANAGER_ID = "第二管理者ID";

    protected void setButtonControls(String loginUserId, String firstApproverId, String secondApproverId) {
        // 各ボタンを非活性で初期化済み。

        boolean firstApproved = firstApproverId != null;
        boolean secondApproved = secondApproverId != null;

        if (AUTHOR_ID.equals(loginUserId)) {

            if (!firstApproved && !secondApproved) {
                // 編集ボタン、削除ボタン、活性
            }

            // 詳細表示ボタン 活性

        } else if (FIRST_MANAGER_ID.equals(loginUserId)) {

            if (!firstApproved) {
                // 承認ボタン、差し戻しボタン、活性
            }

            // 承認詳細ボタン活性

        } else if (SECOND_MANAGER_ID.equals(loginUserId)) {

            if (firstApproved && !secondApproved) {
                // 承認ボタン、差し戻しボタン
            }

            // 詳細表示ボタン活性
        }
    }

ソースコードの書き方に関する問題は、基本的に「こんな書き方思いつかない」という個々のスキル差で発生している。
ソースコードの見直しの意味もこめて、サンプルコードを定期的に書いていきたい。

##リファクタリングコード2
コメント欄で参考になるコードをいただいたので、載せておく。尚、Bool型の扱いについて別の記事と統一がとれるように変数名を一部修正している。
if文が無く、ボタンの活性制御が分かりやすい。

        private void SetButtonControls(string loginUserId, string firstApproverId, string secondApproverId)
        {
            // 第一承認済み
            bool firstApproved = firstApproverId != null;

            // 第二承認済み
            bool secondApproved = firstApproved && secondApproverId != null;

            // 文字列比較はEqualsでも良いが、==でも文字列比較が可能。
            bool isAuthor = AuthorId == loginUserId;
            bool isFirstManager = FirstManagerId == loginUserId;
            bool isSecoundManager = SecondManagerId == loginUserId;

            // ボタン活性制御

            // 編集ボタン
            bool editButtonEnabled = isAuthor && !firstApproved;

            // 削除ボタン
            bool deleteButtonEnabled = isAuthor && !firstApproved;

            // 承認ボタン
            bool approvalButtonEnabled = (isFirstManager && !firstApproved) || (isSecoundManager && !secondApproved);

            // 詳細ボタン
            bool detailButtonEnabled = isAuthor || isFirstManager || isSecoundManager;
        }

java
    private static final String AUTHOR_ID = "投稿者ID";
    private static final String FIRST_MANAGER_ID = "第一管理者ID";
    private static final String SECOND_MANAGER_ID = "第二管理者ID";

    protected void setButtonControls(String loginUserId, String firstApproverId,
            String secondApproverId) {

        // 第一承認済み
        boolean firstApproved = firstApproverId != null;

        // 第二承認済み
        boolean secondApproved = firstApproved && secondApproverId != null;

        boolean isAuthor = AUTHOR_ID.equals(loginUserId);
        boolean isFirstManager = FIRST_MANAGER_ID.equals(loginUserId);
        boolean isSecoundManager = SECOND_MANAGER_ID.equals(loginUserId);

        // ボタン活性制御

        // 編集ボタン
        boolean editButtonEnabled = isAuthor && !firstApproved;

        // 削除ボタン
        boolean deleteButtonEnabled = isAuthor && !firstApproved;

        // 承認ボタン
        boolean approvalButtonEnabled = (isFirstManager && !firstApproved) || (isSecoundManager && !secondApproved);

        // 詳細ボタン
        boolean detailButtonEnabled = isAuthor || isFirstManager || isSecoundManager;
    }

次の記事(ちょっと気になるコード集)

目次

4
3
4

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
4
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?