अतुल्यकालिक संचालन
इस अध्याय में, हम एस्प्रेसो आइडलिंग रिसोर्सेस का उपयोग करके अतुल्यकालिक संचालन का परीक्षण करना सीखेंगे।
आधुनिक अनुप्रयोग की चुनौतियों में से एक चिकनी उपयोगकर्ता अनुभव प्रदान करना है। सुचारू उपयोगकर्ता अनुभव प्रदान करने के लिए पृष्ठभूमि में बहुत काम शामिल है, यह सुनिश्चित करने के लिए कि आवेदन प्रक्रिया कुछ मिलीसेकंड से अधिक समय नहीं लेती है। बैकग्राउंड टास्क सरल एक से लेकर महंगा और जटिल कार्य दूरस्थ API / डेटाबेस से डेटा लाने का होता है। अतीत में चुनौती का सामना करने के लिए, एक डेवलपर बैकग्राउंड थ्रेड में महंगा और लंबे समय तक चलने वाला काम लिखता था और बैकग्राउंड थ्रेड पूरा होते ही मुख्य UIThread के साथ सिंक हो जाता था।
यदि एक बहु-थ्रेडेड एप्लिकेशन विकसित करना जटिल है, तो इसके लिए परीक्षण मामले लिखना और भी जटिल है। उदाहरण के लिए, डेटाबेस से आवश्यक डेटा लोड होने से पहले हमें एक एडॉप्टर व्यू का परीक्षण नहीं करना चाहिए । यदि डेटा को अलग थ्रेड में लाया जाता है, तो थ्रेड पूरा होने तक प्रतीक्षा करने की आवश्यकता होती है। तो, परीक्षण वातावरण को पृष्ठभूमि थ्रेड और UI थ्रेड के बीच समन्वयित किया जाना चाहिए। एस्प्रेसो बहु-थ्रेडेड एप्लिकेशन के परीक्षण के लिए एक उत्कृष्ट समर्थन प्रदान करता है। एक एप्लिकेशन निम्न तरीकों से धागे का उपयोग करता है और एस्प्रेसो हर परिदृश्य का समर्थन करता है।
उपयोगकर्ता इंटरफ़ेस थ्रेडिंग
यह जटिल यूआई तत्वों के साथ चिकनी उपयोगकर्ता अनुभव प्रदान करने के लिए एंड्रॉइड एसडीके द्वारा आंतरिक रूप से उपयोग किया जाता है। एस्प्रेसो पारदर्शी रूप से इस परिदृश्य का समर्थन करता है और किसी भी कॉन्फ़िगरेशन और विशेष कोडिंग की आवश्यकता नहीं है।
Async कार्य
आधुनिक प्रोग्रामिंग भाषाएं थ्रेड प्रोग्रामिंग की जटिलता के बिना हल्के वजन वाले थ्रेडिंग करने के लिए async प्रोग्रामिंग का समर्थन करती हैं। Async टास्क भी एस्प्रेसो फ्रेमवर्क द्वारा पारदर्शी रूप से समर्थित है।
उपयोगकर्ता थ्रेड
एक डेवलपर डेटाबेस से जटिल या बड़ा डेटा लाने के लिए एक नया थ्रेड शुरू कर सकता है। इस परिदृश्य का समर्थन करने के लिए, एस्प्रेसो आइडलिंग संसाधन अवधारणा प्रदान करता है।
आइए इस अध्याय में आइडलिंग रिसोर्स की अवधारणा को जानें और इसे कैसे करें।
अवलोकन
निष्क्रिय संसाधन की अवधारणा बहुत सरल और सहज है। मूल विचार एक चर (बूलियन मान) बनाने के लिए है जब भी एक लंबी थ्रेडिंग प्रक्रिया को एक अलग थ्रेड में शुरू किया जाता है ताकि यह पता लगाया जा सके कि प्रक्रिया चल रही है या नहीं और इसे परीक्षण वातावरण में पंजीकृत किया गया है। परीक्षण के दौरान, परीक्षण धावक पंजीकृत चर की जांच करेगा, यदि कोई पाया गया है और फिर इसकी चल रही स्थिति का पता लगाएं। अगर रनिंग स्टेटस सही है, तो टेस्ट रनर स्टेटस के गलत होने तक इंतजार करेगा।
एस्प्रेसो रनिंग स्टेटस को बनाए रखने के उद्देश्य से एक इंटरफ़ेस, IdlingResources प्रदान करता है। लागू करने के लिए मुख्य विधि isIdleNow () है। यदि आईलडॉनो () सही है, तो एस्प्रेसो परीक्षण प्रक्रिया को फिर से शुरू करेगा या तब तक प्रतीक्षा करेगा जब तक आइडल नोव () गलत नहीं हो जाता। हमें IdlingResources को लागू करने और व्युत्पन्न वर्ग का उपयोग करने की आवश्यकता है। एस्प्रेसो हमारे कार्यभार को आसान बनाने के लिए कुछ बिल्ट-इन IdlingResources कार्यान्वयन भी प्रदान करता है। वे इस प्रकार हैं,
CountingIdlingResource
यह रनिंग कार्य का एक आंतरिक काउंटर बनाए रखता है। यह वेतन वृद्धि () और वेतन वृद्धि () विधियों को उजागर करता है । वेतन वृद्धि () काउंटर में से एक को बढ़ाता है और गिरावट () काउंटर से एक को हटाता है। isIdleNow () केवल तभी सही होता है जब कोई कार्य सक्रिय न हो।
UriIdlingResource
यह CounintIdlingResource के समान है सिवाय इसके कि नेटवर्क विलंबता लेने के लिए विस्तारित अवधि के लिए काउंटर को शून्य करने की आवश्यकता है।
IdlingThreadPoolExecutor
वर्तमान थ्रेड पूल में संख्या सक्रिय रनिंग कार्य को बनाए रखने के लिए यह थ्रेडपूल एक्ज़ीक्यूटर का एक कस्टम कार्यान्वयन है ।
IdlingScheduledThreadPoolExecutor
यह IdlingThreadPoolExecutor के समान है , लेकिन यह एक शेड्यूल के साथ-साथ शेड्यूल किए गए शेड्यूल के कार्यान्वयन को शेड्यूल करता है।
यदि आवेदन में IdlingResources या कस्टम एक के उपरोक्त कार्यान्वयन में से किसी एक का उपयोग किया जाता है, तो हमें नीचे दिए गए IdlingRegistry वर्ग का उपयोग करते हुए आवेदन का परीक्षण करने से पहले इसे परीक्षण वातावरण में पंजीकृत करना होगा ,
IdlingRegistry.getInstance().register(MyIdlingResource.getIdlingResource());
इसके अलावा, इसे नीचे परीक्षण के पूरा होते ही हटाया जा सकता है -
IdlingRegistry.getInstance().unregister(MyIdlingResource.getIdlingResource());
एस्प्रेसो एक अलग पैकेज में यह कार्यक्षमता प्रदान करता है, और पैकेज को नीचे app.gradle में कॉन्फ़िगर करने की आवश्यकता है।
dependencies {
implementation 'androidx.test.espresso:espresso-idling-resource:3.1.1'
androidTestImplementation "androidx.test.espresso.idling:idlingconcurrent:3.1.1"
}
नमूना आवेदन
आइए हम एक अलग थ्रेड में वेब सेवा से प्राप्त करके फलों को सूचीबद्ध करने के लिए एक सरल एप्लिकेशन बनाएं और फिर आइडलिंग रिसोर्स कॉन्सेप्ट का उपयोग करके इसका परीक्षण करें।
Android स्टूडियो प्रारंभ करें।
पहले चर्चा की गई नई परियोजना बनाएं और इसे नाम दें, MyIdlingFruitApp
माइग्रेट का उपयोग कर AndroidX ढांचे के लिए आवेदन Refactor लिए → माइग्रेट AndroidX विकल्प मेनू।
एप्लिकेशन / बिल्ड.gradle में एस्प्रेसो आइडलिंग रिसोर्स लाइब्रेरी जोड़ें (और इसे सिंक करें) नीचे दिए गए अनुसार,
dependencies {
implementation 'androidx.test.espresso:espresso-idling-resource:3.1.1'
androidTestImplementation "androidx.test.espresso.idling:idlingconcurrent:3.1.1"
}
मुख्य गतिविधि में डिफ़ॉल्ट डिज़ाइन निकालें और सूची दृश्य जोड़ें। की सामग्री activity_main.xml इस प्रकार है,
<?xml version = "1.0" encoding = "utf-8"?>
<RelativeLayout 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"
tools:context = ".MainActivity">
<ListView
android:id = "@+id/listView"
android:layout_width = "wrap_content"
android:layout_height = "wrap_content" />
</RelativeLayout>
सूची दृश्य के आइटम टेम्पलेट को निर्दिष्ट करने के लिए नया लेआउट संसाधन, item.xml जोड़ें । वस्तु की सामग्री। Xml इस प्रकार है,
<?xml version = "1.0" encoding = "utf-8"?>
<TextView xmlns:android = "http://schemas.android.com/apk/res/android"
android:id = "@+id/name"
android:layout_width = "fill_parent"
android:layout_height = "fill_parent"
android:padding = "8dp"
/>
एक नया वर्ग बनाएं - MyIdlingResource । MyIdlingResource का उपयोग हमारे IdlingResource को एक स्थान पर रखने और जब भी आवश्यक हो, करने के लिए किया जाता है। हम अपने उदाहरण में CountingIdlingResource का उपयोग करने जा रहे हैं ।
package com.tutorialspoint.espressosamples.myidlingfruitapp;
import androidx.test.espresso.IdlingResource;
import androidx.test.espresso.idling.CountingIdlingResource;
public class MyIdlingResource {
private static CountingIdlingResource mCountingIdlingResource =
new CountingIdlingResource("my_idling_resource");
public static void increment() {
mCountingIdlingResource.increment();
}
public static void decrement() {
mCountingIdlingResource.decrement();
}
public static IdlingResource getIdlingResource() {
return mCountingIdlingResource;
}
}
वैश्विक वैरिएबल की घोषणा करें, mIdlingResource का प्रकार CountingIdlingResource नीचे के रूप में MainActivity क्लास में
@Nullable
private CountingIdlingResource mIdlingResource = null;
नीचे वेब से फलों की सूची लाने के लिए एक निजी विधि लिखें,
private ArrayList<String> getFruitList(String data) {
ArrayList<String> fruits = new ArrayList<String>();
try {
// Get url from async task and set it into a local variable
URL url = new URL(data);
Log.e("URL", url.toString());
// Create new HTTP connection
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
// Set HTTP connection method as "Get"
conn.setRequestMethod("GET");
// Do a http request and get the response code
int responseCode = conn.getResponseCode();
// check the response code and if success, get response content
if (responseCode == HttpURLConnection.HTTP_OK) {
BufferedReader in = new BufferedReader(new InputStreamReader(conn.getInputStream()));
String line;
StringBuffer response = new StringBuffer();
while ((line = in.readLine()) != null) {
response.append(line);
}
in.close();
JSONArray jsonArray = new JSONArray(response.toString());
Log.e("HTTPResponse", response.toString());
for(int i = 0; i < jsonArray.length(); i++) {
JSONObject jsonObject = jsonArray.getJSONObject(i);
String name = String.valueOf(jsonObject.getString("name"));
fruits.add(name);
}
} else {
throw new IOException("Unable to fetch data from url");
}
conn.disconnect();
} catch (IOException | JSONException e) {
e.printStackTrace();
}
return fruits;
}
OnCreate () विधि में एक नया कार्य बनाएँ वेब से डेटा प्राप्त करने के लिए हमारी getFruitList पद्धति का उपयोग करके एक नया एडेप्टर का निर्माण और इसे सूची दृश्य में सेट करना है। इसके अलावा, यदि हमारा कार्य थ्रेड में पूरा हो जाता है, तो निष्क्रिय संसाधन को घटा दें। कोड निम्नानुसार है,
// Get data
class FruitTask implements Runnable {
ListView listView;
CountingIdlingResource idlingResource;
FruitTask(CountingIdlingResource idlingRes, ListView listView) {
this.listView = listView;
this.idlingResource = idlingRes;
}
public void run() {
//code to do the HTTP request
final ArrayList<String> fruitList = getFruitList("http://<your domain or IP>/fruits.json");
try {
synchronized (this){
runOnUiThread(new Runnable() {
@Override
public void run() {
// Create adapter and set it to list view
final ArrayAdapter adapter = new
ArrayAdapter(MainActivity.this, R.layout.item, fruitList);
ListView listView = (ListView)findViewById(R.id.listView);
listView.setAdapter(adapter);
}
});
}
} catch (Exception e) {
e.printStackTrace();
}
if (!MyIdlingResource.getIdlingResource().isIdleNow()) {
MyIdlingResource.decrement(); // Set app as idle.
}
}
}
यहां, फलों के यूआरएल को http: // <अपने डोमेन या IP / फल.json के रूप में माना जाता है और इसे JSON के रूप में तैयार किया जाता है। सामग्री इस प्रकार है,
[
{
"name":"Apple"
},
{
"name":"Banana"
},
{
"name":"Cherry"
},
{
"name":"Dates"
},
{
"name":"Elderberry"
},
{
"name":"Fig"
},
{
"name":"Grapes"
},
{
"name":"Grapefruit"
},
{
"name":"Guava"
},
{
"name":"Jack fruit"
},
{
"name":"Lemon"
},
{
"name":"Mango"
},
{
"name":"Orange"
},
{
"name":"Papaya"
},
{
"name":"Pears"
},
{
"name":"Peaches"
},
{
"name":"Pineapple"
},
{
"name":"Plums"
},
{
"name":"Raspberry"
},
{
"name":"Strawberry"
},
{
"name":"Watermelon"
}
]
Note - फ़ाइल को अपने स्थानीय वेब सर्वर में रखें और उसका उपयोग करें।
अब, दृश्य ढूंढें , फ्रूटटस्क पास करके एक नया धागा बनाएं , आइडलिंग संसाधन को बढ़ाएं और अंत में कार्य शुरू करें।
// Find list view
ListView listView = (ListView) findViewById(R.id.listView);
Thread fruitTask = new Thread(new FruitTask(this.mIdlingResource, listView));
MyIdlingResource.increment();
fruitTask.start();
MainActivity का पूरा कोड इस प्रकार है,
package com.tutorialspoint.espressosamples.myidlingfruitapp;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.annotation.VisibleForTesting;
import androidx.appcompat.app.AppCompatActivity;
import androidx.test.espresso.idling.CountingIdlingResource;
import android.os.Bundle;
import android.util.Log;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.ArrayList;
public class MainActivity extends AppCompatActivity {
@Nullable
private CountingIdlingResource mIdlingResource = null;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// Get data
class FruitTask implements Runnable {
ListView listView;
CountingIdlingResource idlingResource;
FruitTask(CountingIdlingResource idlingRes, ListView listView) {
this.listView = listView;
this.idlingResource = idlingRes;
}
public void run() {
//code to do the HTTP request
final ArrayList<String> fruitList = getFruitList(
"http://<yourdomain or IP>/fruits.json");
try {
synchronized (this){
runOnUiThread(new Runnable() {
@Override
public void run() {
// Create adapter and set it to list view
final ArrayAdapter adapter = new ArrayAdapter(
MainActivity.this, R.layout.item, fruitList);
ListView listView = (ListView) findViewById(R.id.listView);
listView.setAdapter(adapter);
}
});
}
} catch (Exception e) {
e.printStackTrace();
}
if (!MyIdlingResource.getIdlingResource().isIdleNow()) {
MyIdlingResource.decrement(); // Set app as idle.
}
}
}
// Find list view
ListView listView = (ListView) findViewById(R.id.listView);
Thread fruitTask = new Thread(new FruitTask(this.mIdlingResource, listView));
MyIdlingResource.increment();
fruitTask.start();
}
private ArrayList<String> getFruitList(String data) {
ArrayList<String> fruits = new ArrayList<String>();
try {
// Get url from async task and set it into a local variable
URL url = new URL(data);
Log.e("URL", url.toString());
// Create new HTTP connection
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
// Set HTTP connection method as "Get"
conn.setRequestMethod("GET");
// Do a http request and get the response code
int responseCode = conn.getResponseCode();
// check the response code and if success, get response content
if (responseCode == HttpURLConnection.HTTP_OK) {
BufferedReader in = new BufferedReader(new InputStreamReader(conn.getInputStream()));
String line;
StringBuffer response = new StringBuffer();
while ((line = in.readLine()) != null) {
response.append(line);
}
in.close();
JSONArray jsonArray = new JSONArray(response.toString());
Log.e("HTTPResponse", response.toString());
for(int i = 0; i < jsonArray.length(); i++) {
JSONObject jsonObject = jsonArray.getJSONObject(i);
String name = String.valueOf(jsonObject.getString("name"));
fruits.add(name);
}
} else {
throw new IOException("Unable to fetch data from url");
}
conn.disconnect();
} catch (IOException | JSONException e) {
e.printStackTrace();
}
return fruits;
}
}
अब, एप्लिकेशन मैनिफ़ेस्ट फ़ाइल में नीचे कॉन्फ़िगरेशन जोड़ें, AndroidManifest.xml
<uses-permission android:name = "android.permission.INTERNET" />
अब, उपरोक्त कोड संकलित करें और एप्लिकेशन चलाएं। My Idling Fruit App का स्क्रीनशॉट इस प्रकार है,
अब, ExampleInstrumentedTest.java फ़ाइल खोलें और नीचे दिए गए अनुसार ActivTestRule जोड़ें।
@Rule
public ActivityTestRule<MainActivity> mActivityRule =
new ActivityTestRule<MainActivity>(MainActivity.class);
Also, make sure the test configuration is done in app/build.gradle
dependencies {
testImplementation 'junit:junit:4.12'
androidTestImplementation 'androidx.test:runner:1.1.1'
androidTestImplementation 'androidx.test:rules:1.1.1'
androidTestImplementation 'androidx.test.espresso:espresso-core:3.1.1'
implementation 'androidx.test.espresso:espresso-idling-resource:3.1.1'
androidTestImplementation "androidx.test.espresso.idling:idlingconcurrent:3.1.1"
}
नीचे दिए गए सूची दृश्य का परीक्षण करने के लिए एक नया परीक्षण मामला जोड़ें,
@Before
public void registerIdlingResource() {
IdlingRegistry.getInstance().register(MyIdlingResource.getIdlingResource());
}
@Test
public void contentTest() {
// click a child item
onData(allOf())
.inAdapterView(withId(R.id.listView))
.atPosition(10)
.perform(click());
}
@After
public void unregisterIdlingResource() {
IdlingRegistry.getInstance().unregister(MyIdlingResource.getIdlingResource());
}
अंत में, एंड्रॉइड स्टूडियो के संदर्भ मेनू का उपयोग करके टेस्ट केस को चलाएं और जांचें कि क्या सभी टेस्ट केस सफल हो रहे हैं।