Skip to content

Latest commit

 

History

History
670 lines (452 loc) · 22.4 KB

project_and_code_guidelines_ko.md

File metadata and controls

670 lines (452 loc) · 22.4 KB

1. 프로젝트 가이드라인

1.1 프로젝트 구조

새 프로젝트는 Android Gradle plugin user guide 에 정의된 Android Gradle project 구조를 따라야 한다. ribot Boilerplate 가 좋은 레퍼런스.

1.2 파일명명 규칙

1.2.1 클래스 파일

클래스 명은 UpperCamelCase 를 따른다.

Android component 들은 파일명 끝에 component 이름을 명시해야한다. 예) SignInActivity, SignInFragment, ImageUploaderService, ChangePasswordDialog

1.2.2 리소스 파일

리소스파일은 소문자와 언더바로 구성 ( lowercase_underscore )

1.2.2.1 Drawable 파일

drawable 명명법

Asset Type Prefix Example
Action bar ab_ ab_stacked.9.png
Button btn_ btn_send_pressed.9.png
Dialog dialog_ dialog_top.9.png
Divider divider_ divider_horizontal.9.png
Icon ic_ ic_star.png
Menu menu_ menu_submenu_bg.9.png
Notification notification_ notification_bg.9.png
Tabs tab_ tab_pressed.9.png

아이콘 명명법 (taken from Android iconography guidelines):

Asset Type Prefix Example
Icons ic_ ic_star.png
Launcher icons ic_launcher ic_launcher_calendar.png
Menu icons and Action Bar icons ic_menu ic_menu_archive.png
Status bar icons ic_stat_notify ic_stat_notify_msg.png
Tab icons ic_tab ic_tab_recent.png
Dialog icons ic_dialog ic_dialog_info.png

selector 명명법

State Suffix Example
Normal _normal btn_order_normal.9.png
Pressed _pressed btn_order_pressed.9.png
Focused _focused btn_order_focused.9.png
Disabled _disabled btn_order_disabled.9.png
Selected _selected btn_order_selected.9.png

1.2.2.2 Layout files

레이아웃 파일은 안드로이드 컴포넌트 이름이 앞으로 오게 만든다. 예를들어 SignInActivity 를 만든다 치면 레이아웃 파일 이름은 activity_sign_in.xml.

Component Class Name Layout Name
Activity UserProfileActivity activity_user_profile.xml
Fragment SignUpFragment fragment_sign_up.xml
Dialog ChangePasswordDialog dialog_change_password.xml
AdapterView item --- item_person.xml
Partial layout --- partial_stats_bar.xml

약간 다른 경우로 'ListView' 를 만들 때 'Adapter' 로 다뤄지는 layout 파일은 'item_' 으로 시작하게 된다.

이러한 규칙은 모든 레이아웃에 적용될 수는 없을 것이다. 예를 들어 다른 레이아웃의 일부분을 파일로 만들었을 때는 'partial_' 이라는 이름을 붙여주는 식으로 명명할 수 있다.

1.2.2.3 메뉴 파일

레이아웃파일과 비슷하다. UserActivity 에서 쓰이는 메뉴파일이라면 activity_user.xml 라고 만든다. 이미 'menu' 디렉토리에 존재하기 때문에 굳이 'menu' 라는 이름을 붙이지 않아도 된다.

1.2.2.4 Values files

다음과 같이 복수형으로 파일이름을 만든다. -> strings.xml, styles.xml, colors.xml, dimens.xml, attrs.xml

2 Code guidelines

2.1 Java 언어규칙

2.1.1 예외처리 무시하지 않기

밑에 처럼 예외처리 그냥 두지마라

void setServerPort(String value) {
    try {
        serverPort = Integer.parseInt(value);
    } catch (NumberFormatException e) { }
}

당신의 코드가 에러처리를 안해도 된다든가 처리하는데 중요하지 않다고 생각할 수 있지만, 위와 같은 예외를 무시하면 언젠가 다른 사람이(혹은 당신이) 곤란을 겪게 될 것이다. 모든 예외를 원칙적으로 다뤄야하고 구체적인 방법은 사례에 따라 달라진다. - (Android code style guidelines)

수정된 코드는 여기

2.1.2 generic exception 처리하지 마라

try {
    someComplicatedIOFunction();        // may throw IOException
    someComplicatedParsingFunction();   // may throw ParsingException
    someComplicatedSecurityFunction();  // may throw SecurityException
    // phew, made it all the way
} catch (Exception e) {                 // I'll just catch all exceptions
    handleError();                      // with one generic handler!
}

이유와 수정된 코드는 여기참고

2.1.3 finalizers 사용하지 마라

우리는 finalizer를 사용하지 않는다. finalizer 는 언제 불릴지 혹은 불려지기는 할지에 대한 보장이 없다. 대부분의 경우 예외처리만 잘하면 필요없다. 정말 필요하다면 'close()' 메서드같은 것을 정의하고 메서드가 정확히 언제 불려지는지 문서화 하라. InputStream을 예로 들면 finalizer 에서 짧은 로그를 찍는건 나쁘진 않지만 로그가 넘쳐나지 않는 이상 꼭 필요한 것은 아니다.- (Android code style guidelines)

2.1.4 import 정확하게 하기

This is bad: import foo.*;

This is good: import foo.Bar;

여기참고

2.2 Java 스타일 규칙

2.2.1 Fields definition and naming

파일 상단 에 정의 되어야 하는 것들과 명명법

  • Private, non-static 은 m 으로 시작.
  • Private, static 은 s 로 시작.
  • 다른 것들은 소문자로 시작.
  • Static final (상수) 언더바와 대문자(ALL_CAPS_WITH_UNDERSCORES).

Example:

public class MyClass {
    public static final int SOME_CONSTANT = 42;
    public int publicField;
    private static MyClass sSingleton;
    int mPackagePrivate;
    private int mPrivate;
    protected int mProtected;
}

2.2.3 단어와 약어 다루기

Good Bad
클래스 XmlHttpRequest XMLHTTPRequest
메서드 getCustomerId getCustomerID
변수 String url String URL
long id long ID

2.2.4 공백과 들여쓰기 사용

블럭에서는 4 칸 들여쓰기:

if (x == 1) {
    x++;
}

한 라인일 경우 8 칸 들여쓰기:

Instrument i =
        someLongExpression(that, wouldNotFit, on, one, line);

2.2.5 standard brace style

standard brace style 은 다음과 같은것

class MyClass {
    int func() {
        if (something) {
            // ...
        } else if (somethingElse) {
            // ...
        } else {
            // ...
        }
    }
}

조건과 실행을 한라인에 할 수 있으면 다음과 같이 한다.e.g.

if (condition) body();

This is bad:

if (condition)
    body();  // bad!

2.2.6 Annotations

2.2.6.1 Annotations 활용

안드로이드 스타일 가이드에 따르면, 표준 활용은 Java 에 미리 정의되어 있다.

  • @Override: @Override annotation 은 부모클래스에 있는 메서드를 구현하거나 정의할 때 반드시 사용되어야 하는 annotation 이다. 예를 들어 클래스에서 @inheritdocs Javadoc tag 를 사용했다면 (인터페이스말고), 부모클래스의 메서드에 반드시 @Overrides 를 사용해야한다.

  • @SuppressWarnings: The @SuppressWarnings annotation warning 을 도저히 없앨 수 없는 상황에서 사용.

annotation 에 대한 자세한 가이드라인은 여기.

2.2.6.2 Annotations 스타일

클래스, 메서드, 생성자

annotations 이 클래스, 매서드, 생성자 등에 적용될 때 , documatation 블럭 뒤에 오는 것들은 한 줄씩 표기해야한다.

/* This is the documentation block about the class */
@AnnotationA
@AnnotationB
public class MyAnnotatedClass { }

Fields

annotation 을 멤버에게 쓸 때는 한줄에 모두 표기해야한다.

@Nullable @Mock DataManager mDataManager;

2.2.7 변수 범위 제한

지역 변수의 범위는 최소한으로 제한해야한다. 그렇게 함으로써 코드의 가독성과 유지관리성을 높이고 에러 가능성을 줄일 수 있다.

지역 변수는 사용하는 시점에 선언되어야 한다. 대부분의 지역 변수에는 초기화가 포함되어야 한다. 아직 지역 변수를 초기화 하기에 충분한 정보가 없다면 정보가 있을 때까지 선언을 미뤄야 한다. - (Android code style guidelines)

2.2.8 import 순서

안드로이드 스튜디오를 사용하고 있다면 이미 이러한 규칙을 따르고 있으니 걱정할 필요없다.

import 순서:

  1. Android imports
  2. Imports from third parties (com, junit, net, org)
  3. java and javax
  4. Same project imports

To exactly match the IDE settings, the imports should be: IDE 세팅과 맞추려면

  • 알파벳 순서로, 소문자 앞에 대문자 (e.g. a 앞에 Z).
  • 그룹마다 줄바꿈으로 구분해주기 (android, com, junit, net, org, java, javax)

더 많은 정보는 여기

2.2.9 로깅 가이드

로그 출력용 Log 클래스를 사용할 때 다음과 같이 이슈를 분리할 수 있다.

  • Log.v(String tag, String msg) (verbose)
  • Log.d(String tag, String msg) (debug)
  • Log.i(String tag, String msg) (information)
  • Log.w(String tag, String msg) (warning)
  • Log.e(String tag, String msg) (error)

일반적으로 클래스 이름을 static final 로 선언해서 다음과 같이 사용할 수 있다.

public class MyClass {
    private static final String TAG = MyClass.class.getSimpleName();

    public myMethod() {
        Log.e(TAG, "My error message");
    }
}

VERBOSE 과 디버그 로그는 릴리즈 빌드에선 disable 되어야 한다. Infomation, Warning, Error 로그도 마찬가지로 diable 하는걸 추천하지만 릴리즈 빌드의 이슈관리에 사용하고 싶으면 enable 시킬 수 있다. enable 할 때는 민감한 정보(email address, user ID 같은)를 노출시키지 않도록 한다.

디버그 빌드일 때만 사용하려면 다음과 같이

if (BuildConfig.DEBUG) Log.d(TAG, "The value of x is " + x);

2.2.10 클래스 멤버 순서

정해진 법칙은 없지만 유지보수와 가독성 차원에서의 논리적, 일관성 을 지키려면 다음과 같은 순서를 추천한다.

  1. 상수
  2. 멤버변수
  3. 생성자
  4. Override methods and callbacks (public or private)
  5. Public methods
  6. Private methods
  7. Inner classes or interfaces

Example:

public class MainActivity extends Activity {

    private static final String TAG = MainActivity.class.getSimpleName();

    private String mTitle;
    private TextView mTextViewTitle;

    @Override
    public void onCreate() {
        ...
    }

    public void setTitle(String title) {
    	mTitle = title;
    }

    private void setUpView() {
        ...
    }

    static class AnInnerClass {

    }

}

만약 클래스가 Activity나 Fragment 같이 Android component 를 상속받고 있다면 component의 lifecycle 이 좋은 예가 될 수 있다. 예를 들어 onCreate(), onDestroy(), onPause(), onResume() 등을 구현하려면 다음과 같은 순서로

public class MainActivity extends Activity {

	//Order matches Activity lifecycle
    @Override
    public void onCreate() {}

    @Override
    public void onResume() {}

    @Override
    public void onPause() {}

    @Override
    public void onDestroy() {}

}

2.2.11 메서드의 파라미터 순서

안드로이드 프로그래밍을 할 때 Context 를 받는 경우가 많다. 이 경우 Context 파라미터를 첫번째로 해야한다.

반대로 callback 을 받을 때는 마지막 파라미터 로 받아야 한다.

Examples:

// Context always goes first
public User loadUser(Context context, int userId);

// Callbacks always go last
public void loadUserAsync(Context context, int userId, UserCallback callback);

2.2.13 상수문자열 명명, 값

안드로이드에서 SharedPreferences, Bundle, Intent 등은 key-value 를 짝으로 쓰는 것들이다. 여기엔 상수 문자열이 많이 쓰인다.

이러한 component 를 쓸때 반드시 static final로 선언해서 사용해야 한다. 아래와 같은 접두사를 사용하면 좋다.

Element Field Name Prefix
SharedPreferences PREF_
Bundle BUNDLE_
Fragment Arguments ARGUMENT_
Intent Extra EXTRA_
Intent Action ACTION_

Fragment 에서 전달 값 Fragment.getArguments() 을 받아올 때 Bundle 로 받아온다. 일반적인 사용이기 따문에 다른 접두사를 붙여 정의된다.

Example:

// Note the value of the field is the same as the name to avoid duplication issues
static final String PREF_EMAIL = "PREF_EMAIL";
static final String BUNDLE_AGE = "BUNDLE_AGE";
static final String ARGUMENT_USER_ID = "ARGUMENT_USER_ID";

// Intent-related items use full package name as value
static final String EXTRA_SURNAME = "com.myapp.extras.EXTRA_SURNAME";
static final String ACTION_OPEN_USER = "com.myapp.action.ACTION_OPEN_USER";

2.2.14 Fragment 와 Activity 의 전달인자

IntentBundle 을 통해 Activity 혹은 Fragment 에 데이터를 넘길 때 키값은 위에 언급한 바와 같이 작성되어야 한다.

Activity, Fragment가 전달인자를 받을 때 IntentFragment형으로 만들어진 public static 메서드를 통해 전달되어야 한다.

Activity 의 경우 getStartIntent()에서 불릴 때:

public static Intent getStartIntent(Context context, User user) {
	Intent intent = new Intent(context, ThisActivity.class);
	intent.putParcelableExtra(EXTRA_USER, user);
	return intent;
}

Fragment 의 경우 newInstance() 로 전달인자를 사용하여 Fragment 의 생성을 처리할 때:

public static UserFragment newInstance(User user) {
	UserFragment fragment = new UserFragment();
	Bundle args = new Bundle();
	args.putParcelable(ARGUMENT_USER, user);
	fragment.setArguments(args)
	return fragment;
}

Note 1: 이런 메서드는 onCreate() 위에 있어야 한다..

Note 2: If we provide the methods described above, the keys for extras and arguments should be private because there is not need for them to be exposed outside the class.

Note 2: 위에서 언급한 바와 같이, 클래스 밖에서는 알 수 없도록 extras 와 전달인자의 key 값은 private 이어야 한다.

2.2.15 줄 길이 제한

코드는 100글자 를 넘어서는 안된다. 더 길다면 두가지 옵션이 있다.

  • 지역변수와 메서드를 풀어써라
  • 자동 줄 바꿈 사용

100글자를 넘어도 되는 두가지 예외 가 있다.

  • Lines that are not possible to split, e.g. long URLs in comments.
  • 라인을 나눌 수 없을 때, e.g. 주석의 긴 URL
  • packageimport

2.2.15.1 자동 줄바꿈 전략

정확한 공식은 없다. 하지만 상식적인 수준에서 몇가지 규칙은 있다.

연산자 앞에서 끊기

int longName = anotherVeryLongVariable + anEvenLongerOne - thisRidiculousLongOne
        + theFinalOne;

대입 연산자는 예외

int longName =
        anotherVeryLongVariable + anEvenLongerOne - thisRidiculousLongOne + theFinalOne;

메서드 체인의 경우

빌더 패턴같은 메서드 체인의 경우 . 앞에서 끊는다.

Picasso.with(context).load("http://ribot.co.uk/images/sexyjoe.jpg").into(imageView);
Picasso.with(context)
        .load("http://ribot.co.uk/images/sexyjoe.jpg")
        .into(imageView);

긴 매개변수의 경우

매개변수가 많거나 길거나 할 땐 , 뒤에서 끊는다.

loadPicture(context, "http://ribot.co.uk/images/sexyjoe.jpg", mImageViewProfilePicture, clickListener, "Title of the picture");
loadPicture(context,
        "http://ribot.co.uk/images/sexyjoe.jpg",
        mImageViewProfilePicture,
        clickListener,
        "Title of the picture");

2.2.16 RxJava chains styling

. 앞에서 끊는다.

public Observable<Location> syncLocations() {
    return mDatabaseHelper.getAllLocations()
            .concatMap(new Func1<Location, Observable<? extends Location>>() {
                @Override
                 public Observable<? extends Location> call(Location location) {
                     return mRetrofitService.getLocation(location.id);
                 }
            })
            .retry(new Func2<Integer, Throwable, Boolean>() {
                 @Override
                 public Boolean call(Integer numRetries, Throwable throwable) {
                     return throwable instanceof RetrofitError;
                 }
            });
}

2.3 XML style rules

2.3.1 Use self closing tags

When an XML element doesn't have any contents, you must use self closing tags.

This is good:

<TextView
	android:id="@+id/text_view_profile"
	android:layout_width="wrap_content"
	android:layout_height="wrap_content" />

This is bad :

<!-- Don\'t do this! -->
<TextView
    android:id="@+id/text_view_profile"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content" >
</TextView>

2.3.2 리소스 명명

ID 와 이름은 소문자_언더바 .

2.3.2.1 ID 명명

Element Prefix (역자)혹은
TextView text_ tv_
ImageView image_ iv_
Button button_ btn_
Menu menu_

Image view example:

<ImageView
    android:id="@+id/image_profile"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content" />

Menu example:

<menu>
	<item
        android:id="@+id/menu_done"
        android:title="Done" />
</menu>

2.3.2.2 문자열

문자열 이름은 문자열의 카테고리에 따라서 접두어를 붙있다.

Prefix Description
error_ An error message
msg_ A regular information message
title_ A title, i.e. a dialog title
action_ An action such as "Save" or "Create"

2.3.2.3 스타일과 테마

리소스와는 달리 스타일은 UpperCamelCase 로 명명

2.3.3 Attribute 순서

일반적인 규칙의 attribute 순서는

  1. View Id
  2. Style
  3. Layout width and layout height
  4. Other layout attributes, sorted alphabetically
  5. Remaining attributes, sorted alphabetically

2.4 테스트 스타일 규칙

2.4.1 Unit tests

테스트 클래스는 테스트할 타겟 클래스와 이름을 맞춰야한다. 접미사 Test 를 넣고. 예를 들어 DatabaseHelper 를 테스트 한다면 DatabaseHelperTest 라고 명명한다.

테스트 메서드는 @Test annotation 을 붙이고 테스트할 메서드이름과 메서드의 원래상태 혹은 예상결과를 붙여서 명명한다.

  • Template: @Test void methodNamePreconditionExpectedBehaviour()
  • Example: @Test void signInWithEmptyEmailFails()

메서드의 원래상태나 예상결과가 명확하지 않을 때는 안붙여도 된다. ( 이 때는 맘데로 하라는듯)

어떤 클래스는 많은 양의 메서드를 포함하고 있고 각 메서드마다 몇개씩의 테스트가 필요하다. 이런 경우에는 테스트 클래스를 여러개로 나누는걸 추천한다. 예를 들어 DataManager 가 많은 메서드를 가지고 있다치면 DataManagerSignInTest, DataManagerLoadUsersTest 등으로 테스트 클래스를 나눌수 있다. 일반적인 테스트가 어떻게 나뉘는지는 여기를 참고하라

2.4.2 Espresso tests

Espresso 테스트는 보통 Activity 를 타겟으로 한다. 그래서 보통 SignInActivityTest 이런 식으로 명명한다.

체인 메서드는 다음과 같이 사용한다.

onView(withId(R.id.view))
        .perform(scrollTo())
        .check(matches(isDisplayed()))

Reference

https://github.com/ribot/android-guidelines 에서 가져온 것을 번역&의역

License

Copyright 2015 Ribot Ltd.

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

    http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.