KEMBAR78
Android | PDF | Android (Operating System) | Google Play
0% found this document useful (0 votes)
207 views100 pages

Android

The document provides an introduction to Android, including what Android is, why developers create Android apps, different Android versions over time, challenges of Android app development, and how to learn more. Key points are that Android is an operating system for mobile devices, it has the largest user base which motivates app creation, and the Android software development kit helps developers create apps.

Uploaded by

Sudhan
Copyright
© © All Rights Reserved
We take content rights seriously. If you suspect this is your content, claim it here.
Available Formats
Download as PDF, TXT or read online on Scribd
0% found this document useful (0 votes)
207 views100 pages

Android

The document provides an introduction to Android, including what Android is, why developers create Android apps, different Android versions over time, challenges of Android app development, and how to learn more. Key points are that Android is an operating system for mobile devices, it has the largest user base which motivates app creation, and the Android software development kit helps developers create apps.

Uploaded by

Sudhan
Copyright
© © All Rights Reserved
We take content rights seriously. If you suspect this is your content, claim it here.
Available Formats
Download as PDF, TXT or read online on Scribd
You are on page 1/ 100

1.

0: Introduction to Android

Contents:

 What is Android?
 Why develop apps for Android?
 Android versions
 The challenges of Android app development
 Learn more

What is Android?

Android is an operating system and programming platform developed by Google for mobile
phones and other mobile devices, such as tablets. It can run on many different devices from
many different manufacturers. Android includes a software development kit (SDK) that
helps you write original code and assemble software modules to create apps for Android
users. Android also provides a marketplace to distribute apps. All together, Android
represents an ecosystem for mobile apps.

Why develop apps for Android?

Developers create apps for a variety of reasons. They may need to address business
requirements or build new services or businesses, or they may want to offer games and other
types of content for users. Developers choose to develop for Android in order to reach the
majority of mobile device users.

Most popular platform for mobile apps

As the world's most popular mobile platform, Android powers hundreds of millions of
mobile devices in more than 190 countries around the world. It has the largest installed base
of any mobile platform and is still growing fast. Every day another million users power up
their Android-powered devices for the first time and start looking for apps, games, and other
digital content.
Best experience for app users

Android provides a touchscreen user interface (UI) for interacting with apps. Android's UI
is mainly based on direct manipulation. People use touch gestures such as swiping, tapping,
and pinching to manipulate on-screen objects. In addition to the keyboard, there's a
customizable on-screen keyboard for text input. Android can also support game controllers
and full-size physical keyboards connected by Bluetooth or USB.

The Android home screen can contain several panes of app icons, which launch their
associated apps. Home screen panes can also contain app widgets, which display live, auto-
updating content such as the weather, the user's email inbox, or a news ticker. Android can
also play multimedia content such as music, animation, and video. The figure above shows
app icons on the home screen (left), playing music (center), and displaying app widgets
(right). Along the top of the screen is a status bar, showing information about the device and
its connectivity. The Android home screen may be made up of several panes, and the user
swipes back and forth between the panes.

Android is designed to provide immediate response to user input. Besides a dynamic


interface that responds immediately to touch, an Android-powered device can vibrate to
provide haptic feedback. Many apps take advantage of internal hardware such as
accelerometers, gyroscopes, and proximity sensors to respond to additional user actions.
These sensors can also detect screen rotation. For example, you could design a racing game
where the user rotates the device as if it were a steering wheel.
The Android platform, based on the Linux kernel, is designed primarily for touchscreen
mobile devices such as mobile phones and tablets. Because Android-powered devices are
usually battery-powered, Android is designed to manage processes to keep power
consumption at a minimum, providing longer battery use.

It's easy to develop apps

To develop apps that take advantage of the Android operating system and UI, use the
Android software development kit (SDK). The SDK includes software libraries of prewritten
code, a debugger, a device emulator, documentation, sample code, and tutorials. Use the
SDK to create apps that look great and take advantage of the hardware capabilities available
on each Android-powered device.

To develop apps using the SDK, you use the Java programming language to develop the app
and Extensible Markup Language (XML) files to describe data resources. By writing the
code in Java and creating a single app binary, you create an app that can run on both phone
and tablet form factors. You can declare your UI in lightweight sets of XML resources. For
example, create one set for parts of the UI that are common to all form factors, and other
sets for features specific to phones or tablets. At runtime, Android applies the correct
resource sets based on the device's screen size, screen density, locale, and so on.
To help you develop your apps efficiently, Google offers an integrated development
environment (IDE) called Android Studio. It offers advanced features for developing,
debugging, and packaging Android apps. Using Android Studio, you can develop for any
Android-powered device, or create virtual devices that emulate any hardware configuration.

Android provides a rich development architecture. You don't need to know much about the
components of this architecture, but it is useful to know what is available in the system for
your app to use. The following diagram shows the major components of the Android stack—
the operating system and development architecture.

In the figure above:


1. Apps: Your apps live at this level, along with core system apps for email, SMS messaging,
calendars, internet browsing, and contacts.
2. Java API framework: All features for Android development, such as UI
components, resource management, and lifecycle management, are available through
application programming interfaces (APIs). You don't need to know the details of how the
APIs work. You only need to learn how to use them.
3. Libraries and Android runtime: Each app runs in its own process, with its own instance of
the Android runtime. Android includes a set of core runtime libraries that provide most of
the functionality of the Java programming language. Many core Android system components
and services are built from native code that require native libraries written in C and C++.
These native libraries are available to apps through the Java API framework.
4. Hardware abstraction layer (HAL): This layer provides standard interfaces that expose
device hardware capabilities to the higher-level Java API framework. The HAL consists of
multiple library modules, each of which implements an interface for a specific type of
hardware component, such as the camera or Bluetooth module.
5. Linux kernel: The foundation of the Android platform is the Linux kernel. The layers above
the Linux kernel rely on the Linux kernel for threading, low-level memory management,
and other underlying functionality. Using a Linux kernel enables Android to take advantage
of Linux-based security features and allows device manufacturers to develop hardware
drivers for a well-known kernel.

Many distribution options

You can distribute your Android app in many different ways: email, website, or an app
marketplace such as Google Play. Android users download billions of apps and games from
the Google Play store each month. Google Play is a digital distribution service, operated and
developed by Google, that serves as the official app store for Android. Google Play lets
consumers to browse and download apps developed with the Android SDK.

Android versions
Google provides major incremental upgrades to the Android operating system using
confectionery-themed names. The latest major release is Android 8.0 "Oreo".
Code name Version Initial release date API level
number

N/A 1.0 23 September 2008 1

1.1 9 February 2009 2

Cupcake 1.5 27 April 2009 3

1.6 15 September 2009 4

Donut

Eclair 2.0 – 2.1 26 October 2009 5–7

Froyo 2.2 – 2.2.3 20 May 2010 8

Gingerbread 2.3 – 2.3.7 6 December 2010 9–10

Honeycomb 3.0 – 3.2.6 22 February 2011 11–13

Ice Cream Sandwich 4.0 – 4.0.4 18 October 2011 14–15


Jelly Bean 4.1 – 4.3.1 9 July 2012 16–18

KitKat 4.4 – 4.4.4 31 October 2013 19–20

Lollipop 5.0 – 5.1.1 12 November 2014 21–22

Marshmallow 6.0 – 6.0.1 5 October 2015 23

Nougat 7.0 22 August 2016 24

Oreo 8.0 August 21, 2017 26

Tip: See previous versions and their features at The Android Story. The dashboard for
platform versions shows the distribution of active devices running each version of Android,
based on the number of devices that visit the Google Play store. It's a good practice to
support about 90% of the active devices, while targeting your app to the latest version.

Note: To provide the best features and functionality across Android versions, use
the Android Support Library in your app. This library allows your app to use recent Android
platform APIs on older devices.

The challenges of Android app development


While the Android platform provides rich functionality for app development, there are still
a number of challenges you need to address, such as:
 Building for a multiscreen world
 Getting performance right
 Keeping your code and your users more secure
 Making sure your app is compatible with older platform versions
 Understanding the market and the user

Building for a multi-screen world


Android runs on billions of handheld devices around the world and supports various form
factors including wearable devices and televisions. Devices come in different sizes and
shapes, which affects how you design the screens and UI elements in your apps.

In addition, device manufacturers may add their own UI elements, styles, and colors to
differentiate their products. Each manufacturer offers different features with respect to
keyboard forms, screen size, or camera buttons. An app running on one device may look a
bit different on another. Your challenge, as a developer, is to design UI elements that work
on all devices.

Maximizing app performance


An app's performance is determined by how fast it runs, how easily it connects to the
network, and how well it manages battery and memory usage. Performance is affected by
factors such as battery life, multimedia content, and internet access. Be aware that some
features you design for your app may cause performance problems for users. For example,
to save the user's battery power, enable background services only when they are necessary.

Keeping your code and your users more secure

You need to take precautions to make your code, and the user's experience when they use
your app, as secure as possible.

 Use tools such as ProGuard, which is provided in Android Studio. ProGuard detects and
removes unused classes, fields, methods, and attributes.
 Encrypt all of your app's code and resources while packaging the app.
 To protect critical user information such as logins and passwords, secure your
communication channel to protect data in transit across the internet, as well as data at rest
on the device.
Remaining compatible with older versions of Android
The Android platform continues to improve and provide new features you can add to your
apps. However, you should ensure that your app can still run on devices with older versions
of Android. It is impractical to focus only on the most recent Android version, as not all
users may have upgraded or may be able to upgrade their devices. Fortunately Android
Studio provides options for developers to more easily remain compatible with older
versions.

1.1: Your first Android app

Contents:

 The development process


 Using Android Studio
 Exploring a project
 Understanding the Android manifest
 Understanding the build process
 Running the app on an emulator or a device
 Using the log
 Related practical
 Learn more

This chapter describes how to develop applications using Android Studio, which is an
integrated development environment (IDE) for Android.

The development process

An Android app project begins with an idea and a definition of the requirements necessary
to realize that idea. You may want to sketch user interfaces (UIs) for the various app
functions. To show what a UI would look like and how it would work, use drawings,
mockups, and prototypes.

When you are ready to start coding, you use Android Studio to go through the following
steps:

1. Create the project in Android Studio and choose an appropriate template.


2. Define a layout for each screen that has UI elements. You can place UI elements on the
screen using the layout editor, or you can write code directly in the Extensible Markup
Language (XML).
3. Write code using the Java programming language. Create source code for all of the app's
components.
4. Build and run the app on real and virtual devices. Use the default build configuration or
create custom builds for different versions of your app.©
5. Test and debug the app's logic and UI.
6. Publish the app by assembling the final APK (package file) and distributing it through
channels such as Google Play.

Using Android Studio

Android Studio provides a unified development environment for creating apps for all
Android-powered devices. Android Studio includes code templates with sample code for
common app features, extensive testing tools and frameworks, and a flexible build system.

Starting an Android Studio project

After you have successfully installed the Android Studio IDE, double-click the Android
Studio application icon to start it. Click Start a new Android Studio project in the
Welcome window, and name the project the same name that you want to use for the app.

When choosing a unique Company domain, keep in mind that apps published to Google
Play must have a unique package name. Because domains are unique, prepending the app's
name with your name, or your company's domain name, should provide an adequately
unique package name. If you don't plan to publish the app, you can accept the default
example domain. Be aware that changing the package name later is extra work.

Choosing target devices and the minimum SDK

When choosing Target Android Devices, Phone and Tablet are selected by default, as
shown in the figure below. The choice shown in the figure for the Minimum SDK—API 15:
Android 4.0.3 (IceCreamSandwich)—makes your app compatible with 97% of Android-
powered devices active on the Google Play Store.

Different devices run different versions of the Android system, such as Android 4.0.3 or
Android 4.4. Each successive version often adds new APIs not available in the previous
version. To indicate which set of APIs are available, each version specifies an API level.
For instance, Android 1.0 is API level 1 and Android 4.0.3 is API level 15.

The Minimum SDK declares the minimum Android version for your app. Each successive
version of Android provides compatibility for apps that were built using the APIs from
previous versions. That means your app should always be compatible with future versions
of Android, if you use the documented Android APIs.

Choosing an Activity template

An Activity is a single, focused thing that the user can do. It is a crucial component of any
Android app. An Activity typically has a layout associated with it that defines how UI
elements appear on a screen.
Android Studio pre-populates your project with minimal code for an Activity and layout
based on a template. Available Activity templates range from a virtually blank template
(Add No Activity) to an Activity that includes navigation and an options menu.

You can customize the Activity after you select your template. For example, the Empty
Activity choice provides a single Activity with a single layout resource for the screen.
The Configure Activity screen appears after you click Next. On the Configure
Activity screen you can accept the commonly used name for the Activity (such
as MainActivity), or you can change the name.
Tip: This course covers the Activity class in more detail in another practical. You can also
read Introduction to Activities for a comprehensive introduction.

The Configure Activity screen differs depending on which template you chose. In most
cases you can select the following options, if they are not already selected:

 Generate Layout file: Leave this checkbox selected to create the layout resource connected
to this Activity, which is usually named activity_main. The layout defines the UI for
the Activity.
 Backwards Compatibility (AppCompat): Leave this checkbox selected to include
the AppCompat library. Use the AppCompat library to make sure that the app is compatible
with previous versions of Android, even if the app uses features found only in newer Android
versions.

Android Studio creates a folder for your projects, and builds the project with Gradle.

Tip: See the Configure your build developer page for detailed information.

Exploring a project

An Android Studio project contains all of the source code and all resources for an app. The
resources include layouts, strings, colors, dimensions, and images. The Android Studio main
window is made up of several logical areas, or panes, as shown in the figure below.

In the figure above:

1. Toolbar: Provides a wide range of actions, including running the Android app and launching
Android tools.
2. Navigation bar: Navigate through the project and open files for editing.
3. Project pane: Displays project files in a hierarchy. The selected hierarchy in the figure
above is Android.
4. Editor: The contents of a selected file in the project. For example, after you select a layout
(as shown in the figure above), the editor pane shows the layout editor with tools to edit the
layout. After you select a Java code file, the editor pane shows the Java code with tools for
editing the code.
5. Tabs along the left, right, and bottom of the window: You can click tabs to open other panes,
such as Logcat to open the Logcat pane with log messages, or TODO to manage tasks.

The status bar at the bottom of the Android Studio window displays the status of the project
and Android Studio itself, as well as any warnings or messages. You can watch the build
progress in the status bar.

Tip: You can organize the main window to give yourself more screen space by hiding or
moving panes. You can also use keyboard shortcuts to access most features. See Keyboard
Shortcuts for a complete list.

Using the Project pane

You can view the project organization in several ways in the Project pane. If it is not already
selected, click the Project tab. (The Project tab is in the vertical tab column on the left side
of the Android Studio window.)
The Project pane appears. To view the project in the standard Android project hierarchy,
select Android from the Down arrow at the top of the Project pane.

Note: This chapter and other chapters refer to the Project pane, when set to Android, as
the Project > Android pane.

Gradle files

When you first create an app project, the Project > Android pane appears with the Gradle
Scripts folder expanded as shown below. If the Gradle Scripts folder is not expanded, click
the triangle to expand it. This folder contains all the files needed by the build system.

The build.gradle(Module:app) file specifies additional libraries and the module's build
configuration. The Activity template that you select creates this file. The file includes
the minSdkVersion attribute that declares the minimum version for the app, and
the targetSdkVersion attribute that declares the highest (newest) version for which the app
has been optimized.
This file also includes a list of dependencies, which are libraries required by the code—such
as the AppCompat library for supporting a wide range of Android versions.
App code
To view and edit the Java code, expand the app folder, the java folder, and
the com.example.android.helloworld folder. Double-click the MainActivity java file to

open it in the code editor.


The java folder includes Java class files. Each Activity, Service, or other component (such
as a Fragment) is defined as a Java class, usually in its own file. Tests and other Java class
files are also located here.
The java folder contains three subfolders:

 com.example.hello.helloworld (or the domain name you have specified): All the files for a
package are in a folder named after the package. For your Hello World app, there is one
package, and it contains only MainActivity.java. The first Activity (screen) that the user
sees, which also initializes app-wide resources, is customarily called MainActivity. (The file
extension is omitted in the Project > Android pane.)
 com.example.hello.helloworld(androidTest): This folder is for your instrumented tests, and
starts out with a skeleton test file.
 com.example.hello.helloworld(test): This folder is for your unit tests and starts out with an
automatically created skeleton unit test file.

Layout files
To view and edit a layout file, expand the res folder and the layout folder to see the layout
file. In the figure below, the layout file is called activity_main.xml.
Double-click the file to open it in the layout editor. Layout files are written in XML.

Resource files

The res folder holds resources, such as layouts, strings, and images. An Activity is usually
associated with a layout of UI views that are defined as an XML file. This XML file is
usually named after its Activity. The res folder includes these subfolders:

 drawable: Store all your app's images in this folder.


 layout: Every Activity has at least one XML layout file that describes the UI. For Hello
World, this folder contains activity_main.xml.
 mipmap: The launcher icons are stored in this folder. There is a subfolder for each supported
screen density. Android uses the screen density (the number of pixels per inch) to determine
the required image resolution. Android groups all actual screen densities into generalized
densities, such as medium (mdpi), high (hdpi), or extra-extra-extra-high (xxxhdpi).
The ic_launcher.png folder contains the default launcher icons for all the densities supported
by your app.
 values: Instead of hardcoding values like strings, dimensions, and colors in your XML and
Java files, it is best practice to define them in their respective values files. This practice
makes it easier to change the values and keep the values consistent across your app.

The values subfolder includes these subfolders:

 colors.xml: Shows the default colors for your chosen theme. You can add your own colors
or change the colors based on your app's requirements.
 dimens.xml: Store the sizes of views and objects for different resolutions.
 strings.xml: Create resources for all your strings. Doing this makes it easy to translate the
strings to other languages.
 styles.xml: All the styles for your app and theme go here. Styles help give your app a
consistent look for all UI elements.

Using the editor pane


If you select a file, the editor pane appears. A tab appears for the file so that you can open
multiple files and switch between them. For example, if you double-click
the activity_main.xml layout file in the Project > Android pane, the layout editor appears
as shown below.
If you double-click the MainActivity file in the Project > Android pane, the editor changes
to the code editor as shown below, with a tab for MainActivity.java:

At the top of the MainActivity.java file is a package statement that defines the app package.
This package statement is followed by an import block condensed with ..., as shown in the
figure above. Click the dots to expand the block to view it. The import statements import
libraries needed for the app. For example, the following statement imports
the AppCompatActivity library:
import android.support.v7.app.AppCompatActivity;
Each Activity in an app is implemented as a Java class. The following class declaration
extends the AppCompatActivity class to implement features in a way that is backward-
compatible with previous versions of Android:
public class MainActivity extends AppCompatActivity {
// ... Rest of the code for the class.
}

Understanding the Android manifest

Before the Android system can start an app component such as an Activity, the system must
know that the Activity exists. It does so by reading the app's AndroidManifest.xml file,
which describes all of the components of your Android app. Each Activity must be listed in
this XML file, along with all components for the app.
To view and edit the AndroidManifest.xml file, expand the manifests folder in the Project
> Android pane, and double-click AndroidManifest.xml. Its contents appear in the editing
pane:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.android.helloworld">

<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />

<category android:name="android.intent.category.LAUNCHER" />


</intent-filter>
</activity>
</application>

</manifest>

Android namespace and application tag

The Android Manifest is coded in XML and always uses the Android namespace:

xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.android.helloworld">
The package expression shows the unique package name of the new app. Do not change the
package expression after the app is published.
The <application> tag, with its closing </application> tag, defines the manifest settings for
the entire app.

Automatic backup
The android:allowBackup attribute enables automatic app data backup:
android:allowBackup="true"
Setting the android:allowBackup attribute to true enables the app to be backed up
automatically and restored as needed. Users invest time and effort to configure apps.
Switching to a new device can cancel out all that careful configuration. The system performs
this automatic backup for nearly all app data by default, and does so without the developer
having to write any additional app code.
For apps whose target SDK version is Android 6.0 (API level 23) and higher, devices
running Android 6.0 and higher automatically create backups of app data to the cloud
because the android:allowBackup attribute defaults to true if omitted. For apps < API level
22 you have to explicitly add the android:allowBackup attribute and set it to true.
Tip: To learn more about the automatic backup for apps, see Configuring Auto Backup for
Apps.

The app icon

The android:icon attribute sets the icon for the app:


android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
The android:icon attribute assigns to the app an icon in the mipmap folder (inside
the res folder in the Project > Android pane). The icon appears on the home screen or in
the Search Apps screen for launching the app. The icon is also used as the default icon for
app components.
App label and string resources

The android:label attribute shows the string "Hello World" highlighted. If you click the
string, it changes to show the string resource @string/app_name:
android:label="@string/app_name"
Tip: To see the context menu, ctrl-click or right-click app_name in the editor pane.
Select Go To > Declaration to see where the string resource is declared: in
the strings.xml file. When you select Go To > Declaration or open the file by double-
clicking strings.xml inside the values folder in the Project > Android pane, the file's
contents appear in the editor pane.
After opening the strings.xml file, you can see that the string name app_name is set to Hello
World. You can change the app name by changing the Hello World string to something else.
String resources are described in a separate lesson.

App theme

The android:theme attribute sets the app's theme, which defines the appearance of UI
elements such as text:
android:theme="@style/AppTheme">
The theme attribute is set to the standard theme AppTheme. Themes are described in a
separate lesson.

Declaring the Android version


Different devices may run different versions of the Android system, such as Android 4.0 or
Android 4.4. Each successive version can add new APIs not available in the previous
version. To indicate which set of APIs are available, each version specifies an API level.
For instance, Android 1.0 is API level 1 and Android 4.4 is API level 19.

The API level allows a developer to declare the minimum version with which the app is
compatible, using the <uses-sdk> manifest tag and its minSdkVersion attribute. For
example, the Calendar Provider APIs were added in Android 4.0 (API level 14). If your app
can't function without these APIs, declare API level 14 as the app's minimum supported
version like this:
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.android.helloworld">
<uses-sdk android:minSdkVersion="14" android:targetSdkVersion="19" />
// ... Rest of manifest information
</manifest>
The minSdkVersion attribute declares the minimum version for the app, and
the targetSdkVersion attribute declares the highest (newest) version which has been
optimized within the app. Each successive version of Android provides compatibility for
apps that were built using the APIs from previous versions, so the app should always be
compatible with future versions of Android while using the documented Android APIs.
The targetSdkVersion attribute does not prevent an app from being installed on Android
versions that are higher (newer) than the specified value. Even so, the target attribute is
important, because it indicates to the system whether the app should inherit behavior
changes in newer versions.
If you don't update the targetSdkVersion to the latest version, the system assumes that your
app requires backward-compatible behaviors when it runs on the latest version. For example,
among the behavior changes in Android 4.4, alarms created with the AlarmManager APIs
are now inexact by default so that the system can batch app alarms and preserve system
power. If your target API level is lower than "19", the system retains the previous API's
behavior for your app.

Understanding the build process

The Android application package (APK) is the package file format for distributing and
installing Android mobile apps. The build process involves tools and processes that
automatically convert each project into an APK.

Android Studio uses Gradle as the foundation of the build system, with more Android-
specific capabilities provided by the Android Plugin for Gradle. This build system runs as
an integrated tool from the Android Studio menu.

Understanding build.gradle files

When you create a project, Android Studio automatically generates the necessary build files
in the Gradle Scripts folder in the Project > Android pane. Android Studio build files are

named build.gradle as shown below:

build.gradle (Project: apptitle)

This file is the top-level build file for the entire project, located in the root project folder,
which defines build configurations that apply to all modules in your project. This file,
generated by Android Studio, should not be edited to include app dependencies.

If a dependency is something other than a local library or file tree, Gradle looks for the files
in whichever online repositories are specified in the repositories block of this file. By
default, new Android Studio projects declare JCenter and Google (which includes
the Google Maven repository) as the repository locations:

allprojects {
repositories {
google()
jcenter()
}
}

build.gradle (Module: app)


Android Studio creates separate build.gradle (Module: app) files for each module. You can
edit the build settings to provide custom packaging options for each module, such as
additional build types and product flavors, and to override settings in the manifest or top-
level build.gradle file. This file is most often the file to edit when changing app-level
configurations, such as declaring dependencies in the dependencies section. The following
shows the contents of a project's build.gradle (Module: app) file:
apply plugin: 'com.android.application'

android {
compileSdkVersion 26
defaultConfig {
applicationId "com.example.android.helloworld"
minSdkVersion 15
targetSdkVersion 26
versionCode 1
versionName "1.0"
testInstrumentationRunner
"android.support.test.runner.AndroidJUnitRunner"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'),
'proguard-rules.pro'
}
}
}

dependencies {
implementation fileTree(dir: 'libs', include: ['*.jar'])
implementation 'com.android.support:appcompat-v7:26.1.0'
implementation 'com.android.support.constraint:constraint-layout:1.0.2'
testImplementation 'junit:junit:4.12'
androidTestImplementation 'com.android.support.test:runner:1.0.1'
androidTestImplementation
'com.android.support.test.espresso:espresso-core:3.0.1'
}
The build.gradle files use Gradle syntax. Gradle is a Domain Specific Language (DSL) for
describing and manipulating the build logic using Groovy, which is a dynamic language for
the Java Virtual Machine (JVM). You don't need to learn Groovy to make changes, because
the Android Plugin for Gradle introduces most of the DSL elements you need.
Tip: To learn more about the Android plugin DSL, read the DSL reference documentation.

Plugin and Android blocks

In the build.gradle (Module: app) file above, the first statement applies the Android-specific
Gradle plug-in build tasks:
apply plugin: 'com.android.application'

android {
compileSdkVersion 26
// ... Rest of android block.
}
The android { } block specifies the target SDK version for compiling the app code
( compileSdkVersion 26) and several blocks of information.

The defaultConfig block


Core settings and entries for the app are specified in the defaultConfig { } block within
the android { } block:
defaultConfig {
applicationId "com.example.android.helloworld"
minSdkVersion 15
targetSdkVersion 26
versionCode 1
versionName "1.0"
testInstrumentationRunner
"android.support.test.runner.AndroidJUnitRunner"
}
The minSdkVersion and targetSdkVersion settings override
any AndroidManifest.xml settings for the minimum SDK version and the target SDK
version. See "Declaring the Android version" previously in this chapter for background
information on these settings.
The testInstrumentationRunner statement adds the instrumentation support for testing the
UI using Espresso and UIAutomator. These tools are described in a separate lesson.

Build types
Build types for the app are specified in a buildTypes { } block, which controls how the app
is built and packaged.
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'),
'proguard-rules.pro'
}
}
The build type specified is release for the app's release. Another common build type
is debug. Configuring build types is described in a separate lesson.

Dependencies
Dependencies for the app are defined in the dependencies { } block, which is the part of
the build.gradle file that is most likely to change as you start developing code that depends
on other libraries. The block is part of the standard Gradle API and
belongs outside the android { } block.
dependencies {
implementation fileTree(dir: 'libs', include: ['*.jar'])
implementation 'com.android.support:appcompat-v7:26.1.0'
implementation 'com.android.support.constraint:constraint-layout:1.0.2'
testImplementation 'junit:junit:4.12'
androidTestImplementation 'com.android.support.test:runner:1.0.1'
androidTestImplementation
'com.android.support.test.espresso:espresso-core:3.0.1'
}
In the snippet above, the statement implementation fileTree(dir: 'libs', include: ['*.jar']) adds
a dependency of all ".jar" files inside the libs folder.

Syncing your project

When you make changes to the build configuration files in a project, Android Studio requires
that you sync the project files. During the sync, Android Studio imports the build
configuration changes and runs checks to make sure the configuration won't create build
errors.

To sync the project files, click Sync Now in the notification bar that appears when making
a change (as shown in the figure below), or click the Sync Project with Gradle

Files button in the menu bar.

If Android Studio notices any errors with the configuration — for example, if the source
code uses API features that are only available in an API level higher than
the compileSdkVersion—the Messages window appears to describe the issue.

Running the app on an emulator or a device

With virtual device emulators, you can test an app on different devices such as tablets or
smartphones—with different API levels for different Android versions—to make sure it
looks good and works for most users. You don't have to depend on having a physical device
available for app development.

The Android Virtual Device (AVD) manager creates a virtual device or emulator that
simulates the configuration for a particular type of Android-powered device. Use the AVD
Manager to define the hardware characteristics of a device and its API level, and to save it
as a virtual device configuration. When you start the Android emulator, it reads a specified
configuration and creates an emulated device on your computer that behaves exactly like a
physical version of that device.

Creating a virtual device


To run an emulator on your computer, use the AVD Manager to create a configuration that
describes the virtual device. Select Tools > Android > AVD Manager, or click the AVD

Manager icon in the toolbar.

The Your Virtual Devices screen appears showing all of the virtual devices created
previously. Click the +Create Virtual Device button to create a new virtual device.

You can select a device from a list of predefined hardware devices. For each device, the
table provides a column for its diagonal display size (Size), screen resolution in pixels
(Resolution), and pixel density (Density). For example, the pixel density of the Nexus 5
device is xxhdpi, which means the app uses the icons in the xxhdpi folder of
the mipmap folder. Likewise, the app uses layouts and drawables from folders defined for
that density.

After you click Next, the System Image screen appears for choosing the version of the
Android system for the device. The Recommended tab shows the recommended systems
for the device. More versions are available under the x86 Images and Other Images tabs.
If a Download link is visible next to a system image version, it is not installed yet. Click
the link to start the download, and click Finish when it's done.

Running the app on the virtual device

To run the app on the virtual device you created in the previous section, follow these steps:

1. In Android Studio, select Run > Run app or click the Run icon in the toolbar.

2. In the Select Deployment Target window, under Available Emulators, select the virtual
device you created, and click OK.

The emulator starts and boots just like a physical device. Depending on the speed of your
computer, the startup process might take a while. The app builds, and once the emulator is
ready, Android Studio uploads the app to the emulator and runs it.

You should see the app created from the Empty Activity template ("Hello World") as shown
in the following figure, which also shows Android Studio's Run pane that displays the
actions performed to run the app on the emulator.

Tip: When testing on a virtual device, it is a good practice to start it up once, at the very
beginning of your session. Do not close it until you are done testing your app, so t hat your
app doesn't have to go through the device startup process again. To close the virtual device,
select Quit from the menu or press Control-Q in Windows or Command-Q in macOS.

The figure above shows the emulator and the run log:

1. The Emulator running the app.


2. The Run pane, which shows the actions taken to install and run the app.
3. The Run tab, which you click to open or close the Run pane.

Running the app on a physical device

Always test your apps on a physical device. While emulators are useful, they can't show all
possible device states, such as what happens if an incoming call occurs while the app is
running. To run the app on a physical device, you need the following:

 An Android-powered device such as a phone or tablet.


 A data cable to connect your Android-powered device to your computer via the USB port.
 If you are using a Linux or Windows system, you may need to perform additional steps to
run on a hardware device. Check the Using Hardware Devices documentation. You may also
need to install the appropriate USB driver for your device. See OEM USB Drivers.

To let Android Studio communicate with your Android-powered device, you must turn on
USB Debugging on the device. You enable USB Debugging in the device's Developer
options settings. (Note that enabling USB Debugging is not the same as rooting your
device.)

On Android 4.2 and higher, the Developer options screen is hidden by default. To show
developer options and enable USB Debugging:

1. On your device, open Settings > About phone and tap Build number seven times.
2. Return to the previous screen (Settings). Developer options appears at the bottom of the
list. Tap Developer options.
3. Select USB Debugging.
4. Connect the device and run the app from Android Studio.
Using the log

The log is a powerful debugging tool you can use to look at values, execution paths, and
exceptions. After you add logging statements to an app, your log messages appear along
with general log messages in the Logcat pane.

Viewing log messages

To see the Logcat pane, click the Logcat tab at the bottom of the Android Studio window
as shown in the figure below.

In the figure above:

1. The Logcat tab for opening and closing the Logcat pane, which displays information about
your app as it is running. If you add Log statements to your app, Log messages appear here.
2. The Log level menu set to Verbose (the default), which shows all Log messages. Other
settings include Debug, Error, Info, and Warn.

Adding logging statements to your app

Logging statements add whatever messages you specify to the log. Adding logging
statements at certain points in the code allows the developer to look at values, execution
paths, and exceptions. For example, the following logging statement
adds "MainActivity" and "Hello World" to the log:
Log.d("MainActivity", "Hello World");
The following are the elements of this statement:

 Log: The Log class for sending log messages to the Logcat pane.
 d: The Debug Log level setting to filter log message display in the Logcat pane. Other log
levels are e for Error, w for Warn, and i for Info. You assign a log level so that you can
filter the log messages using the drop-down menu in the center of the Logcat pane.
 "MainActivity": The first argument is a tag which can be used to filter messages in
the Logcat pane. This tag is commonly the name of the Activity from which the message
originates. However, you can name the tag anything that is useful to you for debugging.
 "Hello world": The second argument is the actual message.

By convention, log tags are defined as constants for the Activity:


private static final String LOG_TAG = MainActivity.class.getSimpleName();
Use the constant in the logging statements:

Log.d(LOG_TAG, "Hello World");


After you add the Log.d statement shown above, follow these steps to see the log message:

1. If the Logcat pane is not already open, click the Logcat tab at the bottom of Android Studio
to open it.
2. Change the Log level in the Logcat pane to Debug. (You can also leave the Log level
as Verbose, because there are so few log messages.)
3. Run your app on a virtual device.

The following message should appear in the Logcat pane:

11-24 14:06:59.001 4696-4696/? D/MainActivity: Hello World

1.2: Layouts and resources for the UI

Contents:

 Views
 The layout editor
 Editing XML directly
 Resource files
 Responding to View clicks
 Related practicals
 Learn more

This chapter describes the screen's user interface (UI) layout and other resources you create
for your app, and the code you would use to respond to a user's tap of a UI element.

Views

The UI consists of a hierarchy of objects called views — every element of the screen is
a View. The View class represents the basic building block for all UI components, and the
base class for classes that provide interactive UI components such as buttons, checkboxes,
and text entry fields.
A View has a location, expressed as a pair of left and top coordinates, and two dimensions,
expressed as a width and a height. The unit for location and dimensions is the density-
independent pixel (dp).
The Android system provides hundreds of predefined View subclasses. Commonly
used View subclasses described over several lessons include:
 TextView for displaying text
 EditText to enable the user to enter and edit text
 Button and other clickable elements (such as RadioButton, CheckBox, and Spinner) to
provide interactive behavior
 ScrollView and RecyclerView to display scrollable items
 ImageView for displaying images
 ConstraintLayout and LinearLayout for containing other views and positioning them

You can define a View to appear on the screen and respond to a user tap. A View can also
be defined to accept text input, or to be invisible until needed.
You can specify View elements in layout resource files. Layout resources are written in
XML and listed within the layout folder in the res folder in the Project > Android pane.

ViewGroup groups

View elements can be grouped inside a ViewGroup, which acts as a container. The
relationship is parent-child, in which the parent is a ViewGroup, and the child is a View or
another ViewGroup. The following are commonly used ViewGroup groups:

 ConstraintLayout: A group that places UI elements (child View elements) using constraint
connections to other elements and to the layout edges (parent View).
 ScrollView: A group that contains one other child View element and enables scrolling the
child View element.
 RecyclerView: A group that contains a list of other View elements or ViewGroup groups
and enables scrolling them by adding and removing View elements dynamically from the
screen.

Layout ViewGroup groups

The View elements for a screen are organized in a hierarchy. At the root of this hierarchy is
a ViewGroup that contains the layout of the entire screen. The ViewGroup can contain
child View elements or other ViewGroup groups as shown in the following figure.

In the figure above:

1. The root ViewGroup.


2. The first set of child View elements and ViewGroup groups whose parent is the root.
Some ViewGroup groups are designated as layouts because they organize
child View elements in a specific way and are typically used as the root ViewGroup. Some
examples of layouts are:

 ConstraintLayout: A group of child View elements using constraints, edges, and guidelines
to control how the elements are positioned relative to other elements in the
layout. ConstraintLayout was designed to make it easy to click and drag View elements in
the layout editor.
 LinearLayout: A group of child View elements positioned and aligned horizontally or
vertically.
 RelativeLayout: A group of child View elements in which each element is positioned and
aligned relative to other elements within the ViewGroup. In other words, the positions of
the child View elements can be described in relation to each other or to the
parent ViewGroup.
 TableLayout: A group of child View elements arranged into rows and columns.
 FrameLayout: A group of child View elements in a stack. FrameLayout is designed to block
out an area on the screen to display one View. Child View elements are drawn in a stack,
with the most recently added child on top. The size of the FrameLayout is the size of its
largest child View element.
 GridLayout: A group that places its child View elements in a rectangular grid that can be
scrolled.

Tip: Learn more about different layout types in Common Layout Objects.

A simple example of a LinearLayout with child View elements is shown below as a diagram
of the layout file (activity_main.xml), along with a hierarchy diagram (top right) and a
screenshot of the actual finished layout (bottom right).

In the figure above:

1. LinearLayout, the root ViewGroup, contains all the child View elements in a vertical
orientation.
2. Button (button_toast). The first child View element appears at the top in the LinearLayout.
3. TextView (show_count). The second child View element appears under the first
child View element in the LinearLayout.
4. Button (button_count). The third child View element appears under the second
child View element in the LinearLayout.

The layout hierarchy can grow to be complex for an app that shows many View elements on
a screen. It's important to understand the hierarchy, as it affects whether View elements are
visible and how efficiently they are drawn.
Tip: You can explore the layout hierarchy of your app using Hierarchy Viewer. It shows a
tree view of the hierarchy and lets you analyze the performance of View elements on an
Android-powered device. Performance issues are covered in a subsequent chapter.

The layout editor

You define layouts in the layout editor, or by entering XML code.

The layout editor shows a visual representation of XML code. You can drag View elements
into the design or blueprint pane and arrange, resize, and specify attributes for them. You
immediately see the effect of changes you make.
To use the layout editor, double-click the XML layout file (activity_main.xml). The layout
editor appears with the Design tab at the bottom highlighted. (If the Text tab is highlighted
and you see XML code, click the Design tab.) For the Empty Activity template, the layout
is as shown in the figure below.

1. XML layout file (activity_main.xml).


2. Design and Text tabs. Click Design to see the layout editor, or Text to see XML code.
3. Palette pane. The Palette pane provides a list of UI elements and layouts. Add an element
or layout to the UI by dragging it into the design pane.
4. Component Tree. The Component Tree pane shows the layout hierarchy. Click
a View element or ViewGroup in this pane to select it. View elements are organized into a
tree hierarchy of parents and children, in which a child inherits the attributes of its parent.
In the figure above, the TextView is a child of the ConstraintLayout.
5. Design and blueprint panes. Drag View elements from the Palette pane to the design or
blueprint pane to position them in the layout. In the figure above, the layout shows only one
element: a TextView that displays "Hello World".
6. Attributes tab. Click Attributes to display the Attributes pane for setting attributes for
a View element.

Layout editor toolbars

The layout editor toolbars provide buttons to configure your layout and change its
appearance. The top toolbar lets you configure the appearance of the layout preview in the
layout editor:

The figure above shows the top toolbar of the layout editor:
1. Select Design Surface: Select Design to display a color preview of the UI elements in your
layout, or Blueprint to show only outlines of the elements. To see both panes side by side,
select Design + Blueprint.
2. Orientation in Editor: Select Portrait or Landscape to show the preview in a vertical or
horizontal orientation. The orientation setting lets you preview the layout orientations
without running the app on an emulator or device. To create alternative layouts,
select Create Landscape Variation or other variations.
3. Device in Editor: Select the device type (phone/tablet, Android TV, or Android Wear).
4. API Version in Editor: Select the version of Android to use to show the preview.
5. Theme in Editor: Select a theme (such as AppTheme) to apply to the preview.
6. Locale in Editor: Select the language and locale for the preview. This list displays only the
languages available in the string resources (see the lesson on localization for details on how
to add languages). You can also select Preview as Right To Left to see the layout as if an
RTL language had been chosen.

The layout editor also offers a second toolbar that lets you configure the appearance of UI
elements in a ConstraintLayout and zoom in and out of the preview:

The figure above shows the ConstraintLayout editing toolbar:

1. Show: Select Show Constraints and Show Margins to show them in the preview, or to stop
showing them.
2. Autoconnect: Enable or disable Autoconnect. With Autoconnect enabled, you can drag any
element (such as a Button) to any part of a layout to generate constraints against the parent
layout.
3. Clear All Constraints: Clear all constraints in the entire layout.
4. Infer Constraints: Create constraints by inference.
5. Default Margins: Set the default margins.
6. Pack: Pack or expand the selected elements.
7. Align: Align the selected elements.
8. Guidelines: Add vertical or horizontal guidelines.
9. Zoom controls: Zoom in or out.

Using ConstraintLayout

The layout editor offers more features in the Design tab when you use a ConstraintLayout,
including handles for defining constraints.
A constraint is a connection or alignment to another UI element, to the parent layout, or to
an invisible guideline. Each constraint appears as a line extending from a circular handle.
After you select a UI element in the Component Tree pane or click it in the layout editor,
the element shows a resizing handle on each corner and a circular constraint handle in the
middle of each side.

The figure above shows the constraint and resizing handles on View elements in a layout:

1. Resizing handle.
2. Constraint line and handle. In the figure, the constraint aligns the left side of
the Toast Button to the left side of the layout.
3. Constraint handle without a constraint line.
4. Baseline handle. The baseline handle aligns the text baseline of an element to the text
baseline of another element.

In the blueprint or design panes, the following handles appear on the TextView element:
 Constraint handle: To create a constraint, click a constraint handle, shown as a circle on
each side of an element. Then drag the circle to another constraint handle or to a parent

boundary. A zigzag line represents the constraint.

 Resizing handle: You can drag the square resizing handles to resize the element. While

dragging, the handle changes to an angled corner.


You can drag the resizing handles on each corner of the UI element to resize it, but doing so
hard-codes the width and height dimensions, which you should avoid for most elements
because hard-coded dimensions don't adapt to different screen densities.

Constraining a UI element

To add a constraint to a UI element, click the circular handle and drag a line to another
element or to the side of a layout, as shown in the two animated figures below. To remove
a constraint from an element, click the circular handle.
The constraints you define in the layout editor are created as XML attributes, which you can
see in the Text tab as described in "Editing XML directly" in this chapter. For example, the
following XML code is created constraining the top of an element to the top of its parent:

app:layout_constraintTop_toTopOf="parent"

Using a baseline constraint

You can align one UI element that contains text, such as a TextView or Button, with another
UI element that contains text. A baseline constraint lets you constrain the elements so that
the text baselines match. Select the UI element that has text, and then hover your pointer
over the element until the baseline constraint button appears underneath the element.
Click the baseline constraint button. The baseline handle appears, blinking in green as shown
in the animated figure. Drag a baseline constraint line to the baseline of the other UI element.

Tip: For an in-depth tutorial on using ConstraintLayout, see Using ConstraintLayout to


design your views.

Using the Attributes pane

The Attributes pane offers access to all of the XML attributes you can assign to a UI
element. You can find the attributes (known as properties) common to all views in
the View class documentation.
To show the Attributes pane, click the Attributes tab on the right side of the layout editor.
The Attributes pane includes a square sizing panel called the view inspector. The symbols
inside the view inspector represent the height and width settings.

The figure above shows the Attributes pane:

1. Vertical view size control. The vertical size control, which appears on the top and bottom
of the view inspector, specifies the layout_height property. The angles indicate that this size
control is set to wrap_content, which means the UI element expands vertically as needed to
fit its contents. The "8" indicates a standard margin set to 8 dp.
2. Horizontal view size control. The horizontal size control, which appears on the left and
right of the view inspector, specifies the layout_width. The angles indicate that this size
control is set to wrap_content, which means the UI element expands horizontally as needed
to fit its contents, up to a margin of 8 dp.
3. Attributes pane close button. Click to close the pane.

The layout_width and layout_height attributes in the Attributes pane change as you change
the inspector's horizontal and vertical size controls. These attributes can take one of three
values for a ConstraintLayout:

 The match_constraint setting expands the UI element to fill its parent by width or height up
to a margin, if a margin is set. The parent in this case is the ConstraintLayout.
 The wrap_content setting shrinks the UI element to the size of its content. If there is no
content, the element becomes invisible.
 To specify a fixed size that's adjusted for the screen size of the device, set a number
of dp (density-independent pixels). For example, 16dp means 16 density-independent
pixels.

Tip: If you change the layout_width attribute using its popup menu,
the layout_width attribute is set to zero because there is no set dimension. This setting is the
same as match_constraint—the UI element can expand as much as possible to meet
constraints and margin settings.
The Attributes pane offers access to all of the attributes you can assign to a View element.
You can enter values for each attribute, such as the android:id, background, textColor,
and text attributes.

Creating layout variants for orientations and devices

You can preview an app's layout with a horizontal orientation, and with different devices,
without having to run the app on an emulator or device.

To preview the layout for a different orientation, click the Orientation in


Editor button in the top toolbar. To show the layout in a horizontal orientation,
select Switch to Landscape. To return to vertical orientation, select Switch to Portrait.

You can also preview the layout for different devices. Click the Device in
Editor button in the top toolbar, and select a different device in the drop-down
menu. For example, select Nexus 4, Nexus 5, and then Pixel to see differences in the
previews.

To create a variant of the layout strictly for the horizontal orientation, leaving the vertical
orientation layout alone: click the Orientation in Editor button and select Create
Landscape Variation. A new editor window opens with the land/activity_main.xml tab
showing the layout for the landscape (horizontal) orientation. You can change this layout,
which is specifically for horizontal orientation, without changing the original portrait
(vertical) orientation.
In the Project > Android pane, look inside the res > layout directory. You see that Android
Studio automatically creates the variant for you, called activity_main.xml (land).

To create a layout variant for tablet-sized screens, click the Orientation in Editor button
and select Create layout x-large Variation. A new editor window opens with
the xlarge/activity_main.xml tab showing the layout for a tablet-sized device. The editor
also picks a tablet device, such as the Nexus 9 or Nexus 10, for the preview. In the Project
> Android pane, look inside the res > layout directory. You see that Android Studio
automatically creates the variant for you, called activity_main.xml (xlarge). You can change
this layout, which is specifically for tablets, without changing the other layouts.

Editing XML directly

It is sometimes quicker and easier to edit the XML code directly, especially when copying
and pasting the code for similar views.

To view and edit the XML code, open the XML layout file. The layout editor appears with
the Design tab at the bottom highlighted. Click the Text tab to see the XML code. The
following shows the XML code for a LinearLayout with two Button elements with
a TextView in the middle:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context="com.example.android.hellotoast.MainActivity">

<Button
android:id="@+id/button_toast"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginEnd="8dp"
android:layout_marginStart="8dp"
android:layout_marginTop="8dp"
android:background="@color/colorPrimary"
android:onClick="showToast"
android:text="@string/button_label_toast"
android:textColor="@android:color/white" />

<TextView
android:id="@+id/show_count"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="center_vertical"
android:layout_marginBottom="8dp"
android:layout_marginEnd="8dp"
android:layout_marginStart="8dp"
android:layout_marginTop="8dp"
android:background="#FFFF00"
android:text="@string/count_initial_value"
android:textAlignment="center"
android:textColor="@color/colorPrimary"
android:textSize="160sp"
android:textStyle="bold"
android:layout_weight="1"/>

<Button
android:id="@+id/button_count"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="8dp"
android:layout_marginEnd="8dp"
android:layout_marginStart="8dp"
android:background="@color/colorPrimary"
android:onClick="countUp"
android:text="@string/button_label_count"
android:textColor="@android:color/white" />
</LinearLayout>

XML attributes (view properties)

Views have properties that define where a view appears on the screen, its size, how the view
relates to other views, and how it responds to user input. When defining views in XML or
in the layout editor's Attributes pane, the properties are referred to as attributes.

For example, in the following XML description of a TextView,


the android:id, android:layout_width, android:layout_height, android:background, are
XML attributes that are translated automatically into the TextView properties:
<TextView
android:id="@+id/show_count"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@color/myBackgroundColor"
android:textStyle="bold"
android:text="@string/count_initial_value" />
Attributes generally take this form:

android:attribute_name="value"
The attribute_name is the name of the attribute. The value is a string with the value for the
attribute. For example:

android:textStyle="bold"
If the value is a resource, such as a color, the @ symbol specifies what kind of resource. For
example:
android:background="@color/myBackgroundColor"
The background attribute is set to the color resource identified as myBackgroundColor,
which is declared to be #FFF043. Color resources are described in Style-related attributes in
this chapter.
Every View and ViewGroup supports its own variety of XML attributes:

 Some attributes are specific to a View subclass. For example, the TextView subclass
supports the textSize attribute. Any elements that extend the TextView subclass inherit
these subclass-specific attributes.
 Some attributes are common to all View elements, because they are inherited from the
root View class. The android:id attribute is one example.

For descriptions of specific attributes, see the overview section of the View class
documentation.

Identifying a View

To uniquely identify a View and reference it from your code, you must give it an id.
The android:id attribute lets you specify a unique id—a resource identifier for a View.
For example:

android:id="@+id/button_count"
The @+id/button_count part of the attribute creates an id called button_count for
a Button (a subclass of View). You use the plus (+) symbol to indicate that you are creating
a new id.

Referencing a View

To refer to an existing resource identifier, omit the plus (+) symbol. For example, to refer
to a View by its id in another attribute, such as android:layout_toLeftOf (described in the
next section) to control the position of a View, you would use:
android:layout_toLeftOf="@id/show_count"
In the attribute above, @id/show_count refers to the View with the resource
identifier show_count. The attribute positions the element to be "to the left of"
the show_count View.

Positioning a View
Some layout-related positioning attributes are required for a View or a ViewGroup, and
automatically appear when you add the View or ViewGroup to the XML layout.

LinearLayout positioning

LinearLayout is required to have these attributes set:

 android:layout_width
 android:layout_height
 android:orientation

The android:layout_width and android:layout_height attributes can take one of three values:

 match_parent expands the UI element to fill its parent by width or height. When the
LinearLayout is the root ViewGroup, it expands to the size of the device screen. For a UI
element within a root ViewGroup, it expands to the size of the parent ViewGroup.
 wrap_content shrinks the UI element to the size of its content. If there is no content, the
element becomes invisible.
 Use a fixed number of dp (density-independent pixels) to specify a fixed size, adjusted for
the screen size of the device. For example, 16dp means 16 density-independent pixels.
Density-independent pixels and other dimensions are described in "Dimensions" in this
chapter.

The android:orientation can be:

 horizontal: Views are arranged from left to right.


 vertical: Views are arranged from top to bottom.

Other layout-related attributes include:

 android:layout_gravity: This attribute is used with a UI element to control where the element
is arranged within its parent. For example, the following attribute centers the UI element
horizontally within the parent ViewGroup:
android:layout_gravity="center_horizontal"

 Padding is the space, measured in density-independent pixels, between the edges of the UI
element and the element's content, as shown in the figure below.

In the figure above: (1) Padding is the space between the edges of the TextView (dashed
lines) and the content of the TextView (solid line). Padding is not the same as margin, which
is the space from the edge of the View to its parent.
The size of a View includes its padding. The following are commonly used padding
attributes:

 android:padding: Sets the padding of all four edges.


 android:paddingTop: Sets the padding of the top edge.
 android:paddingBottom: Sets the padding of the bottom edge.
 android:paddingLeft: Sets the padding of the left edge.
 android:paddingRight: Sets the padding of the right edge.
 android:paddingStart: Sets the padding of the start of the view, in pixels. Used in place of
the padding attributes listed above, especially with views that are long and narrow.
 android:paddingEnd: Sets the padding of the end edge of the view, in pixels. Used along
with android:paddingStart.

Tip: To see all of the XML attributes for a LinearLayout, see the Summary section of
the LinearLayout class definition. Other root layouts, such
as RelativeLayout and AbsoluteLayout, also list their XML attributes in the Summary
sections.

RelativeLayout Positioning

Another useful Viewgroup for layout is RelativeLayout, which you can use to position
child View elements relative to each other or to the parent. The attributes you can use
with RelativeLayout include the following:

 android:layout_toLeftOf: Positions the right edge of this View to the left of


another View (identified by its ID).
 android:layout_toRightOf: Positions the left edge of this View to the right of
another View (identified by its ID).
 android:layout_centerHorizontal: Centers this View horizontally within its parent.
 android:layout_centerVertical: Centers this View vertically within its parent.
 android:layout_alignParentTop: Positions the top edge of this View to match the top edge
of the parent.
 android:layout_alignParentBottom: Positions the bottom edge of this View to match the
bottom edge of the parent.

For a complete list of attributes for View and View subclass elements in a RelativeLayout,
see RelativeLayout.LayoutParams.

Style-related attributes

You specify style attributes to customize the appearance of a View.


A View that doesn't have style attributes, such as android:textColor, android:textSize,
and android:background, takes on the styles defined in the app's theme.
The following are style-related attributes used in lesson on using the layout editor:

 android:background: Specifies a color or drawable resource to use as the background.


 android:text: Specifies text to display in the view.
 android:textColor: Specifies the text color.
 android:textSize: Specifies the text size.
 android:textStyle: Specifies the text style, such as bold.

Resource files

Resource files are a way of separating static values from code so that you don't have to
change the code itself to change the values. You can store all the strings, layouts,
dimensions, colors, styles, and menu text separately in resource files.

Resource files are stored in folders located in the res folder when viewing the Project >
Android pane. These folders include:

 drawable: For images and icons


 layout: For layout resource files
 menu: For menu items
 mipmap: For pre-calculated, optimized collections of app icons used by the Launcher
 values: For colors, dimensions, strings, and styles (theme attributes)

The syntax to reference a resource in an XML layout is as follows:

@package_name:resource_type/resource_name

 package_name is the name of the package in which the resource is located. The package
name is not required when you reference resources that are stored in the res folder of your
project, because these resources are from the same package.
 resource_type is the R subclass for the resource type. See Resource Types for more about
the resource types and how to reference them.
 resource_name is either the resource filename without the extension, or
the android:name attribute value in the XML element.

For example, the following XML layout statement sets the android:text attribute to
a string resource:
android:text="@string/button_label_toast"

 No package_name is included, because the resource is stored in the strings.xml file in the
project.
 The resource_type is string.
 The resource_name is button_label_toast.

Another example: this XML layout statement sets the android:background attribute to
a color resource, and since the resource is defined in the project (in the colors.xml file),
the package_name is not specified:
android:background="@color/colorPrimary"
In the following example, the XML layout statement sets the android:textColor attribute to
a color resource. However, the resource is not defined in the project but supplied by
Android, so you need to specify the package_name, which is android, followed by a colon:
android:textColor="@android:color/white"
Tip: For more about accessing resources from code, see Accessing Resources. For Android
color constants, see the Android standard R.color resources.

Values resource files

Keeping values such as strings and colors in separate resource files makes it easier to manage
them, especially if you use them more than once in your layouts.

For example, it is essential to keep strings in a separate resource file for translating and
localizing your app, so that you can create a string resource file for each language without
changing your code. Resource files for images, colors, dimensions, and other attributes are
handy for developing an app for different device screen sizes and orientations.

Strings

String resources are located in the strings.xml file (inside res > values in the Project >
Android pane). You can edit this file directly by opening it in the editor pane:
<resources>
<string name="app_name">Hello Toast</string>
<string name="button_label_count">Count</string>
<string name="button_label_toast">Toast</string>
<string name="count_initial_value">0</string>
</resources>
The name (for example, button_label_count) is the resource name you use in your XML
code, as in the following attribute:
android:text="@string/button_label_count"
The string value of this name is the word (Count) enclosed within
the <string></string> tags. (You don't use quotation marks unless the quotation marks are
part of the string value.)

Extracting strings to resources


You should also extract hard-coded strings in an XML layout file to string resources.

To extract a hard-coded string in an XML layout, follow these steps, as shown in the figure
above:

1. Click the hard-coded string and press Alt-Enter in Windows, or Option-Return in Mac OS
X.
2. Select Extract string resource.
3. Edit the Resource name for the string value.

You can then use the resource name in your XML code. Use the
expression "@string/resource_name" (including quotation marks) to refer to the string
resource:
android:text="@string/button_label_count"

Colors

Color resources are located in the colors.xml file (inside res > values in the Project >
Android pane). You can edit this file directly in the editor pane:
<resources>
<color name="colorPrimary">#3F51B5</color>
<color name="colorPrimaryDark">#303F9F</color>
<color name="colorAccent">#FF4081</color>
<color name="myBackgroundColor">#FFF043</color>
</resources>
The name (for example, colorPrimary) is the resource name you use in your XML code:
android:textColor="@color/colorPrimary"
The color value of this name is the hexadecimal color value (#3F51B5) enclosed within
the <color></color> tags. The hexadecimal value specifies red, green, and blue (RGB)
values. The value always begins with a pound (#) character, followed by the Alpha-Red-
Green-Blue information. For example, the hexadecimal value for black is #000000, while
the hexadecimal value for a variant of sky blue is #559fe3. Base color values are listed in
the Color class documentation.
The colorPrimary color is one of the predefined base colors and is used for the app bar. In a
production app, you could, for example, customize this to fit your brand. Using the base
colors for other UI elements creates a uniform UI.
Tip: For the Material Design specification for Android colors, see Style and Using the
Material Theme. For common color hexadecimal values, see Color Hex Color Codes. For
Android color constants, see the Android standard R.color resources.

You can see a small block of the color choice in the left margin next to the color resource
declaration in colors.xml, and also in the left margin next to the attribute that uses the
resource name in the layout XML file.

Tip: To see the color in a popup, turn on the Autopopup documentation feature.
Select Preferences > Editor > General > Code Completion, and select the "Autopopup
documentation in (ms)" option. You can then hover your cursor over a color resource name
to see the color.

Dimensions
To make dimensions easier to manage, you should separate the dimensions from your code,
especially if you need to adjust your layout for devices with different screen densities.
Keeping dimensions separate from code also makes it easy to have consistent sizing for UI
elements, and to change the size of multiple elements by changing one dimension resource.

Dimension resources are located in the dimens.xml file (inside res > values in the Project
> Android pane). The dimens.xml file can actually be a folder holding more than
one dimens.xml file—one for each device screen resolution. You can edit
each dimens.xml file directly:
<resources>
<!-- Default screen margins, per the Android Design guidelines. -->
<dimen name="activity_horizontal_margin">16dp</dimen>
<dimen name="activity_vertical_margin">16dp</dimen>
<dimen name="my_view_width">300dp</dimen>
<dimen name="count_text_size">200sp</dimen>
<dimen name="counter_height">300dp</dimen>
</resources>
The name (for example, activity_horizontal_margin) is the resource name you use in the
XML code:
android:paddingLeft="@dimen/activity_horizontal_margin"
The value of this name is the measurement (16dp) enclosed within
the <dimen></dimen> tags.
You can extract dimensions in the same way as strings:

1. Click the hard-coded dimension, and press Alt-Enter in Windows, or press Option-
Return in Mac OS X.
2. Select Extract dimension resource.
3. Edit the Resource name for the dimension value.

Density-independent pixels (dp) are independent of screen resolution. For


example, 10px (10 fixed pixels) look a lot smaller on a higher resolution screen, but Android
scales 10dp (10 device-independent pixels) to look right on different resolution screens.
Text sizes can also be set to look right on different resolution screens using scaled-pixel (sp)
sizes.
Tip: For more information about dp and sp units, see Supporting Different Densities.

Styles

A style is a resource that specifies common attributes such as height, padding, font color,
font size, background color. Styles are meant for attributes that modify the look of the view.

Styles are defined in the styles.xml file (inside res > values in the Project > Android pane).
You can edit this file directly. Styles are covered in a later chapter, along with the Material
Design Specification.

Other resource files


Android Studio defines other resources that are covered in other chapters:

 Images and icons: The drawable folder provides icon and image resources. If your app does
not have a drawable folder, you can manually create it inside the res folder. For more
information about drawable resources, see Drawable Resources in the App Resources
section of the Android Developer's Guide.
 Optimized icons: The mipmap folder typically contains pre-calculated, optimized
collections of app icons used by the Launcher. Expand the folder to see that versions of
icons are stored as resources for different screen densities.
 Menus: You can use an XML resource file to define menu items and store them in your
project in the menu folder. Menus are described in a later chapter.

Responding to View clicks

A click event occurs when the user taps or clicks a clickable View, such as
a Button, ImageButton, ImageView, or FloatingActionButton. When such an event occurs,
your code performs an action. In order to make this pattern work, you have to:

 Write a Java method that performs the specific action you want the app to do when this event
occurs. This method is typically referred to as an event handler.
 Associate this event-handler method to the View, so that the method executes when the
event occurs.

The onClick attribute

Android Studio provides a shortcut for setting up a clickable View, and for associating an
event handler with the View: use the android:onClick attribute in the XML layout.
For example, the following XML attribute sets a Button to be clickable, and
sets showToast() as the event handler:
<Button
android:id="@+id/button_toast"
android:onClick="showToast"
When the user taps the button_toast Button, the button's android:onClick attribute calls
the showToast() method. In order to work with the android:onClick attribute,
the showToast() method must be public and return void. To know which View called the
method, the showToast() method must require a view parameter.
Android Studio provides a shortcut for creating an event handler stub (a placeholder for a
method that you can fill in later) in the code for the Activity associated with the XML layout.
Follow these steps:

1. Inside the XML layout file (such as activity_main.xml), click the method name in
the android:onClick attribute statement (showToast in the XML snippet above).
2. Press Alt-Enter in Windows or Option-Return in Mac OS X, and select Create onClick
event handler.
3. Select the Activity associated with the layout file (such as MainActivity) and click OK.
Android Studio creates a placeholder method stub in MainActivity.java as shown below.

public void showToast(View view) {


// Do something in response to the button click.
}

Updating a View
To update a View, for example to replace the text in a TextView, your code must first
instantiate an object from the View. Your code can then update the object, which updates
the screen.
To refer to the View in your code, use the findViewById() method of the View class, which
looks for a View based on the resource id. For example, the following statement
sets mShowCount to be the TextView in the layout with the resource id show_count:
mShowCount = (TextView) findViewById(R.id.show_count);
From this point on, your code can use mShowCount to represent the TextView, so that when
you update mShowCount, the TextView is updated.
For example, when the following Button with the android:onClick attribute is
tapped, onClick calls the countUp() method:
android:onClick="countUp"
You can implement countUp() to increment the count, convert the count to a string, and set
the string as the text for the mShowCount object:
public void countUp(View view) {
mCount++;
if (mShowCount != null)
mShowCount.setText(Integer.toString(mCount));
}
Since you had already associated mShowCount with the TextView for displaying the count,
the mShowCount.setText() method updates the TextView on the screen.

1.3: Text and scrolling views

Contents:

 TextView
 Scrolling views
 Related practical
 Learn more

This chapter describes one of the most often used View subclasses in apps: the TextView,
which shows textual content on the screen. A TextView can be used to show a message, a
response from a database, or even entire magazine-style articles that users can scroll. This
chapter also shows how you can create a scrolling view of text and other elements.

TextView

One View subclass you may use often is the TextView class, which displays text on the
screen. You can use TextView for a View of any size, from a single character or word to a
full screen of text. You can add a resource id to the TextView in the layout, and control how
the text appears using attributes in the layout file.
You can refer to a TextView in your Java code by using its resource id in order to update
the text or its attributes from your code. If you want to allow users to edit the text,
use EditText, a subclass of TextView that allows text input and editing. You learn all
about EditText in another lesson.

TextView attributes
You can use XML attributes for a TextView to control:
 Where the TextView is positioned in a layout (like any other view)
 How the TextView itself appears, such as with a background color
 What the text looks like within the TextView, such as the initial text and its style, size, and
color

For example, to set the width, height, and initial text value of the view:

<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Hello World!"
<!-- more attributes -->
/>
You can extract the text string into a string resource (perhaps called hello_world) that's
easier to maintain for multiple-language versions of the app, or if you need to change the
string in the future. After extracting the string, use the string resource name with @string/ to
specify the text:
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/hello_world"
<!-- more attributes -->
/>
In addition to android:layout_width and android:layout_height (which are required for
a TextView), the most often used attributes with TextView are the following:

 android:text: Set the text to display.


 android:textColor: Set the color of the text. You can set the attribute to a color value, a
predefined resource, or a theme. Color resources and themes are described in other chapters.
 android:textAppearance: The appearance of the text, including its color, typeface, style, and
size. You set this attribute to a predefined style resource or theme that already defines these
values.
 android:textSize: Set the text size (if not already set by android:textAppearance).
Use sp (scaled-pixel) sizes such as 20sp or 14.5sp, or set the attribute to a predefined
resource or theme.
 android:textStyle: Set the text style (if not already set by android:textAppearance).
Use normal, bold, italic, or bold|italic.
 android:typeface: Set the text typeface (if not already set by android:textAppearance).
Use normal, sans, serif, or monospace.
 android:lineSpacingExtra: Set extra spacing between lines of text. Use sp (scaled-pixel) or
dp (device-independent pixel) sizes, or set the attribute to a predefined resource or theme.
 android:autoLink: Controls whether links such as URLs and email addresses are
automatically found and converted to clickable (touchable) links.

Use one of the following with android:autoLink:

 none: Match no patterns (default).


 web: Match web URLs.
 email: Match email addresses.
 phone: Match phone numbers.
 map: Match map addresses.
 all: Match all patterns (equivalent to web|email|phone|map).

For example, to set the attribute to match web URLs, use android:autoLink="web".

Using embedded tags in text

In an app that accesses magazine or newspaper articles, the articles that appear would
probably come from an online source or might be saved in advance in a database on the
device. You can also create text as a single long string in the strings.xml resource.

In either case, the text may contain embedded HTML tags or other text formatting codes.
To properly display in a text view, text must be formatted following these rules:

 Enter \n to represent the end of a line, and another \n to represent a blank line. You need to
add end-of-line characters to keep paragraphs from running into each other.
 If you have an apostrophe (') in your text, you must escape it by preceding it with a backslash
(\'). If you have a double-quote in your text, you must also escape it (\"). You must also
escape any other non-ASCII characters. See the "Formatting and Styling" section of String
Resources for more details.
 Enter the HTML and </b> tags around words that should be in bold.
 Enter the HTML and </i> tags around words that should be in italics. Note, however, that if
you use curled apostrophes within an italic phrase, you should replace them with straight
apostrophes.
 You can combine bold and italics by combining the tags, as in ... words...</i></b>. Other
HTML tags are ignored.
 To create a long string of text in the strings.xml file, enclose the entire text within <string
name="your_string_name"></string> (your_string_name is the name you provide the string
resource, such as article_text).
 As you enter or paste text in the strings.xml file, the text lines don't wrap around to the next
line—they extend beyond the right margin. This is the correct behavior—each new line of
text starting at the left margin represents an entire paragraph.

Tip: If you want to see the text wrapped in strings.xml, you can press Return to enter hard
line endings, or format the text first in a text editor with hard line endings. The endings will
not be displayed on the screen.

Referring to a TextView in code

To refer to a TextView in your Java code, use its resource id. For example, to update
a TextView with new text, you would:

1. Find the TextView and assign it to a variable. You use the findViewById() method of
the View class, and refer to the view you want to find using this format:
2. R.id.view_id

In which view_id is the resource identifier for the view (such as show_count) :

mShowCount = (TextView) findViewById(R.id.show_count);


3. After retrieving the View as a TextView member variable, you can then set the text to new
text (in this case, mCount_text) using the setText() method of the TextView class:
4. mShowCount.setText(mCount_text);

Scrolling views

If the information you want to show in your app is larger than the device's display, you can
create a scrolling view that the user can scroll vertically by swiping up or down, or
horizontally by swiping right or left.

You would typically use a scrolling view for news stories, articles, or any lengthy text that
doesn't completely fit on the display. You can also use a scrolling view to combine views
(such as a TextView and a Button) within a scrolling view.

Creating a layout with a ScrollView


The ScrollView class provides the layout for a vertical scrolling view. (For horizontal
scrolling, you would use HorizontalScrollView.) ScrollView is a subclass of FrameLayout,
which means that you can place only one View as a child within it; that child contains the

entire contents to scroll.


Even though you can place only one child View inside a ScrollView, the child View can be
a ViewGroup with a hierarchy of child View elements, such as a LinearLayout. A good
choice for a View within a ScrollView is a LinearLayout that is arranged in a vertical

orientation.

ScrollView and performance

All of the contents of a ScrollView (such as a ViewGroup with View elements) occupy
memory and the view hierarchy even if portions are not displayed on screen. This
makes ScrollView useful for smoothly scrolling pages of free-form text, because the text is
already in memory. However, a ScrollView with a ViewGroup with View elements can use
up a lot of memory, which can affect the performance of the rest of your app.
Using nested instances of LinearLayout can also lead to an excessively deep view hierarchy,
which can slow down performance. Nesting several instances of LinearLayout that use
the android:layout_weight attribute can be especially expensive as each child View needs to
be measured twice. Consider using flatter layouts such as RelativeLayout or GridLayout to
improve performance.
Complex layouts with ScrollView may suffer performance issues, especially with images.
We recommend that you not use images within a ScrollView. To display long lists of items,
or images, consider using a RecyclerView, which is covered in another lesson.

ScrollView with a TextView

To display a scrollable magazine article on the screen, you might use a RelativeLayout that
includes a separate TextView for the article heading, another for the article subheading, and
a third TextView for the scrolling article text (see the figure below), set within a ScrollView.
The only part of the screen that would scroll would be the ScrollView with the article text.

ScrollView with a LinearLayout

A ScrollView can contain only one child View; however, that View can be
a ViewGroup that contains several View elements, such as LinearLayout. You
can nest a ViewGroup such as LinearLayout within the ScrollView, thereby scrolling
everything that is inside the LinearLayout.
For example, if you want the subheading of an article to scroll along with the article even if
they are separate TextView elements, add a LinearLayout to the ScrollView as a single
child View as shown in the figure below, and then move the TextView subheading and
article elements into the LinearLayout. The user scrolls the entire LinearLayout which
includes the subheading and the article.

When adding a LinearLayout inside a ScrollView, use match_parent for


the LinearLayout android:layout_width attribute to match the width of the
parent ScrollView, and use wrap_content for
the LinearLayout android:layout_height attribute to make it only large enough to enclose its
contents.
Since ScrollView only supports vertical scrolling, you must set
the LinearLayout orientation attribute to vertical (android:orientation="vertical"), so that
the entire LinearLayout will scroll vertically. For example, the following XML layout
scrolls the article TextView along with the article_subheading TextView:
<ScrollView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_below="@id/article_heading">

<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical">

<TextView
android:id="@+id/article_subheading"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:padding="@dimen/padding_regular"
android:text="@string/article_subtitle"
android:textAppearance=
"@android:style/TextAppearance.DeviceDefault" />
<TextView
android:id="@+id/article"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:autoLink="web"
android:lineSpacingExtra="@dimen/line_spacing"
android:padding="@dimen/padding_regular"
android:text="@string/article_text" />
</LinearLayout>

</ScrollView>

2.1: Activities and intents

Contents:

 Introduction
 About Activity
 Creating an Activity
 About Intent
 Starting an Activity with an explicit Intent
 Passing data from one Activity to another
 Getting data back from an Activity
 Activity navigation
 Related practical
 Learn more

Introduction

In this chapter you learn about the Activity class, the major building block of your app's user
interface (UI). You also learn about using an Intent to communicate from one activity to
another.

About activities

An activity represents a single screen in your app with an interface the user can interact with.
For example, an email app might have one activity that shows a list of new emails, another
activity to compose an email, and another activity for reading individual messages. Your
app is probably a collection of activities that you create yourself, or that you reuse from
other apps.
Although the activities in your app work with each other to form a cohesive user experience,
each activity is independent of the others. This enables your app to start an activity in another
app, and it enables other apps to start activities in your app (if your app allows this). For
example, a messaging app could start an activity in a camera app to take a picture, then start
an activity in an email app to let the user share the picture in email.

Typically, one Activity in an app is specified as the "main" activity, for


example MainActivity. The user sees the main activity when they launch the app for the first
time. Each activity can start other activities to perform different actions.
Each time a new activity starts, the previous activity is stopped, but the system preserves
the activity in a stack (the "back stack"). When the user is done with the current activity and
presses the Back button, the activity is popped from the stack and destroyed, and the
previous activity resumes.

When an activity is stopped because a new activity starts, the first activity is notified by way
of the activity lifecycle callback methods. The activity lifecycle is the set of states
an Activity can be in: when the activity is first created, when it's stopped or resumed, and
when the system destroys it. You learn more about the activity lifecycle in a later chapter.

Creating an Activity

To implement an Activity in your app, do the following:

 Create an Activity Java class.


 Implement a basic UI for the Activity in an XML layout file.
 Declare the new Activity in the AndroidManifest.xml file.

When you create a new project for your app, or add a new Activity to your app by
choosing File > New > Activity, the template automatically performs the steps listed above.
Create the Activity

When you create a new project in Android Studio and choose the Backwards Compatibility
(AppCompat) option, the MainActivity is, by default, a subclass of
the AppCompatActivity class. The AppCompatActivity class lets you use up-to-date
Android app features such as the app bar and Material Design, while still enabling your app
to be compatible with devices running older versions of Android.
Here is a skeleton subclass of AppCompatActivity:
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
}
The first task for you in your Activity subclass is to implement the
standard Activity lifecycle callback methods (such as onCreate()) to handle the state
changes for your Activity. These state changes include things such as when the Activity is
created, stopped, resumed, or destroyed. You learn more about the Activity lifecycle and
lifecycle callbacks in a different chapter.
The one required callback your app must implement is the onCreate() method. The system
calls this method when it creates your Activity, and all the essential components of
your Activity should be initialized here. Most importantly, the onCreate() method
calls setContentView() to create the primary layout for the Activity.
You typically define the UI for your Activity in one or more XML layout files. When
the setContentView() method is called with the path to a layout file, the system creates all
the initial views from the specified layout and adds them to your Activity. This is often
referred to as inflating the layout.
You may often also want to implement the onPause() method in your Activity. The system
calls this method as the first indication that the user is leaving your Activity (though it does
not always mean that the Activity is being destroyed). This is usually where you should
commit any changes that should be persisted beyond the current user session (because the
user might not come back). You learn more about onPause() and all the other lifecycle
callbacks in a later chapter.
In addition to lifecycle callbacks, you may also implement methods in your Activity to
handle other behavior such as user input or button clicks.

Implement the activity's UI

The UI for an activity is provided by a hierarchy of View elements, which controls a


particular space within the activity window and can respond to user interaction.
The most common way to define a UI using View elements is with an XML layout file stored
as part of your app's resources. Defining your layout in XML enables you to maintain the
design of your UI separately from the source code that defines the activity behavior.
You can also create new View elements directly in your activity code by inserting
new View objects into a ViewGroup, and then passing the
root ViewGroup to setContentView(). After your layout has been inflated—regardless of its
source—you can add more View elements anywhere in the View hierarchy.

Declare the Activity in AndroidManifest.xml


Each Activity in your app must be declared in the AndroidManifest.xml file with
the <activity> element, inside the <application> section. When you create a new project or
add a new Activity to your project in Android Studio, the AndroidManifest.xml file is
created or updated to include skeleton declarations for each Activity. Here's the declaration
for MainActivity:
<activity android:name=".MainActivity" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
The <activity> element includes a number of attributes to define properties of
the Activity such as its label, icon, or theme. The only required attribute is android:name,
which specifies the class name for the Activity (such as MainActivity). See
the <activity> element reference for more information on Activity declarations.
The <activity> element can also include declarations for Intent filters. The Intent filters
specify the kind of Intent your Activity will accept.
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
Intent filters must include at least one <action> element, and can also include
a <category> and optional <data>. The MainActivity for your app needs an Intent filter that
defines the "main" action and the "launcher" category so that the system can launch your
app. Android Studio creates this Intent filter for the MainActivity in your project.
The <action> element specifies that this is the "main" entry point to the app.
The <category> element specifies that this Activity should be listed in the system's app
launcher (to allow users to launch this Activity).
Each Activity in your app can also declare Intent filters, but only your MainActivity should
include the "main" action. You learn more about how to use an
implicit Intent and Intent filters in a later section.

Add another Activity to your project

The MainActivity for your app and its associated layout file is supplied by
an Activity template in Android Studio such as Empty Activity or Basic Activity. You can
add a new Activity to your project by choosing File > New > Activity. Choose
the Activity template you want to use, or open the Gallery to see all the available templates.

When you choose an Activity template, you see the same set of screens for creating the new
activity that you did when you created the project. Android Studio provides three things for
each new activity in your app:

 A Java file for the new Activity with a skeleton class definition and onCreate() method. The
new Activity, like MainActivity, is a subclass of AppCompatActivity.
 An XML file containing the layout for the new activity. Note that
the setContentView() method in the Activity class inflates this new layout.
 An additional <activity> element in the AndroidManifest.xml file that specifies the new
activity. The second Activity definition does not include any Intent filters. If you plan to use
this activity only within your app (and not enable that activity to be started by any other
app), you do not need to add filters.

About intents

Each activity is started or activated with an Intent, which is a message object that makes a
request to the Android runtime to start an activity or other app component in your app or in
some other app.
When your app is first started from the device home screen, the Android runtime sends
an Intent to your app to start your app's main activity (the one defined with the MAIN action
and the LAUNCHER category in the AndroidManifest.xml file). To start another activity in
your app, or to request that some other activity available on the device perform an action,
you build your own intent and call the startActivity() method to send the intent.
In addition to starting an activity, an intent can also be used to pass data between one activity
and another. When you create an intent to start a new activity, you can include information
about the data you want that new activity to operate on. So, for example, an
email Activity that displays a list of messages can send an Intent to the Activity that
displays that message. The display activity needs data about the message to display, and you
can include that data in the intent.
In this chapter you learn about using intents with activities, but intents can also be used to
start services or broadcast receivers. You learn how to use those app components in another
practical.

Intent types

Intents can be explicit or implicit:

 Explicit intent: You specify the receiving activity (or other component) using the activity's
fully qualified class name. You use explicit intents to start components in your own app (for
example, to move between screens in the UI), because you already know the package and
class name of that component.
 Implicit intent: You do not specify a specific activity or other component to receive the
intent. Instead, you declare a general action to perform, and the Android system matches
your request to an activity or other component that can handle the requested action. You
learn more about using implicit intents in another practical.

Intent objects and fields

For an explicit Intent, the key fields include the following:

 The Activity class (for an explicit Intent). This is the class name of the Activity or other
component that should receive the Intent; for example, com.example.SampleActivity.class.
Use the Intent constructor or the setComponent(), setComponentName(),
or setClassName() methods to specify the class.
 The Intent data. The Intent data field contains a reference to the data you want the
receiving Activity to operate on as a Uri object.
 Intent extras. These are key-value pairs that carry information the
receiving Activity requires to accomplish the requested action.
 Intent flags. These are additional bits of metadata, defined by the Intent class. The flags may
instruct the Android system how to launch an Activity or how to treat it after it's launched.

For an implicit Intent, you may need to also define the Intent action and category. You learn
more about Intent actions and categories in another chapter.

Starting an Activity with an explicit Intent

To start a specific Activity from another Activity, use an explicit Intent and
the startActivity() method. An explicit Intent includes the fully qualified class name for
the Activity or other component in the Intent object. All the other Intent fields are optional,
and null by default.
For example, if you want to start the ShowMessageActivity to show a specific message in
an email app, use code like this:
Intent messageIntent = new Intent(this, ShowMessageActivity.class);
startActivity(messageIntent);
The intent constructor takes two arguments for an explicit Intent:

 An application context. In this example, the Activity class provides the context (this).
 The specific component to start (ShowMessageActivity.class).

Use the startActivity() method with the new Intent object as the only argument.
The startActivity() method sends the Intent to the Android system, which launches
the ShowMessageActivity class on behalf of your app. The new Activity appears on the
screen, and the originating Activity is paused.
The started Activity remains on the screen until the user taps the Back button on the device,
at which time that Activity closes and is reclaimed by the system, and the
originating Activity is resumed. You can also manually close the started Activity in
response to a user action (such as a Button click) with the finish() method:
public void closeActivity (View view) {
finish();
}

Passing data from one Activity to another

In addition to simply starting one Activity from another Activity, you also use
an Intent to pass information from one Activity to another. The Intent object you use to start
an Activity can include Intent data (the URI of an object to act on), or Intent extras, which
are bits of additional data the Activity might need.
In the first (sending) Activity, you:

1. Create the Intent object.


2. Put data or extras into that Intent.
3. Start the new Activity with startActivity().

In the second (receiving) Activity, you:

1. Get the Intent object the Activity was started with.


2. Retrieve the data or extras from the Intent object.

When to use Intent data or Intent extras


You can use either Intent data or Intent extras to pass data from one Activity to another.
There are several key differences between data and extras that determine which you should
use.
The Intent data can hold only one piece of information: a URI representing the location of
the data you want to operate on. That URI could be a web page URL (http://), a telephone
number (tel://), a geographic location (geo://) or any other custom URI you define.
Use the Intent data field:

 When you only have one piece of information you need to send to the started Activity.
 When that information is a data location that can be represented by a URI.

Intent extras are for any other arbitrary data you want to pass to the
started Activity. Intent extras are stored in a Bundle object as key and value pairs.
A Bundle is a map, optimized for Android, in which a key is a string, and a value can be any
primitive or object type (objects must implement the Parcelable interface). To put data into
the Intent extras you can use any of the Intent class putExtra() methods, or create your
own Bundle and put it into the Intent with putExtras().
Use the Intent extras:

 If you want to pass more than one piece of information to the started Activity.
 If any of the information you want to pass is not expressible by a URI.

Intent data and extras are not exclusive; you can use data for a URI and extras for any
additional information the started Activity needs to process the data in that URI.

Add data to the Intent

To add data to an explicit Intent from the originating Activity, create the Intent object as
you did before:
Intent messageIntent = new Intent(this, ShowMessageActivity.class);
Use the setData() method with a Uri object to add that URI to the Intent. Some examples of
using setData() with URIs:
// A web page URL
messageIntent.setData(Uri.parse("http://www.google.com"));
// a Sample file URI
messageIntent.setData(Uri.fromFile(new File("/sdcard/sample.jpg")));
// A sample content: URI for your app's data model
messageIntent.setData(Uri.parse("content://mysample.provider/data"));
// Custom URI
messageIntent.setData(Uri.parse("custom:" + dataID + buttonId));
Keep in mind that the data field can only contain a single URI; if you call setData() multiple
times only the last value is used. Use Intent extras to include additional information
(including URIs.)
After you've added the data, you can start the Activity with the Intent as usual:
startActivity(messageIntent);

Add extras to the Intent

To add Intent extras to an explicit Intent from the originating Activity:

1. Determine the keys to use for the information you want to put into the extras, or define your
own. Each piece of information needs its own unique key.
2. Use the putExtra() methods to add your key/value pairs to the Intent extras. Optionally you
can create a Bundle object, add your data to the Bundle, and then add the Bundle to
the Intent.

The Intent class includes extra keys you can use, defined as constants that begin with the
word EXTRA_. For example, you could use Intent.EXTRA_EMAIL to indicate an array of
email addresses (as strings), or Intent.EXTRA_REFERRER to specify information about
the originating Activity that sent the Intent.
You can also define your own Intent extra keys. Conventionally you define Intent extra keys
as static variables with names that begin with EXTRA_. To guarantee that the key is unique,
the string value for the key itself should be prefixed with your app's fully qualified class
name. For example:
public final static String EXTRA_MESSAGE =
"com.example.mysampleapp.MESSAGE";
public final static String EXTRA_POSITION_X = "com.example.mysampleapp.X";
public final static String EXTRA_POSITION_Y = "com.example.mysampleapp.Y";
Create an Intent object (if one does not already exist):
Intent messageIntent = new Intent(this, ShowMessageActivity.class);
Use a putExtra() method with a key to put data into the Intent extras. The Intent class
defines many putExtra() methods for different kinds of data:
messageIntent.putExtra(EXTRA_MESSAGE, "this is my message");
messageIntent.putExtra(EXTRA_POSITION_X, 100);
messageIntent.putExtra(EXTRA_POSITION_Y, 500);
Alternately, you can create a new Bundle and populate that Bundle with
your Intent extras. Bundle defines many "put" methods for different kinds of primitive data
as well as objects that implement Android's Parcelable interface or Java's Serializable.
Bundle extras = new Bundle();
extras.putString(EXTRA_MESSAGE, "this is my message");
extras.putInt(EXTRA_POSITION_X, 100);
extras.putInt(EXTRA_POSITION_Y, 500);
After you've populated the Bundle, add it to the Intent with the putExtras() method (note the
"s" in Extras):
messageIntent.putExtras(extras);
Start the Activity with the Intent as usual:
startActivity(messageIntent);

Retrieve the data from the Intent in the started Activity

When you start an Activity with an Intent, the started Activity has access to the Intent and
the data it contains.
To retrieve the Intent the Activity (or other component) was started with, use
the getIntent() method:
Intent intent = getIntent();
Use getData() to get the URI from that Intent:
Uri locationUri = intent.getData();
To get the extras out of the Intent, you need to know the keys for the key/value pairs. You
can use the standard Intent extras if you used those, or you can use the keys you defined in
the originating Activity (if they were defined as public.)
Use one of the getExtra() methods to extract extra data out of the Intent object:
String message = intent.getStringExtra(MainActivity.EXTRA_MESSAGE);
int positionX = intent.getIntExtra(MainActivity.EXTRA_POSITION_X);
int positionY = intent.getIntExtra(MainActivity.EXTRA_POSITION_Y);
Or you can get the entire extras Bundle from the Intent and extract the values with the
various Bundle methods:
Bundle extras = intent.getExtras();
String message = extras.getString(MainActivity.EXTRA_MESSAGE);

Getting data back from an Activity


When you start an Activity with an Intent, the originating Activity is paused, and the
new Activity remains on the screen until the user clicks the Back button, or you call
the finish() method in a click handler or other function that ends the user's involvement with
this Activity.
Sometimes when you send data to an Activity with an Intent, you would like to also get data
back from that Intent. For example, you might start a photo gallery Activity that lets the user
pick a photo. In this case your original Activity needs to receive information about the photo
the user chose back from the launched Activity.
To launch a new Activity and get a result back, do the following steps in your
originating Activity:

1. Instead of launching the Activity with startActivity(), call startActivityForResult() with


the Intent and a request code.
2. Create a new Intent in the launched Activity and add the return data to that Intent.
3. Implement onActivityResult() in the originating Activity to process the returned data.

You learn about each of these steps in the following sections.

Use startActivityForResult() to launch the Activity


To get data back from a launched Activity, start that Activity with
the startActivityForResult() method instead of startActivity().
startActivityForResult(messageIntent, TEXT_REQUEST);
The startActivityForResult() method, like startActivity(), takes an Intent argument that
contains information about the Activity to be launched and any data to send to that Activity.
The startActivityForResult() method, however, also needs a request code.
The request code is an integer that identifies the request and can be used to differentiate
between results when you process the return data. For example, if you launch one Activity to
take a photo and another to pick a photo from a gallery, you need different request codes to
identify which request the returned data belongs to.
Conventionally you define request codes as static integer variables with names that
include REQUEST. Use a different integer for each code. For example:
public static final int PHOTO_REQUEST = 1;
public static final int PHOTO_PICK_REQUEST = 2;
public static final int TEXT_REQUEST = 3;

Return a response from the launched Activity


The response data from the launched Activity back to the originating Activity is sent in
an Intent, either in the data or the extras. You construct this return Intent and put the data
into it in much the same way you do for the sending Intent. Typically your
launched Activity will have an onClick() or other user input callback method in which you
process the user's action and close the Activity. This is also where you construct the
response.
To return data from the launched Activity, create a new empty Intent object.
Intent returnIntent = new Intent();
Note: To avoid confusing sent data with returned data, use a new Intent object rather than
reusing the original sending Intent object.
A return result Intent does not need a class or component name to end up in the right place.
The Android system directs the response back to the originating Activity for you.
Add data or extras to the Intent the same way you did with the original Intent. You may need
to define keys for the return Intent extras at the start of your class.
public final static String EXTRA_RETURN_MESSAGE =
"com.example.mysampleapp.RETURN_MESSAGE";
Then put your return data into the Intent as usual. In the following, the return message is
an Intent extra with the key EXTRA_RETURN_MESSAGE.
messageIntent.putExtra(EXTRA_RETURN_MESSAGE, mMessage);
Use the setResult() method with a response code and the Intent with the response data:
setResult(RESULT_OK,replyIntent);
The response codes are defined by the Activity class, and can be

 RESULT_OK: The request was successful.


 RESULT_CANCELED: The user canceled the operation.
 RESULT_FIRST_USER: For defining your own result codes.

You use the result code in the originating Activity.


Finally, call finish() to close the Activity and resume the originating Activity:
finish();

Read response data in onActivityResult()


Now that the launched Activity has sent data back to the originating Activity with an Intent,
that first Activity must handle that data. To handle returned data in the originating Activity,
implement the onActivityResult() callback method. Here is a simple example.
public void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == TEXT_REQUEST) {
if (resultCode == RESULT_OK) {
String reply =
data.getStringExtra(SecondActivity.EXTRA_RETURN_MESSAGE);
// process data
}
}
}
The three arguments to onActivityResult() contain all the information you need to handle
the return data.

 Request code: The request code you set when you launched
the Activity with startActivityForResult(). If you launch a different Activity to accomplish
different operations, use this code to identify the specific data you're getting back.
 Result code: the result code set in the launched Activity, usually one
of RESULT_OK or RESULT_CANCELED.
 Intent data: the Intent that contains the data returned from the launch Activity.

The example method shown above shows the typical logic for handling the request and
response codes. The first test is for the TEXT_REQUEST request, and that the result was
successful. Inside the body of those tests you extract the return information out of the Intent.
Use getData() to get the Intent data, or getExtra() to retrieve values out of the Intent extras
with a specific key.

Activity navigation
Any app of any complexity that you build will include more than one Activity. As your users
move around your app and from one Activity to another, consistent navigation becomes
more important to the app's user experience. Few things frustrate users more than basic
navigation that behaves in inconsistent and unexpected ways. Thoughtfully designing your
app's navigation will make using your app predictable and reliable for your users.
Android system supports two different forms of navigation strategies for your app.

 Back (temporal) navigation, provided by the device Back button, and the back stack.
 Up (ancestral) navigation, provided by you as an option in the app bar.

Back navigation, tasks, and the back stack

Back navigation allows your users to return to the previous Activity by tapping the device
back button . Back navigation is also called temporal navigation because the back
button navigates the history of recently viewed screens, in reverse chronological order.
The back stack is the set of each Activity that the user has visited and that can be returned
to by the user with the back button. Each time a new Activity starts, it is pushed onto the
back stack and takes user focus. The previous Activity is stopped but is still available in the
back stack. The back stack operates on a "last in, first out" mechanism, so when the user is
done with the current Activity and presses the Back button, that Activity is popped from the
stack (and destroyed) and the previous Activity resumes.
Because an app can start an Activity both inside and outside a single app, the back stack
contains each Activity that has been launched by the user in reverse order. Each time the
user presses the Back button, each Activity in the stack is popped off to reveal the previous
one, until the user returns to the Home screen.

Android provides a back stack for each task. A task is an organizing concept for
each Activity the user interacts with when performing an operation, whether they are inside
your app or across multiple apps. Most tasks start from the Android home screen, and
tapping an app icon starts a task (and a new back stack) for that app. If the user uses an app
for a while, taps home, and starts a new app, that new app launches in its own task and has
its own back stack. If the user returns to the first app, that first task's back stack returns.
Navigating with the Back button returns only to the Activity in the current task, not for all
tasks running on the device. Android enables the user to navigate between tasks with the
overview or recent tasks screen, accessible with the square button on lower right corner of

the device .
In most cases you don't have to worry about managing either tasks or the back stack for your
app—the system keeps track of these things for you, and the back button is always available
on the device.

There may, however, be times where you may want to override the default behavior for tasks
or for the back stack. For example, if your screen contains an embedded web browser where
users can navigate between web pages, you may wish to use the browser's default back
behavior when users press the device's Back button, rather than returning to the
previous Activity. You may also need to change the default behavior for your app in other
special cases such as with notifications or widgets, where an Activity deep within your app
may be launched as its own task, with no back stack at all. You learn more about managing
tasks and the back stack in a later section.

Up navigation
Up navigation, sometimes referred to as ancestral or logical navigation, is used to navigate
within an app based on the explicit hierarchical relationships between screens. With Up
navigation, each Activity is arranged in a hierarchy, and each "child" Activity shows a left-
facing arrow in the app bar that returns the user to the "parent" Activity. The
topmost Activity in the hierarchy is usually MainActivity, and the user cannot go up from

there.
For instance, if the main Activity in an email app is a list of all messages, selecting a
message launches a second Activity to display that single email. In this case the
message Activity would provide an Up button that returns to the list of messages.
The behavior of the Up button is defined by you in each Activity based on how you design
your app's navigation. In many cases, Up and Back navigation may provide the same
behavior: to just return to the previous Activity. For example, a Settings Activity may be
available from any Activity in your app, so "up" is the same as back—just return the user to
their previous place in the hierarchy.
Providing Up behavior for your app is optional, but a good design practice, to provide
consistent navigation for your app.

Implement Up navigation with a parent Activity

With the standard template projects in Android Studio, it's straightforward to implement Up
navigation. If one Activity is a child of another Activity in your app's Activity hierarchy,
specify the parent of that other Activity in the AndroidManifest.xml file.
Beginning in Android 4.1 (API level 16), declare the logical parent of each Activity by
specifying the android:parentActivityName attribute in the <activity> element. To support
older versions of Android, include <meta-data> information to define the
parent Activity explicitly. Use both methods to be backwards-compatible with all versions
of Android.
The following are the skeleton definitions in AndroidManifest.xml for both a main
(parent) Activity (MainActivity) and a second (child) Activity (SecondActivity):
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<!-- The main activity (it has no parent activity) -->
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />

<category android:name="android.intent.category.LAUNCHER" />


</intent-filter>
</activity>
<!-- The child activity) -->
<activity android:name=".SecondActivity"
android:label = "Second Activity"
android:parentActivityName=".MainActivity">
<meta-data
android:name="android.support.PARENT_ACTIVITY"
android:value="com.example.android.twoactivities.MainActivity" />
</activity>
</application>

3.1: The Android Studio debugger


Contents:

 About debugging
 Running the debugger
 Using breakpoints
 Stepping through code
 Viewing execution stack frames
 Inspecting and modifying variables
 Setting watches
 Evaluating expressions
 More tools for debugging
 Trace logging and the Android manifest
 Related practical
 Learn more

In this chapter you'll learn about debugging your apps in Android Studio.

About debugging
Debugging is the process of finding and fixing errors (bugs) or unexpected behavior in your
code. All code has bugs, from incorrect behavior in your app, to behavior that excessively
consumes memory or network resources, to actual app freezing or crashing.

Bugs can result for many reasons:

 Errors in your design or implementation


 Android framework limitations (or bugs)
 Missing requirements or assumptions for how the app should work
 Device limitations (or bugs)

Use the debugging, testing, and profiling capabilities in Android Studio to help you
reproduce, find, and resolve all of these problems. Those capabilities include:

 The Logcat pane for log messages


 The Debugger pane for viewing frames, threads, and variables
 Debug mode for running apps with breakpoints
 Test frameworks such as JUnit or Espresso
 Dalvik Debug Monitor Server (DDMS), to track resource usage

In this chapter you learn how to debug your app with the Android Studio debugger, set and
view breakpoints, step through your code, and examine variables.

Running the debugger

Running an app in debug mode is similar to running the app. You can either run an app in
debug mode, or attach the debugger to an already-running app.

Run your app in debug mode

To start debugging, click Debug in the toolbar. Android Studio builds an APK, signs it
with a debug key, installs it on your selected device, then runs it and opens the Debug pane
with the Debugger and Console tabs.

The figure above shows the Debug pane with the Debugger and Console tabs.
The Debugger tab is selected, showing the Debugger pane with the following features:

1. Frames tab: Click to show the Frames pane with the current execution stack frames for a
given thread. The execution stack shows each class and method that have been called in your
app and in the Android runtime, with the most recent method at the top.
Click the Threads tab to replace the Frames pane with the Threads pane.

2. Watches button: Click to show the Watches pane within the Variables pane, which shows
the values for any variable watches you have set. Watches allow you to keep track of a
specific variable in your program, and see how that variable changes as your program runs.

3. Variables pane: Shows the variables in the current scope and their values. Each variable in
this pane has an expand icon to expand the list of object properties for the variable. Try
expanding a variable to explore its properties.

Debug a running app


If your app is already running on a device or emulator, start debugging that app with these
steps:

1. Select Run > Attach debugger to Android process or click the Attach icon in the
toolbar.
2. In the Choose Process dialog, select the process to which you want to attach the debugger.

By default, the debugger shows the device and app process for the current project, as well
as any connected hardware devices or virtual devices on your computer. Check the Show
all processes option to show all processes on all devices.

3. Click OK. The Debug pane appears as before.

Resume or stop debugging


To resume executing an app after debugging it, select Run > Resume Program or click the
Resume icon.

To stop debugging your app, select Run > Stop or click the Stop icon in the toolbar.

Using breakpoints
Android Studio supports several types of breakpoints that trigger different debugging
actions. The most common type is a breakpoint that pauses the execution of your app at a
specified line of code. While paused, you can examine variables, evaluate expressions, then
continue execution line by line to determine the causes of runtime errors.

You can set a breakpoint on any executable line of code.

Add breakpoints
To add a breakpoint to a line in your code, use these steps:

1. Locate the line of code where you want to pause execution.


2. Click in the left gutter of the editor pane at that line, next to the line numbers. A red dot
appears at that line, indicating a breakpoint. The red dot includes a check mark if the app is
already running in debug mode.
As an alternative, you can choose Run > Toggle Line Breakpoint or press Control-
F8 (Command-F8 on a Mac) to set or clear a breakpoint at a line.
If your app is already running, you don't need to update it to add the breakpoint.

If you click a breakpoint by mistake, you can undo it by clicking the breakpoint. If you
clicked a line of code that is not executable, the red dot includes an "x" and a warning
appears that the line of code is not executable.

When your code execution reaches the breakpoint, Android Studio pauses execution of your
app. You can then use the tools in the Debug pane to view the state of the app and debug
that app as it runs.

View and configure breakpoints

To view all the breakpoints you've set and configure breakpoint settings, click t he View
Breakpoints icon on the left edge of the Debug pane. The Breakpoints window
appears.

In this window all the breakpoints you have set appear in the left pane, and you can enable
or disable each breakpoint with the checkboxes. If a breakpoint is disabled, Android Studio
does not pause your app when execution reaches that breakpoint.

Select a breakpoint from the list to configure its settings. You can configure a breakpoint to
be disabled at first and have the system enable it after a different breakpoint is encountered.
You can also configure whether a breakpoint should be disabled after it has been reached.

To set a breakpoint for any exception, select Exception Breakpoints in the list of
breakpoints.

Disable (mute) all breakpoints

Disabling a breakpoint enables you to temporarily "mute" that breakpoint without removing
it from your code. If you remove a breakpoint altogether you also lose any conditions or
other features you created for that breakpoint, so disabling it can be a better choice.
To mute all breakpoints, click the Mute Breakpoints icon . Click the icon again to
enable (unmute) all breakpoints.

Use conditional breakpoints

Conditional breakpoints are breakpoints that only stop execution of your app if the test in
the condition is true. To define a test for a conditional breakpoint, use these steps:

1. Right-click (or Control-click) a breakpoint, and enter a test in the Condition field.

The test you enter in this field can be any Java expression as long as it returns a boolean
value. You can use variable names from your app as part of the expression.

You can also use the Breakpoints window to enter a breakpoint condition.

2. Run your app in debug mode. Execution of your app stops at the conditional breakpoint, if
the condition evaluates to true.

Stepping through code

After your app's execution has stopped because a breakpoint has been reached, you can
execute your code from that point one line at a time with the Step Over, Step Into, and Step
Out functions.
To use any of the step functions:

1. Begin debugging your app. Pause the execution of your app with a breakpoint.

Your app's execution stops, and the Debugger pane shows the current state of the app. The
current line is highlighted in your code.

2. Click the Step Over icon , select Run > Step Over, or press F8. Step Over executes
the next line of the code in the current class and method, executing all of the method calls
on that line and remaining in the same file.

3. Click the Step Into icon , select Run > Step Into, or press F7. Step Into jumps into
the execution of a method call on the current line (as compared to just executing that method
and remaining on the same line). The Frames pane (which you'll learn about in the next
section) updates to show the new stack frame (the new method). If the method call is
contained in another class, the file for that class is opened and the current line in that file is
highlighted. You can continue stepping over lines in this new method call, or step deeper
into other methods.
4. Click the Step Out icon , select Run > Step Out, or press Shift-F8. Step Out finishes
executing the current method and returns to the point where that method was called.
5. To resume normal execution of the app, select Run > Resume Program or click the
Resume icon.

Viewing execution stack frames

The Frames pane of the Debug pane allows you to inspect the execution stack and the
specific frame that caused the current breakpoint to be reached.

The execution stack shows all the classes and methods (frames) that are being executed up
to this point in the app, in reverse order (most recent frame first). As execution of a particular
frame finishes, that frame is popped from the stack and execution returns to the next frame.

Clicking a line for a frame in the Frames pane opens the associated source in the editor and
highlights the line where that frame was initially executed.
The Variables and Watches panes also update to reflect the state of the execution
environment when that frame was last entered.

Inspecting and modifying variables

The Variables pane of the Debugger pane allows you to inspect the variables available at
the current stack frame when the system stops your app on a breakpoint. Variables that hold
objects or collections such as arrays can be expanded to view their components.

The Variables pane also allows you to evaluate expressions on the fly using static methods
or variables available within the selected frame.
If the Variables pane is not visible, click the Restore Variables View icon .

To modify variables in your app as it runs:

1. Right-click (or Control-click) any variable in the Variables pane, and select Set Value.
You can also press F2.
2. Enter a new value for the variable, and press Return.

The value you enter must be of the appropriate type for that variable, or Android Studio
returns a "type mismatch" error.

Setting watches

The Watches pane provides similar functionality to the Variables pane except that
expressions added to the Watches pane persist between debugging sessions. Add watches
for variables and fields that you access frequently or that provide state that is helpful for the
current debugging session.

To use watches:

1. Begin debugging your app.


2. Click the Show Watches icon . The Watches pane appears next to the Variables pane.
3. In the Watches pane, click the plus (+) button. In the text box that appears, type the name
of the variable or expression you want to watch and then press Enter.

Remove an item from the Watches list by selecting the item and then clicking the minus (–
) button.

Change the order of the elements in the Watches pane list by selecting an item and then
clicking the up or down icons.

Evaluating expressions

Use Evaluate Expression to explore the state of variables and objects in your app, including
calling methods on those objects. To evaluate an expression:

1. Click the Evaluate Expression icon , or select Run > Evaluate Expression.
The Evaluate Code Fragment window appears. You can also right-click on any variable
and choose Evaluate Expression.

2. Enter any Java expression into the top field of the Evaluate Code Fragment window, and
click Evaluate.

The Result field shows the result of that expression. Note that the result you get from
evaluating an expression is based on the app's current state. Depending on the values of the
variables in your app at the time you evaluate expressions, you may get different results.
Changing the values of variables in your expressions also changes the current running state
of the app.

More tools for debugging

Android Studio and the Android SDK include a number of other tools to help you find and
correct issues in your code.

System log (Logcat pane)

As you've learned in another chapter, you can use the Log class to send messages to the
Android system log, and view those messages in Android Studio in the Logcat pane.
To write log messages in your code, use the Log class. Log messages help you understand
the execution flow by collecting the system debug output while you interact with your
app. Log messages can tell you what part of your app failed. For more information about
logging, see Reading and Writing Logs.

Tracing and logging

Analyzing traces allows you to see how much time is spent in certain methods, and which
ones are taking the longest times.

To create the trace files, include the Debug class and call one of
the startMethodTracing() methods. In the call, you specify a base name for the trace files
that the system generates.
To stop tracing, call stopMethodTracing(). These methods start and stop method tracing
across the entire virtual machine. For example, you could call startMethodTracing() in
the onCreate() method of your Activity, and call stopMethodTracing() in the onDestroy()
method of that Activity.

Other tools

 The Android Debug Bridge (ADB) is a command-line tool that lets you communicate with
an emulator instance or connected Android-powered device.
 The Android Profiler provides real-time data for your app's CPU, memory, and network
activity. You can perform sample-based method tracing to time your code execution, capture
heap dumps, view memory allocations, and inspect the details of network-transmitted files.
 The CPU Profiler helps you inspect your app's CPU usage and thread activity in real-time,
and record method traces, so you can optimize and debug your app's code.
 The Network Profiler displays real-time network activity on a timeline, showing data sent
and received, as well as the current number of connections. This lets you examine how and
when your app transfers data, and optimize the underlying code appropriately.
Trace logging and the AndroidManifest.xml file

There are multiple types of debugging available to you beyond setting breakpoints and
stepping through code. You can also use logging and tracing to find issues with your code.
When you have a trace log file (generated by adding tracing code to your app), you can load
the log files in Traceview, which displays the log data in two panes:

 A timeline pane describes when each thread and method started and stopped.
 A profile pane provides a summary of what happened inside a method.

Likewise, to make the app debuggable even when the app is running on a device in user
mode, you can set android:debuggable in the <application> tag of
the AndroidManifest.xml file to "true". By default, the debuggable value is set to "false".
You can create and configure build types in the module-level build.gradle file inside
the android {} block. When you create a new module, Android Studio creates the debug and
release build types for you. Although the debug build type doesn't appear in the build
configuration file, Android Studio configures it with debuggable true. This allows you to
debug the app on secure Android-powered devices and configures APK signing with a
generic debug keystore. If you want to add or change certain settings, add the debug build
type to your configuration.
When you prepare your app for release, you must remove all the extra code in your source
files that you wrote for testing purposes.

In addition to prepping the code itself, there are a few other tasks you need to complete in
order to get your app ready to publish. These include:

 Removing logging statements.


 Remove any calls to show Toast messages.
 Disable debugging in the AndroidManifest.xml file by either
removing android:debuggable attribute from the <application> tag, or
setting android:debuggable to "false".
 Remove all debug tracing calls from your source code files such
as startMethodTracing() and stopMethodTracing().

7.3 Broadcasts

Contents:

 Introduction
 Broadcasts
 Broadcast receivers
 Restricting broadcasts
 Best practices
 Related practical
 Learn more

In this chapter you learn about broadcasts and broadcast receivers. Broadcasts are
messaging components used for communicating across different apps, and also with the
Android system, when an event of interest occurs. Broadcast receivers are the components
in your Android app that listen for these events and respond accordingly.
Broadcasts

Broadcasts are messages that the Android system and Android apps send when events occur
that might affect the functionality of other apps. For example, the Android system sends an
event when the system boots up, when power is connected or disconnected, and when
headphones are connected or disconnected. Your Android app can also broadcast events, for
example when new data is downloaded.

In general, broadcasts are messaging components used for communicating across apps when
events of interest occur. There are two types of broadcasts:

 System broadcasts are delivered by the system.


 Custom broadcasts are delivered by your app.

System broadcasts

A system broadcast is a message that the Android system sends when a system event occurs.
System broadcasts are wrapped in Intent objects. The intent object's action field contains
event details such as android.intent.action.HEADSET_PLUG, which is sent when a wired
headset is connected or disconnected. The intent can also contain more data about the event
in its extra field, for example a boolean extra indicating whether a headset is connected or
disconnected.
Examples:

 When the device boots, the system broadcasts a system Intent with the
action ACTION_BOOT_COMPLETED.
 When the device is disconnected from external power, the system sends a system Intent with
the action field ACTION_POWER_DISCONNECTED.

System broadcasts aren't targeted at specific recipients. Interested apps must register a
component to "listen" for these events. This listening component is called a broadcast
receiver.

As the Android system evolves, significant changes are made to improve the system's
performance. For example, starting from Android 7.0, the system broadcast
actions ACTION_NEW_PICTURE and ACTION_NEW_VIDEO are not supported. Apps
can no longer receive broadcasts about these actions, regardless of the targetSDK version
on the device. This is because device cameras take pictures and record videos frequently, so
sending a system broadcast every time one of these actions occurred would strain a device's
memory and battery.
To get the complete list of broadcast actions that the system can send for a particular SDK
version, check the broadcast_actions.txt file in your SDK folder, at the following
path: Android/sdk/platforms/android-xx/data, where xx is the SDK version.

Custom broadcasts

Custom broadcasts are broadcasts that your app sends out. Use a custom broadcast when
you want your app to take an action without launching an activity. For example, use a custom
broadcast when you want to let other apps know that data has been downloaded to the device
and is available for them to use. More than one broadcast receiver can be registered to
receive your broadcast.
To create a custom broadcast, define a custom Intent action.
Note: When you specify the action for the Intent, use your unique package name (for
example com.example.myproject) to make sure that your intent doesn't conflict with an
intent that is broadcast from a different app or from the Android system.
There are three ways to deliver a custom broadcast:

 For a normal broadcast, pass the intent to sendBroadcast().


 For an ordered broadcast, pass the intent to sendOrderedBroadcast().
 For a local broadcast, pass the intent to LocalBroadcastManager.sendBroadcast().

Normal, ordered, and local broadcasts are described in more detail below.

Normal broadcasts

The sendBroadcast() method sends broadcasts to all the registered receivers at the same
time, in an undefined order. This is called a normal broadcast. A normal broadcast is the
most efficient way to send a broadcast. With normal broadcasts, receivers can't propagate
the results among themselves, and they can't cancel the broadcast.
The following method sends a normal broadcast to all interested broadcast receivers:

public void sendBroadcast() {


Intent intent = new Intent();
intent.setAction("com.example.myproject.ACTION_SHOW_TOAST");
// Set the optional additional information in extra field.
intent.putExtra("data","This is a normal broadcast");
sendBroadcast(intent);
}

Ordered broadcasts

To send a broadcast to one receiver at a time, use the sendOrderedBroadcast() method:

 The android:priority attribute that's specified in the intent filter determines the order in
which the broadcast is sent.
 If more than one receiver with same priority is present, the sending order is random.
 The Intent is propagated from one receiver to the next.
 During its turn, a receiver can update the Intent, or it can cancel the broadcast. (If the
receiver cancels the broadcast, the Intent can't be propagated further.)

For example, the following method sends an ordered broadcast to all interested broadcast
receivers:

public void sendOrderedBroadcast() {


Intent intent = new Intent();

// Set a unique action string prefixed by your app package name.


intent.setAction("com.example.myproject.ACTION_NOTIFY");
// Deliver the Intent.
sendOrderedBroadcast(intent);
}
Local broadcasts

If you don't need to send broadcasts to a different app, use


the LocalBroadcastManager.sendBroadcast() method, which sends broadcasts to receivers
within your app. This method is efficient, because it doesn't involve interprocess
communication. Also, using local broadcasts protects your app against some security issues.
To send a local broadcast:

1. To get an instance of LocalBroadcastManager, call getInstance() and pass in the application


context.
2. Call sendBroadcast() on the instance. Pass in the intent that you want to broadcast.
3. LocalBroadcastManager.getInstance(this).sendBroadcast(customBroadcastIntent);

Broadcast receivers

Broadcast receivers are app components that can register for system events or app events.
When an event occurs, registered broadcast receivers are notified via an Intent. For instance,
if you are implementing a media app and you're interested in knowing when the user
connects or disconnects a headset, register for the ACTION_HEADSET_PLUG intent
action.
Use broadcast receivers to respond to messages that have been broadcast from apps or from
the Android system. To create a broadcast receiver:

1. Define a subclass of the BroadcastReceiver class and implement the onReceive() method.
2. Register the broadcast receiver, either statically or dynamically.

These steps are described below.

Subclass a BroadcastReceiver

To create a broadcast receiver, define a subclass of the BroadcastReceiver class. This


subclass is where Intent objects are delivered if they match the intent filters you register for.
Within your subclass:

 Implement the onReceive() method, which is called when the BroadcastReceiver object
receives an Intent broadcast .
 Inside onReceive(), include any other logic that your broadcast receiver needs.

Example: Create a broadcast receiver

In this example, the myReceiver class a subclass of BroadcastReceiver. If the incoming


broadcast intent has the ACTION_SHOW_TOAST action, the myReceiver class shows a
toast message:
//Subclass of the BroadcastReceiver class.
private class myReceiver extends BroadcastReceiver {
// Override the onReceive method to receive the broadcasts
@Override
public void onReceive(Context context, Intent intent) {
//Check the Intent action and perform the required operation
if (intent.getAction().equals(ACTION_SHOW_TOAST)) {
CharSequence text = "Broadcast Received!";
int duration = Toast.LENGTH_SHORT;

Toast toast = Toast.makeText(context, text, duration);


toast.show();
}
}
}
The onReceive() method is called when your app receives a registered Intent broadcast.
The onReceive() method runs on the main thread unless it is explicitly asked to run on a
different thread in the registerReceiver() method.
The onReceive() method has a timeout of 10 seconds. After 10 seconds, the Android system
considers your receiver to be blocked, and the system might show the user an "application
not responding" error. For this reason, you shouldn't implement long-running operations
in onReceive().
Important: Don't use asynchronous operations to run a long-running operation in
your onReceive() implementation, because once your code returns from onReceive(), the
system considers the BroadcastReceiver component to be finished. If onReceive() started an
asynchronous operation, the system would stop the BroadcastReceiver process before the
asynchronous operation had a chance to complete.
In particular:

 Don't try to show a dialog from within a BroadcastReceiver. Instead, display a notification
using the NotificationManager API.
 Don't try to bind to a service from within a BroadcastReceiver. Instead,
use Context.startService() to send a command to the service.

If you need to perform a long-running operation inside BroadcastReceiver,


use WorkManager to schedule a job. When you schedule a task with WorkManager, the task
is guaranteed to run. WorkManager chooses the appropriate way to run your task, based on
such factors as the device API level and the app state.

Register your broadcast receiver and set intent filters


There are two types of broadcast receivers:

 Static receivers, which you register in the Android manifest file.


 Dynamic receivers, which you register using a context.

Static receivers

Static receivers are also called manifest-declared receivers. To register a static receiver,
include the following attributes inside the <receiver> element in
your AndroidManifest.xml file:
 android:name
The value for this attribute is the fully classified name of the BroadcastReceiver subclass,
including the package name. To use the package name that's specified in the manifest, prefix
the subclass name with a period, for example .AlarmReceiver.
 android:exported (optional)
If this boolean value is set to false, other apps cannot send broadcasts to your receiver. This
attribute is important for security.
 <intent-filter>
Include this nested element to specify the broadcast Intent actions that your broadcast
receiver component is listening for.
The following code snippet shows static registration of a broadcast receiver that listens for
a custom broadcast Intent with the action "ACTION_SHOW_TOAST":

 The receiver's name is the name of the BroadcastReceiver subclass (.AlarmReceiver).


 The receiver is not exported, meaning that no other apps can deliver broadcasts to this app.
 The intent filter checks whether incoming intents include an action
named ACTION_SHOW_TOAST, which is a custom Intent action defined within the app.
 <receiver
 android:name=".AlarmReceiver"
 android:exported="false">
 <intent-filter>
 <action android:name=
 "com.example.myproject.intent.action.ACTION_SHOW_TOAST"/>
 </intent-filter>
 </receiver>

Note: For Android 8.0 (API level 26) and higher, static receivers can't receive most implicit
broadcasts. (Implicit broadcasts are broadcasts that don't target your app specifically.) Even
if you register for these broadcasts in the manifest, the Android system won't deliver them
to your app. However, you can still use a dynamic receiver to register for these broadcasts.

A few broadcasts, such


as ACTION_BOOT_COMPLETED and ACTION_TIMEZONE_CHANGED, are excepted
from this restriction. You can declare them in the manifest, no matter what the target API
level. To learn more, see the complete list of implicit broadcast exceptions.
</div>

Intent filters

An intent filter specifies the types of intents that a component can receive. When the system
receives an Intent as a broadcast, it searches the broadcast receivers based on the values
specified in receivers' intent filters.
To create an intent filter statically, use the <intent-filter> element in the Android manifest.
An <intent-filter> element has one required element, <action>. The Android system
compares the Intent action of the incoming broadcast to the filter
action's android:name strings. If any of the names in the filter match the action name in the
incoming broadcast, the broadcast is sent to your app.
This <intent-filter> listens for a system broadcast that's sent when the device boots up.
Only Intent objects with an action named BOOT_COMPLETED match the filter:
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"/>
</intent-filter>
If no intent filters are specified, the broadcast receiver can only be activated with an explicit
broadcast Intent that specifies the component by name. (This is similar to how explicit
intents are used to launch activities by their class names.)
To learn more about using intent filters, see the Intent resolution section of the Intent guide.

Dynamic receivers

Dynamic receivers are also called context-registered receivers. You register a dynamic
receiver using an application context or an Activity context. A dynamic receiver receives
broadcasts as long as the registering context is valid:

 If you use the application context to register your receiver, your app receives relevant
broadcasts as long as your app is running in either the foreground or the background.
 If you use an Activity context to register your receiver, your app receives relevant
broadcasts until that Activity is destroyed.

To use the context to register your receiver dynamically:

1. Create an IntentFilter and add the Intent actions that you want your app to listen for. You
can add more than one action to the same IntentFilter object.
2. IntentFilter intentFilter = new IntentFilter();
3. filter.addAction(Intent.ACTION_POWER_CONNECTED);
4. filter.addAction(Intent.ACTION_POWER_DISCONNECTED);
When the device power is connected or disconnected, the Android system broadcasts
the Intent.ACTION_POWER_CONNECTED and Intent.ACTION_POWER_DISCONNE
CTED intent actions.
5. Register the receiver by calling registerReceiver() method on the context. Pass in
the BroadcastReceiver object and the IntentFilter object.
6. mReceiver = new AlarmReceiver();
7. this.registerReceiver(mReceiver, intentFilter);
In this example the Activity context (this) is used to register your receiver. So the app will
receive the broadcast as long as the Activity is running.

Local broadcasts

You must register local receivers dynamically, because static registration in the manifest is
not possible for a local broadcasts.

To register a receiver for local broadcasts:

1. Get an instance of LocalBroadcastManager by calling the getInstance() method.


2. Call registerReceiver(), passing in the receiver and an IntentFilter object.
3. LocalBroadcastManager.getInstance(this)
4. .registerReceiver(mReceiver,
5. new IntentFilter(CustomReceiver.ACTION_CUSTOM_BROADCAST));

Unregister the receiver

To save system resources and avoid leaks, unregister dynamic receivers when your app no
longer needs them, or before the Activity or app is destroyed. This is also true for local
broadcast receivers, because they're registered dynamically.
To unregister a normal broadcast receiver:
1. Call unregisterReceiver() and pass in your BroadcastReceiver object:
2. unregisterReceiver(mReceiver);
To unregister a local broadcast receiver:

3. Get an instance of the LocalBroadcastManager.


4. Call LocalBroadcastManager.unregisterReceiver() and pass in
your BroadcastReceiver object:
5. LocalBroadcastManager.getInstance(this)
6. .unregisterReceiver(mReceiver);
Where you call these unregisterReceiver() methods depends on the desired lifecycle of
your BroadcastReceiver object:
7. Sometimes the receiver is only needed when your activity is visible, for example to disable
a network function when the network is not available. In these cases, register the receiver
in onResume() and unregister the receiver in onPause().
8. You can also use the onStart()/onStop() or onCreate()/onDestroy() method pairs, if they are
more appropriate for your use case.

Restricting broadcasts

An unrestricted broadcast can pose a security threat, because any registered receiver can
receive it. For example, if your app uses a normal broadcast to send an implicit Intent that
includes sensitive information, an app that contains malware could receive that broadcast.
Restricting your broadcast is strongly recommended.
Ways to restrict a broadcast:

 If possible, use a LocalBroadcastManager, which keeps the data inside your app, avoiding
any security leaks. You can only use LocalBroadcastManager if you don't need interprocess
communication or communication with other apps.
 Use the setPackage() method and pass in the package name. Your broadcast is restricted to
apps that match the specified package name.
 Enforce access permissions on the sender side, on the receiver side, or both.

To enforce a permission when sending a broadcast:

 Supply a non-null permission argument to sendBroadcast(). Only receivers that request this
permission using the <uses-permission> tag in their AndroidManifest.xml file can receive
the broadcast.

To enforce a permission when receiving a broadcast:

 If you register your receiver dynamically, supply a non-null permission


to registerReceiver().
 If you register your receiver statically, use the android:permission attribute inside
the <receiver> tag in your AndroidManifest.xml.

Best practices

 In your intent actions, prefix String constants with your app's package name. Otherwise, you
may conflict with other apps' intents. The Intent namespace is global.
 Restrict broadcast receivers, as described above.
 Prefer dynamic receivers over static receivers.
 Do not start an Activity from a broadcast receiver—use a notification instead. Starting an
activity from a broadcast receiver causes a bad user experience if more than one receiver is
listening for the same broadcast event.
 Never perform a long running operation in the broadcast receiver's onReceive(Context,
Intent) method, because the method runs on the main UI thread. Consider using
a JobScheduler or a WorkManager instead.

7.4: Services

Contents:

 Introduction
 What is a service?
 Declaring services in the manifest
 Started services
 Bound services
 Service lifecycle
 Foreground services
 Background services and API 26
 Scheduled services
 Learn more

In this chapter you learn about the different types of services, how to use them, and how to
manage their lifecycles within your app.

What is a service?

A service is an app component that performs long-running operations, usually in the


background. Unlike an Activity, a service doesn't provide a user interface (UI). Services are
defined by the Service class or one of its subclasses.
A service can be started, bound, or both:

 A started service is a service that an app component starts by calling startService().


Use started services for tasks that run in the background to perform long-running operations.
Also use started services for tasks that perform work for remote processes.

 A bound service is a service that an app component binds to itself by calling bindService().
Use bound services for tasks that another app component interacts with to perform
interprocess communication (IPC). For example, a bound service might handle network
transactions, perform file I/O, play music, or interact with a database.

A service runs in the main thread of its hosting process—the service doesn't create its own
thread and doesn't run in a separate process unless you specify that it should.

If your service is going to do any CPU-intensive work or blocking operations (such as MP3
playback or networking), create a new thread within the service to do that work. By using a
separate thread, you reduce the risk of the user seeing "application not responding" (ANR)
errors, and the app's main thread can remain dedicated to user interaction with your
activities.
In Android 8.0 (Oreo, API 26) or higher, the system imposes some new restrictions on
running background services when the app itself isn't in the foreground. For details about
these restrictions, see Background services and API 26.

To implement any kind of service in your app, do the following steps:

1. Declare the service in the manifest.


2. Extend a Service class such as IntentService and create implementation code, as described
in Started services and Bound services, below.
3. Manage the service lifecycle.

Declaring services in the manifest

As with activities and other components, you must declare all services in your Android
manifest. To declare a service, add a <service> element as a child of
the <application> element. For example:
<manifest ... >
...
<application ... >
<service android:name="ExampleService"
android:exported="false" />
...
</application>
</manifest>
To block access to a service from other apps, declare the service as private. To do this, set
the android:exported attribute to false. This stops other apps from starting your service, even
when they use an explicit intent.

Started services

How a service starts:

1. An app component such as an Activity calls startService() and passes in an Intent.


The Intent specifies the service and includes any data for the service to use.
2. The system calls the service's onCreate() method and any other appropriate callbacks on the
main thread. It's up to the service to implement these callbacks with the appropriate
behavior, such as creating a secondary thread in which to work.
3. The system calls the service's onStartCommand() method, passing in the Intent supplied by
the client in step 1. (The client in this context is the app component that calls the service.)

Once started, a service can run in the background indefinitely, even if the component that
started it is destroyed. Usually, a started service performs a single operation and does not
return a result to the caller. For example, it might download or upload a file over the network.
When the operation is done, the service should stop itself by calling stopSelf(), or another
component can stop it by calling stopService().
For instance, suppose an Activity needs to save data to an online database.
The Activity starts a companion service by passing an Intent to startService(). The service
receives the intent in onStartCommand(), connects to the internet, and performs the database
transaction. When the transaction is done, the service uses stopSelf() to stop itself and is
destroyed. (This is an example of a service you want to run in a worker thread instead of the
main thread.)
IntentService

Most started services don't need to handle multiple requests simultaneously, and if they did,
it could be a complex and error-prone multi-threading scenario. For these reasons,
the IntentService class is a useful subclass of Service on which to base your service:

 IntentService automatically provides a worker thread to handle your Intent.


 IntentService handles some of the boilerplate code that regular services need (such as
starting and stopping the service).
 IntentService can create a work queue that passes one intent at a time to
your onHandleIntent() implementation, so you don't have to worry about multi-threading.

Note:IntentService is subject to the new restrictions on background services in Android 8.0


(API 26). For this reason, Android Support Library 26.0.0 introduces a
new JobIntentService class, which provides the same functionality as IntentService but uses
jobs instead of services when running on Android 8.0 or higher.
To implement IntentService:

1. Provide a small constructor for the service.


2. Create an implementation of onHandleIntent() to do the work that the client provides.

Here's an example implementation of IntentService:


public class HelloIntentService extends IntentService {
/**
* A constructor is required, and must call the
* super IntentService(String) constructor with a name
* for the worker thread.
*/
public HelloIntentService() {
super("HelloIntentService");
}

/**
* The IntentService calls this method from the default
* worker thread with the intent that started the service.
* When this method returns, IntentService stops the service,
* as appropriate.
*/
@Override
protected void onHandleIntent(Intent intent) {
// Normally we would do some work here, like download a file.
// For our sample, we just sleep for 5 seconds.
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
// Restore interrupt status.
Thread.currentThread().interrupt();
}
}
}
Bound services

A service is "bound" when an app component binds to it by calling bindService(). A bound


service offers a client-server interface that allows components to interact with the service,
send requests, and get results, sometimes using interprocess communication (IPC) to send
and receive information across processes. A bound service runs only as long as another app
component is bound to it. Multiple components can bind to the service at once, but when all
of them unbind, the service is destroyed.
A bound service generally does not allow components to start it by calling startService().

Implementing a bound service

To implement a bound service, define the interface that specifies how a client can
communicate with the service. This interface, which your service returns from
the onBind() callback method, must be an implementation of IBinder.
To retrieve the IBinder interface, a client app component calls bindService(). Once the client
receives the IBinder, the client interacts with the service through that interface.
There are multiple ways to implement a bound service, and the implementation is more
complicated than a started service. For complete details about bound services, see Bound
Services.

Binding to a service

To bind to a service that is declared in the manifest and implemented by an app component,
use bindService() with an explicit Intent.
Caution: Do not use an implicit intent to bind to a service. Doing so is a security hazard,
because you can't be certain what service will respond to your intent, and the user can't see
which service starts. Beginning with Android 5.0 (API level 21), the system throws an
exception if you call bindService() with an implicit Intent .

Service lifecycle

The lifecycle of a service is simpler than the Activity lifecycle. However, it's even more
important that you pay close attention to how your service is created and destroyed. Because
a service has no UI, services can continue to run in the background with no way for the user
to know, even if the user switches to another app. This situation can potentially consume
resources and drain the device battery.
Similar to an Activity, a service has lifecycle callback methods that you can implement to
monitor changes in the service's state and perform work at the appropriate times. The
following skeleton service implementation demonstrates each of the lifecycle methods:
public class ExampleService extends Service {
// indicates how to behave if the service is killed.
int mStartMode;

// interface for clients that bind.


IBinder mBinder;

// indicates whether onRebind should be used


boolean mAllowRebind;

@Override
public void onCreate() {
// The service is being created.
}

@Override
public int onStartCommand(Intent intent,
int flags, int startId) {
// The service is starting, due to a call to startService().
return mStartMode;
}

@Override
public IBinder onBind(Intent intent) {
// A client is binding to the service with bindService().
return mBinder;
}

@Override
public boolean onUnbind(Intent intent) {
// All clients have unbound with unbindService()
return mAllowRebind;
}

@Override
public void onRebind(Intent intent) {
// A client is binding to the service with bindService(),
// after onUnbind() has already been called
}

@Override
public void onDestroy() {
// The service is no longer used and is being destroyed
}
}

Lifecycle of started services vs. bound services

A bound service exists only to serve the app component that's bound to it, so when no more
components are bound to the service, the system destroys it. Bound services don't need to
be explicitly stopped the way started services do (using stopService() or stopSelf()).
The diagram below shows a comparison between the started and bound service lifecycles.

Foreground services

While most services run in the background, some run in the foreground. A foreground
service is a service that the user is aware is running. Although both Activities and Services
can be killed if the system is low on memory, a foreground service has priority over other
resources.

For example, a music player that plays music from a service should be set to run in the
foreground, because the user is aware of its operation. The notification in the status bar
might indicate the current song and allow the user to launch an Activity to interact with the
music player.
To request that a service run in the foreground, call startForeground() instead
of startService(). This method takes two parameters: an integer that uniquely identifies the
notification and the Notification object for the status bar notification. This notification
is ongoing, meaning that it can't be dismissed. It stays in the status bar until the service is
stopped or removed from the foreground.
For example:

Intent notificationIntent = new Intent(this, ExampleActivity.class);


PendingIntent pendingIntent =
PendingIntent.getActivity(this, 0, notificationIntent, 0);

Notification notification =
new Notification.Builder(this, CHANNEL_DEFAULT_IMPORTANCE)
.setContentTitle(getText(R.string.notification_title))
.setContentText(getText(R.string.notification_message))
.setSmallIcon(R.drawable.icon)
.setContentIntent(pendingIntent)
.setTicker(getText(R.string.ticker_text))
.build();

startForeground(ONGOING_NOTIFICATION_ID, notification);
Note: The integer ID for the notification you give to startForeground() must not be 0.
To remove the service from the foreground, call stopForeground(). This method takes a
boolean, indicating whether to remove the status bar notification. This method doesn't stop
the service. However, if you stop the service while it's still running in the foreground, then
the notification is also removed.

Background services and API 26

Services running in the background can consume device resources, potentially using device
battery and resulting in a worse user experience. To mitigate this problem, the system now
applies limitations on started services running in the background, beginning with Android
version 8.0 (Oreo, API 26) or higher. These limitations don't affect foreground services or
bound services.

Here are a few of the specific changes:

 The startService() method now throws an IllegalStateException if an app targeting API 26


tries to use that method in a situation when it isn't permitted to create background services.
 The new startForegroundService() method starts a foreground service from an app
component. The system allows apps to call this method even while the app is in the
background. However, the app must call that service's startForeground() method within five
seconds after the service is created.

While an app is in the foreground, it can create and run both foreground and background
services freely. When an app goes into the background, it has several minutes in which it is
still allowed to create and use services. At the end of that time, the app is considered to be
idle. The system stops the app's background services, just as if the app had called the
services' Service.stopSelf() methods.
Android 8.0 and higher does not allow a background app to create a background service.
Android 8.0 introduces the new method startForegroundService() to start a new service in
the foreground. After the system has created the service, the app has five seconds to call the
service's startForeground() method to show the new service's user-visible notification. If the
app does not call startForeground() within the time limit, the system stops the service and
declares the app to be ANR (Application Not Responding).
For more information on these changes, see Background Execution Limits.

Scheduled services

For API level 21 and higher, you can launch services using the JobScheduler API, and with
the background service restrictions in API 26 this may be a better alternative to services
altogether. To use JobScheduler, you need to register jobs and specify their requirements for
network and timing. The system schedules jobs for execution at appropriate times.
The JobScheduler interface provides many methods to define service-execution conditions.
For details, see the JobScheduler reference.
Note: JobScheduler is only available on devices running API 21+, and is not available in
the support library. If your app targets devices with earlier API levels, look into the
backwards compatible WorkManager, a new API currently in alpha, that allows you to
schedule background tasks that need guaranteed completion (regardless of whether the app
process is around or not). WorkManager provides JobScheduler-like capabilities to API 14+
devices, even those without Google Play Services.

You might also like