লোডার

Android 9 (API স্তর 28) হিসাবে লোডারগুলিকে অবমূল্যায়ন করা হয়েছে৷ Activity এবং Fragment লাইফসাইকেল পরিচালনা করার সময় লোডিং ডেটা নিয়ে কাজ করার জন্য প্রস্তাবিত বিকল্পটি হল ViewModel অবজেক্ট এবং LiveData এর সংমিশ্রণ ব্যবহার করা। দেখুন মডেলগুলি লোডারগুলির মতো কনফিগারেশন পরিবর্তনগুলি থেকে বেঁচে থাকে তবে কম বয়লারপ্লেট কোড সহ৷ LiveData ডেটা লোড করার একটি জীবনচক্র-সচেতন উপায় প্রদান করে যা আপনি একাধিক ভিউ মডেলে পুনরায় ব্যবহার করতে পারেন। আপনি MediatorLiveData ব্যবহার করে LiveData একত্রিত করতে পারেন। যেকোন পর্যবেক্ষণযোগ্য প্রশ্ন, যেমন একটি রুম ডাটাবেস থেকে, ডেটাতে পরিবর্তনগুলি পর্যবেক্ষণ করতে ব্যবহার করা যেতে পারে।

ViewModel এবং LiveData এমন পরিস্থিতিতেও উপলব্ধ যেখানে আপনার LoaderManager এ অ্যাক্সেস নেই, যেমন একটি Service । UI লাইফসাইকেল মোকাবেলা না করেই আপনার অ্যাপের প্রয়োজনীয় ডেটা অ্যাক্সেস করার জন্য দুটিকে একসাথে ব্যবহার করা একটি সহজ উপায় প্রদান করে। LiveData সম্পর্কে আরও জানতে, LiveData ওভারভিউ দেখুন। ViewModel সম্পর্কে আরও জানতে, ViewModel ওভারভিউ দেখুন।

লোডার API আপনাকে FragmentActivity বা Fragment প্রদর্শনের জন্য একটি সামগ্রী প্রদানকারী বা অন্যান্য ডেটা উত্স থেকে ডেটা লোড করতে দেয়৷

লোডার ছাড়া, আপনি যে সমস্যার সম্মুখীন হতে পারেন তার মধ্যে নিম্নলিখিতগুলি অন্তর্ভুক্ত রয়েছে:

  • যদি আপনি সরাসরি কার্যকলাপ বা খণ্ডে ডেটা আনেন, তাহলে আপনার ব্যবহারকারীরা UI থ্রেড থেকে সম্ভাব্য ধীরগতির প্রশ্নগুলি সম্পাদন করার কারণে প্রতিক্রিয়াশীলতার অভাবের সম্মুখীন হন।
  • আপনি যদি অন্য থ্রেড থেকে ডেটা আনেন, সম্ভবত AsyncTask এর সাথে, তাহলে আপনি বিভিন্ন কার্যকলাপ বা ফ্র্যাগমেন্ট লাইফসাইকেল ইভেন্ট, যেমন onDestroy() এবং কনফিগারেশন পরিবর্তনের মাধ্যমে সেই থ্রেড এবং UI থ্রেড উভয়ই পরিচালনা করার জন্য দায়ী।

লোডাররা এই সমস্যাগুলি সমাধান করে এবং অন্যান্য সুবিধাগুলি অন্তর্ভুক্ত করে:

  • ধীরগতির বা প্রতিক্রিয়াশীল UI রোধ করতে লোডারগুলি পৃথক থ্রেডে চলে।
  • ঘটনা ঘটলে লোডার কলব্যাক পদ্ধতি প্রদান করে থ্রেড পরিচালনাকে সহজ করে।
  • ডুপ্লিকেট ক্যোয়ারী রোধ করতে লোডারগুলি স্থির থাকে এবং কনফিগারেশন পরিবর্তন জুড়ে ফলাফল ক্যাশে করে।
  • অন্তর্নিহিত ডেটা উৎসের পরিবর্তনের জন্য নিরীক্ষণের জন্য লোডাররা একজন পর্যবেক্ষককে প্রয়োগ করতে পারে। উদাহরণস্বরূপ, CursorLoader স্বয়ংক্রিয়ভাবে একটি ContentObserver নিবন্ধন করে যখন ডেটা পরিবর্তন হয় তখন একটি পুনরায় লোড ট্রিগার করতে।

লোডার API সারাংশ

একটি অ্যাপে লোডার ব্যবহার করার সময় একাধিক ক্লাস এবং ইন্টারফেস জড়িত থাকতে পারে। সেগুলি নিম্নলিখিত সারণীতে সংক্ষিপ্ত করা হয়েছে:

ক্লাস/ইন্টারফেস বর্ণনা
LoaderManager এক বা একাধিক Loader দৃষ্টান্ত পরিচালনার জন্য একটি FragmentActivity বা Fragment সাথে যুক্ত একটি বিমূর্ত শ্রেণী। প্রতি কার্যকলাপ বা খণ্ডের জন্য শুধুমাত্র একটি LoaderManager আছে, কিন্তু একটি LoaderManager একাধিক লোডার পরিচালনা করতে পারে।

একটি LoaderManager পেতে, কার্যকলাপ বা খণ্ড থেকে getSupportLoaderManager() কল করুন।

একটি লোডার থেকে ডেটা লোড করা শুরু করতে, হয় initLoader() অথবা restartLoader() কল করুন। সিস্টেম স্বয়ংক্রিয়ভাবে নির্ধারণ করে যে একই পূর্ণসংখ্যা আইডি সহ একটি লোডার ইতিমধ্যেই বিদ্যমান কিনা এবং হয় একটি নতুন লোডার তৈরি করে বা একটি বিদ্যমান লোডার পুনরায় ব্যবহার করে।

LoaderManager.LoaderCallbacks এই ইন্টারফেসে কলব্যাক পদ্ধতি রয়েছে যা লোডার ঘটনা ঘটলে বলা হয়। ইন্টারফেস তিনটি কলব্যাক পদ্ধতি সংজ্ঞায়িত করে:
  • onCreateLoader(int, Bundle) : সিস্টেমের একটি নতুন লোডার তৈরি করার প্রয়োজন হলে বলা হয়। আপনার কোডে, একটি Loader অবজেক্ট তৈরি করুন এবং এটি সিস্টেমে ফেরত দিন।
  • onLoadFinished(Loader<D>, D) : একটি লোডার যখন ডেটা লোড করা শেষ করে তখন বলা হয়। আপনি সাধারণত আপনার কোডে ব্যবহারকারীর কাছে ডেটা প্রদর্শন করেন।
  • onLoaderReset(Loader<D>) : বলা হয় যখন পূর্বে তৈরি করা লোডার রিসেট করা হচ্ছে, যখন আপনি destroyLoader(int) কল করেন বা যখন কার্যকলাপ বা খণ্ডটি ধ্বংস হয়ে যায়, যার ফলে ডেটা অনুপলব্ধ হয়। আপনার কোডে, লোডারের ডেটার যেকোনো রেফারেন্স মুছে ফেলুন।
আপনার কার্যকলাপ বা খণ্ডটি সাধারণত এই ইন্টারফেসটি প্রয়োগ করে, এবং আপনি যখন initLoader() বা restartLoader() কল করেন তখন এটি নিবন্ধিত হয়।
Loader লোডার ডাটা লোডিং সঞ্চালন. এই ক্লাসটি বিমূর্ত এবং সমস্ত লোডারের জন্য বেস ক্লাস হিসাবে কাজ করে। আপনি সরাসরি Loader সাবক্লাস করতে পারেন বা বাস্তবায়নকে সহজ করতে নিম্নলিখিত অন্তর্নির্মিত সাবক্লাসগুলির মধ্যে একটি ব্যবহার করতে পারেন:
  • AsyncTaskLoader : একটি বিমূর্ত লোডার যা একটি পৃথক থ্রেডে লোড অপারেশন করার জন্য একটি AsyncTask প্রদান করে।
  • CursorLoader : একটি ContentProvider থেকে অসিঙ্ক্রোনাসভাবে ডেটা লোড করার জন্য AsyncTaskLoader এর একটি কংক্রিট সাবক্লাস। এটি একটি ContentResolver জিজ্ঞাসা করে এবং একটি Cursor প্রদান করে।

নিম্নলিখিত বিভাগগুলি আপনাকে দেখায় কিভাবে একটি অ্যাপ্লিকেশনে এই ক্লাস এবং ইন্টারফেসগুলি ব্যবহার করতে হয়।

একটি অ্যাপ্লিকেশনে লোডার ব্যবহার করুন

এই বিভাগে বর্ণনা করা হয়েছে কিভাবে একটি Android অ্যাপ্লিকেশনে লোডার ব্যবহার করতে হয়। একটি অ্যাপ্লিকেশন যা লোডার ব্যবহার করে সাধারণত নিম্নলিখিতগুলি অন্তর্ভুক্ত করে:

  • একটি FragmentActivity বা Fragment
  • LoaderManager এর একটি উদাহরণ।
  • একটি ContentProvider দ্বারা ব্যাক করা ডেটা লোড করার জন্য একটি CursorLoader ৷ বিকল্পভাবে, আপনি অন্য কোনো উৎস থেকে ডেটা লোড করতে Loader বা AsyncTaskLoader এর নিজের সাবক্লাস প্রয়োগ করতে পারেন।
  • LoaderManager.LoaderCallbacks এর জন্য একটি বাস্তবায়ন। এখানেই আপনি নতুন লোডার তৈরি করেন এবং বিদ্যমান লোডারগুলিতে আপনার রেফারেন্স পরিচালনা করেন।
  • লোডারের ডেটা প্রদর্শনের একটি উপায়, যেমন একটি SimpleCursorAdapter
  • একটি ডেটা উৎস, যেমন একটি ContentProvider , যখন একটি CursorLoader ব্যবহার করে।

একটি লোডার শুরু করুন

LoaderManager একটি FragmentActivity বা Fragment মধ্যে এক বা একাধিক Loader দৃষ্টান্ত পরিচালনা করে। কার্যকলাপ বা খণ্ড প্রতি শুধুমাত্র একটি LoaderManager আছে.

আপনি সাধারণত কার্যকলাপের onCreate() পদ্ধতি বা খণ্ডের onCreate() পদ্ধতির মধ্যে একটি Loader শুরু করেন। আপনি নিম্নলিখিত হিসাবে এটি করবেন:

কোটলিন

supportLoaderManager.initLoader(0, null, this)

জাভা

// Prepare the loader.  Either re-connect with an existing one,
// or start a new one.
getSupportLoaderManager().initLoader(0, null, this);

initLoader() পদ্ধতি নিম্নলিখিত পরামিতি গ্রহণ করে:

  • একটি অনন্য আইডি যা লোডারকে শনাক্ত করে। এই উদাহরণে, ID হল 0
  • নির্মাণের সময় লোডারকে সরবরাহ করার জন্য ঐচ্ছিক আর্গুমেন্ট (এই উদাহরণে null )।
  • একটি LoaderManager.LoaderCallbacks বাস্তবায়ন, যা LoaderManager লোডার ইভেন্ট রিপোর্ট করতে কল করে। এই উদাহরণে, স্থানীয় ক্লাস LoaderManager.LoaderCallbacks ইন্টারফেস প্রয়োগ করে, তাই এটি নিজেই একটি রেফারেন্স পাস করে, this

initLoader() কল নিশ্চিত করে যে একটি লোডার চালু এবং সক্রিয়। এর দুটি সম্ভাব্য ফলাফল রয়েছে:

  • যদি আইডি দ্বারা নির্দিষ্ট করা লোডারটি ইতিমধ্যেই বিদ্যমান থাকে তবে সর্বশেষ তৈরি লোডারটি পুনরায় ব্যবহার করা হয়।
  • যদি ID দ্বারা নির্দিষ্ট করা লোডারটি বিদ্যমান না থাকে, initLoader() LoaderManager.LoaderCallbacks পদ্ধতি onCreateLoader() ট্রিগার করে। এখানেই আপনি একটি নতুন লোডারকে তাৎক্ষণিক এবং ফেরত দেওয়ার জন্য কোডটি প্রয়োগ করেন। আরও আলোচনার জন্য, onCreateLoader সম্পর্কে বিভাগটি দেখুন।

উভয় ক্ষেত্রেই, প্রদত্ত LoaderManager.LoaderCallbacks ইমপ্লিমেন্টেশন লোডারের সাথে যুক্ত থাকে এবং লোডার স্টেট পরিবর্তন হলে ডাকা হয়। যদি, এই কলের সময়ে, কলকারী তার শুরু অবস্থায় থাকে এবং অনুরোধ করা লোডারটি ইতিমধ্যেই বিদ্যমান থাকে এবং তার ডেটা তৈরি করে থাকে, তাহলে initLoader() চলাকালীন সিস্টেম অবিলম্বে onLoadFinished() কল করে। এটি ঘটার জন্য আপনাকে প্রস্তুত থাকতে হবে। এই কলব্যাকের আরও আলোচনার জন্য, onLoadFinished সম্পর্কে বিভাগটি দেখুন।

initLoader() পদ্ধতিটি তৈরি করা Loader ফেরত দেয়, তবে আপনাকে এটির একটি রেফারেন্স ক্যাপচার করতে হবে না। LoaderManager স্বয়ংক্রিয়ভাবে লোডারের জীবন পরিচালনা করে। LoaderManager যখন প্রয়োজন হয় তখন লোড হওয়া শুরু করে এবং বন্ধ করে এবং লোডার এবং এর সাথে সম্পর্কিত বিষয়বস্তুর অবস্থা বজায় রাখে।

এটি বোঝায়, আপনি খুব কমই সরাসরি লোডারদের সাথে যোগাযোগ করেন। বিশেষ ঘটনা ঘটলে লোডিং প্রক্রিয়ায় হস্তক্ষেপ করার জন্য আপনি সাধারণত LoaderManager.LoaderCallbacks পদ্ধতি ব্যবহার করেন। এই বিষয়ে আরো আলোচনার জন্য, LoaderManager কলব্যাকস বিভাগটি দেখুন।

একটি লোডার পুনরায় চালু করুন

আপনি যখন initLoader() ব্যবহার করেন, যেমনটি পূর্ববর্তী বিভাগে দেখানো হয়েছে, এটি নির্দিষ্ট আইডি সহ একটি বিদ্যমান লোডার ব্যবহার করে যদি একটি থাকে। যদি না থাকে তবে এটি একটি তৈরি করে। কিন্তু কখনও কখনও আপনি আপনার পুরানো ডেটা বাতিল করে আবার শুরু করতে চান।

আপনার পুরানো ডেটা বাতিল করতে, restartLoader() ব্যবহার করুন। উদাহরণ স্বরূপ, SearchView.OnQueryTextListener এর নিম্নলিখিত বাস্তবায়ন ব্যবহারকারীর ক্যোয়ারী পরিবর্তিত হলে লোডার পুনরায় চালু করে। লোডারটিকে পুনরায় চালু করতে হবে যাতে এটি একটি নতুন কোয়েরি করতে সংশোধিত অনুসন্ধান ফিল্টার ব্যবহার করতে পারে।

কোটলিন

fun onQueryTextChanged(newText: String?): Boolean {
    // Called when the action bar search text has changed.  Update
    // the search filter and restart the loader to do a new query
    // with this filter.
    curFilter = if (newText?.isNotEmpty() == true) newText else null
    supportLoaderManager.restartLoader(0, null, this)
    return true
}

জাভা

public boolean onQueryTextChanged(String newText) {
    // Called when the action bar search text has changed.  Update
    // the search filter, and restart the loader to do a new query
    // with this filter.
    curFilter = !TextUtils.isEmpty(newText) ? newText : null;
    getSupportLoaderManager().restartLoader(0, null, this);
    return true;
}

LoaderManager কলব্যাক ব্যবহার করুন

LoaderManager.LoaderCallbacks হল একটি কলব্যাক ইন্টারফেস যা একটি ক্লায়েন্টকে LoaderManager এর সাথে যোগাযোগ করতে দেয়।

লোডার, বিশেষ করে CursorLoader , বন্ধ হওয়ার পরে তাদের ডেটা ধরে রাখার আশা করা হয়। এটি অ্যাপ্লিকেশানগুলিকে তাদের ডেটা অ্যাক্টিভিটি বা ফ্র্যাগমেন্টের onStop() এবং onStart() পদ্ধতিতে রাখতে দেয়, যাতে ব্যবহারকারীরা যখন একটি অ্যাপ্লিকেশনে ফিরে আসে, তখন তাদের ডেটা পুনরায় লোড হওয়ার জন্য অপেক্ষা করতে না হয়।

আপনি কখন একটি নতুন লোডার তৈরি করতে হবে তা জানতে এবং লোডারের ডেটা ব্যবহার বন্ধ করার সময় হলে অ্যাপ্লিকেশনটিকে জানাতে LoaderManager.LoaderCallbacks পদ্ধতিগুলি ব্যবহার করুন৷

LoaderManager.LoaderCallbacks এই পদ্ধতিগুলি অন্তর্ভুক্ত করে:

  • onCreateLoader() : প্রদত্ত আইডির জন্য একটি নতুন Loader ইনস্ট্যান্টিয়েট করে এবং ফেরত দেয়।
  • onLoadFinished() : পূর্বে তৈরি করা লোডার তার লোড শেষ করলে বলা হয়।
  • onLoaderReset() : বলা হয় যখন পূর্বে তৈরি করা লোডার রিসেট করা হয়, ফলে এটির ডেটা অনুপলব্ধ হয়।

এই পদ্ধতিগুলি নিম্নলিখিত বিভাগে আরও বিশদে বর্ণনা করা হয়েছে।

onCreateLoader

আপনি যখন একটি লোডার অ্যাক্সেস করার চেষ্টা করেন, যেমন initLoader() এর মাধ্যমে, এটি আইডি দ্বারা নির্দিষ্ট করা লোডার বিদ্যমান কিনা তা পরীক্ষা করে। যদি এটি না হয়, এটি LoaderManager.LoaderCallbacks পদ্ধতি onCreateLoader() ট্রিগার করে। এখানেই আপনি একটি নতুন লোডার তৈরি করেন। সাধারণত এটি একটি CursorLoader , কিন্তু আপনি আপনার নিজস্ব Loader সাবক্লাস বাস্তবায়ন করতে পারেন।

নিম্নলিখিত উদাহরণে, onCreateLoader() কলব্যাক পদ্ধতিটি তার কনস্ট্রাক্টর পদ্ধতি ব্যবহার করে একটি CursorLoader তৈরি করে, যার জন্য প্রয়োজন সম্পূর্ণ তথ্যের সেটের প্রয়োজন যা ContentProvider এর কাছে একটি ক্যোয়ারী করার জন্য। বিশেষত, এটি নিম্নলিখিত প্রয়োজন:

  • uri : সামগ্রী পুনরুদ্ধার করার জন্য URI।
  • অভিক্ষেপ : কোন কলামগুলি ফেরত দিতে হবে তার একটি তালিকা। null পাস করা সমস্ত কলাম ফেরত দেয়, যা অদক্ষ।
  • নির্বাচন : একটি ফিল্টার ঘোষণা করে যে কোন সারিগুলি ফিরতে হবে, একটি SQL WHERE ক্লজ হিসাবে ফর্ম্যাট করা হয়েছে (WHERE নিজেই বাদ দিয়ে)। null পাস করা প্রদত্ত URI-এর জন্য সমস্ত সারি প্রদান করে।
  • SelectionArgs : আপনি যদি নির্বাচনের মধ্যে ?s অন্তর্ভুক্ত করেন, তাহলে সেগুলি সিলেকশনআর্গের মানের দ্বারা প্রতিস্থাপিত হবে যে ক্রমে সেগুলি নির্বাচনে প্রদর্শিত হবে। মানগুলি স্ট্রিং হিসাবে আবদ্ধ।
  • sortOrder : সারিগুলিকে কিভাবে অর্ডার করতে হয়, একটি SQL ORDER BY ক্লজ হিসাবে ফর্ম্যাট করা হয় (নিজের দ্বারা ORDER বাদ দিয়ে)। null পাস করা ডিফল্ট সাজানোর ক্রম ব্যবহার করে, যা অক্রমহীন হতে পারে।

কোটলিন

// If non-null, this is the current filter the user has provided.
private var curFilter: String? = null
...
override fun onCreateLoader(id: Int, args: Bundle?): Loader<Cursor> {
    // This is called when a new Loader needs to be created.  This
    // sample only has one Loader, so we don't care about the ID.
    // First, pick the base URI to use depending on whether we are
    // currently filtering.
    val baseUri: Uri = if (curFilter != null) {
        Uri.withAppendedPath(ContactsContract.Contacts.CONTENT_URI, Uri.encode(curFilter))
    } else {
        ContactsContract.Contacts.CONTENT_URI
    }

    // Now create and return a CursorLoader that will take care of
    // creating a Cursor for the data being displayed.
    val select: String = "((${Contacts.DISPLAY_NAME} NOTNULL) AND (" +
            "${Contacts.HAS_PHONE_NUMBER}=1) AND (" +
            "${Contacts.DISPLAY_NAME} != ''))"
    return (activity as? Context)?.let { context ->
        CursorLoader(
                context,
                baseUri,
                CONTACTS_SUMMARY_PROJECTION,
                select,
                null,
                "${Contacts.DISPLAY_NAME} COLLATE LOCALIZED ASC"
        )
    } ?: throw Exception("Activity cannot be null")
}

জাভা

// If non-null, this is the current filter the user has provided.
String curFilter;
...
public Loader<Cursor> onCreateLoader(int id, Bundle args) {
    // This is called when a new Loader needs to be created.  This
    // sample only has one Loader, so we don't care about the ID.
    // First, pick the base URI to use depending on whether we are
    // currently filtering.
    Uri baseUri;
    if (curFilter != null) {
        baseUri = Uri.withAppendedPath(Contacts.CONTENT_FILTER_URI,
                  Uri.encode(curFilter));
    } else {
        baseUri = Contacts.CONTENT_URI;
    }

    // Now create and return a CursorLoader that will take care of
    // creating a Cursor for the data being displayed.
    String select = "((" + Contacts.DISPLAY_NAME + " NOTNULL) AND ("
            + Contacts.HAS_PHONE_NUMBER + "=1) AND ("
            + Contacts.DISPLAY_NAME + " != '' ))";
    return new CursorLoader(getActivity(), baseUri,
            CONTACTS_SUMMARY_PROJECTION, select, null,
            Contacts.DISPLAY_NAME + " COLLATE LOCALIZED ASC");
}

অনলোড সমাপ্ত

এই পদ্ধতিটি বলা হয় যখন পূর্বে তৈরি করা লোডার তার লোড শেষ করে। এই পদ্ধতিটি নিশ্চিত করা হয় যে এই লোডারের জন্য সরবরাহ করা শেষ ডেটা প্রকাশের আগে কল করা হবে। এই মুহুর্তে, পুরানো ডেটার সমস্ত ব্যবহার মুছে ফেলুন, যেহেতু এটি প্রকাশিত হতে চলেছে। কিন্তু নিজেই ডেটা প্রকাশ করবেন না-লোডার এটির মালিক এবং এটির যত্ন নেয়।

লোডার ডেটা প্রকাশ করে যখন এটি জানে যে অ্যাপ্লিকেশনটি আর এটি ব্যবহার করছে না। উদাহরণস্বরূপ, যদি ডেটা একটি CursorLoader থেকে একটি কার্সার হয়, তাহলে এটিতে close() কল করবেন না। যদি কার্সারটি CursorAdapter স্থাপন করা হয়, swapCursor() পদ্ধতিটি ব্যবহার করুন যাতে পুরানো Cursor বন্ধ না হয়, যেমনটি নিম্নলিখিত উদাহরণে দেখানো হয়েছে:

কোটলিন

private lateinit var adapter: SimpleCursorAdapter
...
override fun onLoadFinished(loader: Loader<Cursor>, data: Cursor?) {
    // Swap the new cursor in. (The framework will take care of closing the
    // old cursor once we return.)
    adapter.swapCursor(data)
}

জাভা

// This is the Adapter being used to display the list's data.
SimpleCursorAdapter adapter;
...
public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
    // Swap the new cursor in. (The framework will take care of closing the
    // old cursor once we return.)
    adapter.swapCursor(data);
}

অনলোডার রিসেট

এই পদ্ধতিটি বলা হয় যখন পূর্বে তৈরি করা লোডার রিসেট করা হয়, ফলে এটির ডেটা অনুপলব্ধ হয়। এই কলব্যাক আপনাকে জানতে দেয় যে ডেটা কখন প্রকাশিত হতে চলেছে যাতে আপনি এটি থেকে আপনার রেফারেন্স মুছে ফেলতে পারেন৷

এই বাস্তবায়নটি swapCursor() null এর মান দিয়ে কল করে:

কোটলিন

private lateinit var adapter: SimpleCursorAdapter
...
override fun onLoaderReset(loader: Loader<Cursor>) {
    // This is called when the last Cursor provided to onLoadFinished()
    // above is about to be closed.  We need to make sure we are no
    // longer using it.
    adapter.swapCursor(null)
}

জাভা

// This is the Adapter being used to display the list's data.
SimpleCursorAdapter adapter;
...
public void onLoaderReset(Loader<Cursor> loader) {
    // This is called when the last Cursor provided to onLoadFinished()
    // above is about to be closed.  We need to make sure we are no
    // longer using it.
    adapter.swapCursor(null);
}

উদাহরণ

উদাহরণ স্বরূপ, এখানে একটি Fragment সম্পূর্ণ বাস্তবায়ন রয়েছে যা পরিচিতি সামগ্রী প্রদানকারীর বিরুদ্ধে একটি প্রশ্নের ফলাফল ধারণকারী একটি ListView প্রদর্শন করে। এটি প্রদানকারীর উপর ক্যোয়ারী পরিচালনা করতে একটি CursorLoader ব্যবহার করে।

যেহেতু এই উদাহরণটি একটি ব্যবহারকারীর পরিচিতি অ্যাক্সেস করার জন্য একটি অ্যাপ্লিকেশন থেকে এসেছে, তাই এর ম্যানিফেস্টে অবশ্যই READ_CONTACTS অনুমতি অন্তর্ভুক্ত করতে হবে।

কোটলিন

private val CONTACTS_SUMMARY_PROJECTION: Array<String> = arrayOf(
        Contacts._ID,
        Contacts.DISPLAY_NAME,
        Contacts.CONTACT_STATUS,
        Contacts.CONTACT_PRESENCE,
        Contacts.PHOTO_ID,
        Contacts.LOOKUP_KEY
)


class CursorLoaderListFragment :
        ListFragment(),
        SearchView.OnQueryTextListener,
        LoaderManager.LoaderCallbacks<Cursor> {

    // This is the Adapter being used to display the list's data.
    private lateinit var mAdapter: SimpleCursorAdapter

    // If non-null, this is the current filter the user has provided.
    private var curFilter: String? = null

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        // Prepare the loader.  Either re-connect with an existing one,
        // or start a new one.
        loaderManager.initLoader(0, null, this)
    }

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)

        // Give some text to display if there is no data.  In a real
        // application, this would come from a resource.
        setEmptyText("No phone numbers")

        // We have a menu item to show in action bar.
        setHasOptionsMenu(true)

        // Create an empty adapter we will use to display the loaded data.
        mAdapter = SimpleCursorAdapter(activity,
                android.R.layout.simple_list_item_2,
                null,
                arrayOf(Contacts.DISPLAY_NAME, Contacts.CONTACT_STATUS),
                intArrayOf(android.R.id.text1, android.R.id.text2),
                0
        )
        listAdapter = mAdapter
    }

    override fun onCreateOptionsMenu(menu: Menu, inflater: MenuInflater) {
        // Place an action bar item for searching.
        menu.add("Search").apply {
            setIcon(android.R.drawable.ic_menu_search)
            setShowAsAction(MenuItem.SHOW_AS_ACTION_IF_ROOM)
            actionView = SearchView(activity).apply {
                setOnQueryTextListener(this@CursorLoaderListFragment)
            }
        }
    }

    override fun onQueryTextChange(newText: String?): Boolean {
        // Called when the action bar search text has changed.  Update
        // the search filter, and restart the loader to do a new query
        // with this filter.
        curFilter = if (newText?.isNotEmpty() == true) newText else null
        loaderManager.restartLoader(0, null, this)
        return true
    }

    override fun onQueryTextSubmit(query: String): Boolean {
        // Don't care about this.
        return true
    }

    override fun onListItemClick(l: ListView, v: View, position: Int, id: Long) {
        // Insert desired behavior here.
        Log.i("FragmentComplexList", "Item clicked: $id")
    }

    override fun onCreateLoader(id: Int, args: Bundle?): Loader<Cursor> {
        // This is called when a new Loader needs to be created.  This
        // sample only has one Loader, so we don't care about the ID.
        // First, pick the base URI to use depending on whether we are
        // currently filtering.
        val baseUri: Uri = if (curFilter != null) {
            Uri.withAppendedPath(Contacts.CONTENT_URI, Uri.encode(curFilter))
        } else {
            Contacts.CONTENT_URI
        }

        // Now create and return a CursorLoader that will take care of
        // creating a Cursor for the data being displayed.
        val select: String = "((${Contacts.DISPLAY_NAME} NOTNULL) AND (" +
                "${Contacts.HAS_PHONE_NUMBER}=1) AND (" +
                "${Contacts.DISPLAY_NAME} != ''))"
        return (activity as? Context)?.let { context ->
            CursorLoader(
                    context,
                    baseUri,
                    CONTACTS_SUMMARY_PROJECTION,
                    select,
                    null,
                    "${Contacts.DISPLAY_NAME} COLLATE LOCALIZED ASC"
            )
        } ?: throw Exception("Activity cannot be null")
    }

    override fun onLoadFinished(loader: Loader<Cursor>, data: Cursor) {
        // Swap the new cursor in.  (The framework will take care of closing the
        // old cursor once we return.)
        mAdapter.swapCursor(data)
    }

    override fun onLoaderReset(loader: Loader<Cursor>) {
        // This is called when the last Cursor provided to onLoadFinished()
        // above is about to be closed.  We need to make sure we are no
        // longer using it.
        mAdapter.swapCursor(null)
    }
}

জাভা

public static class CursorLoaderListFragment extends ListFragment
        implements OnQueryTextListener, LoaderManager.LoaderCallbacks<Cursor> {

    // This is the Adapter being used to display the list's data.
    SimpleCursorAdapter mAdapter;

    // If non-null, this is the current filter the user has provided.
    String curFilter;

    @Override public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        // Prepare the loader.  Either re-connect with an existing one,
        // or start a new one.
        getLoaderManager().initLoader(0, null, this);
    }

    @Override public void onViewCreated(@NonNull View view, Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);

        // Give some text to display if there is no data.  In a real
        // application, this would come from a resource.
        setEmptyText("No phone numbers");

        // We have a menu item to show in action bar.
        setHasOptionsMenu(true);

        // Create an empty adapter we will use to display the loaded data.
        mAdapter = new SimpleCursorAdapter(getActivity(),
                android.R.layout.simple_list_item_2, null,
                new String[] { Contacts.DISPLAY_NAME, Contacts.CONTACT_STATUS },
                new int[] { android.R.id.text1, android.R.id.text2 }, 0);
        setListAdapter(mAdapter);
    }

    @Override public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
        // Place an action bar item for searching.
        MenuItem item = menu.add("Search");
        item.setIcon(android.R.drawable.ic_menu_search);
        item.setShowAsAction(MenuItem.SHOW_AS_ACTION_IF_ROOM);
        SearchView sv = new SearchView(getActivity());
        sv.setOnQueryTextListener(this);
        item.setActionView(sv);
    }

    public boolean onQueryTextChange(String newText) {
        // Called when the action bar search text has changed.  Update
        // the search filter, and restart the loader to do a new query
        // with this filter.
        curFilter = !TextUtils.isEmpty(newText) ? newText : null;
        getLoaderManager().restartLoader(0, null, this);
        return true;
    }

    @Override public boolean onQueryTextSubmit(String query) {
        // Don't care about this.
        return true;
    }

    @Override public void onListItemClick(ListView l, View v, int position, long id) {
        // Insert desired behavior here.
        Log.i("FragmentComplexList", "Item clicked: " + id);
    }

    // These are the Contacts rows that we will retrieve.
    static final String[] CONTACTS_SUMMARY_PROJECTION = new String[] {
        Contacts._ID,
        Contacts.DISPLAY_NAME,
        Contacts.CONTACT_STATUS,
        Contacts.CONTACT_PRESENCE,
        Contacts.PHOTO_ID,
        Contacts.LOOKUP_KEY,
    };
    public Loader<Cursor> onCreateLoader(int id, Bundle args) {
        // This is called when a new Loader needs to be created.  This
        // sample only has one Loader, so we don't care about the ID.
        // First, pick the base URI to use depending on whether we are
        // currently filtering.
        Uri baseUri;
        if (curFilter != null) {
            baseUri = Uri.withAppendedPath(Contacts.CONTENT_FILTER_URI,
                    Uri.encode(curFilter));
        } else {
            baseUri = Contacts.CONTENT_URI;
        }

        // Now create and return a CursorLoader that will take care of
        // creating a Cursor for the data being displayed.
        String select = "((" + Contacts.DISPLAY_NAME + " NOTNULL) AND ("
                + Contacts.HAS_PHONE_NUMBER + "=1) AND ("
                + Contacts.DISPLAY_NAME + " != '' ))";
        return new CursorLoader(getActivity(), baseUri,
                CONTACTS_SUMMARY_PROJECTION, select, null,
                Contacts.DISPLAY_NAME + " COLLATE LOCALIZED ASC");
    }

    public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
        // Swap the new cursor in.  (The framework will take care of closing the
        // old cursor once we return.)
        mAdapter.swapCursor(data);
    }

    public void onLoaderReset(Loader<Cursor> loader) {
        // This is called when the last Cursor provided to onLoadFinished()
        // above is about to be closed.  We need to make sure we are no
        // longer using it.
        mAdapter.swapCursor(null);
    }
}

আরো উদাহরণ

নিম্নলিখিত উদাহরণগুলি কীভাবে লোডার ব্যবহার করতে হয় তা ব্যাখ্যা করে:

  • LoaderCursor : পূর্ববর্তী স্নিপেটের একটি সম্পূর্ণ সংস্করণ।
  • পরিচিতিগুলির একটি তালিকা পুনরুদ্ধার করুন : একটি ওয়াকথ্রু যা পরিচিতি প্রদানকারীর থেকে ডেটা পুনরুদ্ধার করতে CursorLoader ব্যবহার করে।
  • LoaderThrottle : একটি বিষয়বস্তু প্রদানকারী যখন তার ডেটা পরিবর্তিত হয় তখন প্রশ্নের সংখ্যা কমাতে থ্রটলিং ব্যবহার করার একটি উদাহরণ।