अतुल्यकालिक संचालन

इस अध्याय में, हम एस्प्रेसो आइडलिंग रिसोर्सेस का उपयोग करके अतुल्यकालिक संचालन का परीक्षण करना सीखेंगे।

आधुनिक अनुप्रयोग की चुनौतियों में से एक चिकनी उपयोगकर्ता अनुभव प्रदान करना है। सुचारू उपयोगकर्ता अनुभव प्रदान करने के लिए पृष्ठभूमि में बहुत काम शामिल है, यह सुनिश्चित करने के लिए कि आवेदन प्रक्रिया कुछ मिलीसेकंड से अधिक समय नहीं लेती है। बैकग्राउंड टास्क सरल एक से लेकर महंगा और जटिल कार्य दूरस्थ 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"
/>
  • एक नया वर्ग बनाएं - MyIdlingResourceMyIdlingResource का उपयोग हमारे 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());
}
  • अंत में, एंड्रॉइड स्टूडियो के संदर्भ मेनू का उपयोग करके टेस्ट केस को चलाएं और जांचें कि क्या सभी टेस्ट केस सफल हो रहे हैं।