Class files
Class names are written in UpperCamelCase.For classes that extend an Android component, the name of the class should end with the name of the component; for example:
SignInActivity
, SignInFragment
, ImageUploaderService
, ChangePasswordDialog
.1.2.2 Resources files
Resources file names are written in lowercase_underscore.1.2.2.1 Drawable files
Naming conventions for drawables: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 |
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 |
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
Layout files should match the name of the Android components that they are intended for but moving the top level component name to the beginning. For example, if we are creating a layout for theSignInActivity
, the name of the layout file should be 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 |
Adapter
, e.g to populate a ListView
. In this case, the name of the layout should start with item_
.Note that there are cases where these rules will not be possible to apply. For example, when creating layout files that are intended to be part of other layouts. In this case you should use the prefix
partial_
.1.2.2.3 Menu files
Similar to layout files, menu files should match the name of the component. For example, if we are defining a menu file that is going to be used in theUserActivity
, then the name of the file should be activity_user.xml
A good practice is to not include the word
menu
as part of the name because these files are already located in the menu
directory.1.2.2.4 Values files
Resource files in the values folder should be plural, e.g.strings.xml
, styles.xml
, colors.xml
, dimens.xml
, attrs.xml
2 Code guidelines
2.1 Java language rules
2.1.1 Don't ignore exceptions
You must never do the following:void setServerPort(String value) {
try {
serverPort = Integer.parseInt(value);
} catch (NumberFormatException e) { }
}
See alternatives here.
2.1.2 Don't catch generic exception
You should not do this: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 Don't use finalizers
We don't use finalizers. There are no guarantees as to when a finalizer will be called, or even that it will be called at all. In most cases, you can do what you need from a finalizer with good exception handling. If you absolutely need it, define aclose()
method (or the like) and document exactly when that method needs to be called. See InputStream
for an example. In this case it is appropriate but not required to
print a short log message from the finalizer, as long as it is not
expected to flood the logs. - (Android code style guidelines)2.1.4 Fully qualify imports
This is bad:import foo.*;
This is good:
import foo.Bar;
See more info here
2.2 Java style rules
2.2.1 Fields definition and naming
Fields should be defined at the top of the file and they should follow the naming rules listed below.- Private, non-static field names start with m.
- Private, static field names start with s.
- Other fields start with a lower case letter.
- Static final fields (constants) are ALL_CAPS_WITH_UNDERSCORES.
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 Treat acronyms as words
Good | Bad |
---|---|
XmlHttpRequest |
XMLHTTPRequest |
getCustomerId |
getCustomerID |
String url |
String URL |
long id |
long ID |
2.2.4 Use spaces for indentation
Use 4 space indents for blocks:if (x == 1) {
x++;
}
Instrument i =
someLongExpression(that, wouldNotFit, on, one, line);
2.2.5 Use standard brace style
Braces go on the same line as the code before them.class MyClass {
int func() {
if (something) {
// ...
} else if (somethingElse) {
// ...
} else {
// ...
}
}
}
If the condition and the body fit on one line and that line is shorter than the max line length, then braces are not required, e.g.
if (condition) body();
if (condition)
body(); // bad!
2.2.6 Annotations
2.2.6.1 Annotations practices
According to the Android code style guide, the standard practices for some of the predefined annotations in Java are:-
@Override
: The @Override annotation must be used whenever a method overrides the declaration or implementation from a super-class. For example, if you use the @inheritdocs Javadoc tag, and derive from a class (not an interface), you must also annotate that the method @Overrides the parent class's method.
-
@SuppressWarnings
: The @SuppressWarnings annotation should only be used under circumstances where it is impossible to eliminate a warning. If a warning passes this "impossible to eliminate" test, the @SuppressWarnings annotation must be used, so as to ensure that all warnings reflect actual problems in the code.
2.2.6.2 Annotations style
Classes, Methods and ConstructorsWhen annotations are applied to a class, method, or constructor, they are listed after the documentation block and should appear as one annotation per line .
/* This is the documentation block about the class */
@AnnotationA
@AnnotationB
public class MyAnnotatedClass { }
Annotations applying to fields should be listed on the same line, unless the line reaches the maximum line length.
@Nullable @Mock DataManager mDataManager;
2.2.7 Limit variable scope
The scope of local variables should be kept to a minimum (Effective Java Item 29). By doing so, you increase the readability and maintainability of your code and reduce the likelihood of error. Each variable should be declared in the innermost block that encloses all uses of the variable.Local variables should be declared at the point they are first used. Nearly every local variable declaration should contain an initializer. If you don't yet have enough information to initialize a variable sensibly, you should postpone the declaration until you do. - (Android code style guidelines)
2.2.8 Order import statements
If you are using an IDE such as Android Studio, you don't have to worry about this because your IDE is already obeying these rules. If not, have a look below.The ordering of import statements is:
- Android imports
- Imports from third parties (com, junit, net, org)
- java and javax
- Same project imports
- Alphabetically ordered within each grouping, with capital letters before lower case letters (e.g. Z before a).
- There should be a blank line between each major grouping (android, com, junit, net, org, java, javax).
2.2.9 Logging guidelines
Use the logging methods provided by theLog
class to print out error messages or other information that may be useful for developers to identify issues: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
field at the top of the file. For example:public class MyClass {
private static final String TAG = MyClass.class.getSimpleName();
public myMethod() {
Log.e(TAG, "My error message");
}
}
To only show logs on debug builds:
if (BuildConfig.DEBUG) Log.d(TAG, "The value of x is " + x);
2.2.10 Class member ordering
There is no single correct solution for this but using a logical and consistent order will improve code learnability and readability. It is recommendable to use the following order:- Constants
- Fields
- Constructors
- Override methods and callbacks (public or private)
- Public methods
- Private methods
- Inner classes or interfaces
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 {
}
}
onCreate()
, onDestroy()
, onPause()
and onResume()
, then the correct order is: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 Parameter ordering in methods
When programming for Android, it is quite common to define methods that take aContext
. If you are writing a method like this, then the Context must be the first parameter.The opposite case are callback interfaces that should always be the last parameter.
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 String constants, naming, and values
Many elements of the Android SDK such asSharedPreferences
, Bundle
, or Intent
use a key-value pair approach so it's very likely that even for a small
app you end up having to write a lot of String constants.When using one of these components, you must define the keys as a
static final
fields and they should be prefixed as indicated below.Element | Field Name Prefix |
---|---|
SharedPreferences | PREF_ |
Bundle | BUNDLE_ |
Fragment Arguments | ARGUMENT_ |
Intent Extra | EXTRA_ |
Intent Action | ACTION_ |
Fragment.getArguments()
- are also a Bundle. However, because this is a quite common use of Bundles, we define a different prefix for them.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 Arguments in Fragments and Activities
When data is passed into anActivity
or Fragment
via an Intent
or a Bundle
, the keys for the different values must follow the rules described in the section above.When an
Activity
or Fragment
expects arguments, it should provide a public static
method that facilitates the creation of the relevant Intent
or Fragment
.In the case of Activities the method is usually called
getStartIntent()
:public static Intent getStartIntent(Context context, User user) {
Intent intent = new Intent(context, ThisActivity.class);
intent.putParcelableExtra(EXTRA_USER, user);
return intent;
}
newInstance()
and handles the creation of the Fragment with the right arguments:public static UserFragment newInstance(User user) {
UserFragment fragment = new UserFragment();
Bundle args = new Bundle();
args.putParcelable(ARGUMENT_USER, user);
fragment.setArguments(args)
return fragment;
}
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.2.2.15 Line length limit
Code lines should not exceed 100 characters. If the line is longer than this limit there are usually two options to reduce its length:- Extract a local variable or method (preferable).
- Apply line-wrapping to divide a single line into multiple ones.
- Lines that are not possible to split, e.g. long URLs in comments.
package
andimport
statements.
2.2.15.1 Line-wrapping strategies
There isn't an exact formula that explains how to line-wrap and quite often different solutions are valid. However there are a few rules that can be applied to common cases.Break at operators
When the line is broken at an operator, the break comes before the operator. For example:
int longName = anotherVeryLongVariable + anEvenLongerOne - thisRidiculousLongOne
+ theFinalOne;
An exception to the
break at operators
rule is the assignment operator =
, where the line break should happen after the operator.int longName =
anotherVeryLongVariable + anEvenLongerOne - thisRidiculousLongOne + theFinalOne;
When multiple methods are chained in the same line - for example when using Builders - every call to a method should go in its own line, breaking the line before the
.
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);
When a method has many parameters or its parameters are very long, we should break the line after every comma
,
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
Rx chains of operators require line-wrapping. Every operator must go in a new line and the line should be broken before the.
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" />
<!-- 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 Resources naming
Resource IDs and names are written in lowercase_underscore.2.3.2.1 ID naming
IDs should be prefixed with the name of the element in lowercase underscore. For example:Element | Prefix |
---|---|
TextView |
text_ |
ImageView |
image_ |
Button |
button_ |
Menu |
menu_ |
<ImageView
android:id="@+id/image_profile"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
<menu>
<item
android:id="@+id/menu_done"
android:title="Done" />
</menu>
2.3.2.2 Strings
String names start with a prefix that identifies the section they belong to. For exampleregistration_email_hint
or registration_name_hint
. If a string doesn't belong to any section, then you should follow the rules below: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 Styles and Themes
Unlike the rest of resources, style names are written in UpperCamelCase.2.3.3 Attributes ordering
As a general rule you should try to group similar attributes together. A good way of ordering the most common attributes is:- View Id
- Style
- Layout width and layout height
- Other layout attributes, sorted alphabetically
- Remaining attributes, sorted alphabetically
2.4 Tests style rules
2.4.1 Unit tests
Test classes should match the name of the class the tests are targeting, followed byTest
. For example, if we create a test class that contains tests for the DatabaseHelper
, we should name it DatabaseHelperTest
.Test methods are annotated with
@Test
and should
generally start with the name of the method that is being tested,
followed by a precondition and/or expected behaviour.- Template:
@Test void methodNamePreconditionExpectedBehaviour()
- Example:
@Test void signInWithEmptyEmailFails()
Sometimes a class may contain a large amount of methods, that at the same time require several tests for each method. In this case, it's recommendable to split up the test class into multiple ones. For example, if the
DataManager
contains a lot of methods we may want to divide it into DataManagerSignInTest
, DataManagerLoadUsersTest
, etc. Generally you will be able to see what tests belong together because they have common test fixtures.2.4.2 Espresso tests
Every Espresso test class usually targets an Activity, therefore the name should match the name of the targeted Activity followed byTest
, e.g. SignInActivityTest
When using the Espresso API it is a common practice to place chained methods in new lines.
onView(withId(R.id.view))
.perform(scrollTo())
.check(matches(isDisplayed()))
No comments:
Post a Comment