بررسی اجمالی فرآیندها و موضوعات

هنگامی که یک جزء برنامه شروع می شود و برنامه هیچ مؤلفه دیگری در حال اجرا ندارد، سیستم Android یک فرآیند لینوکس جدید را برای برنامه با یک رشته اجرا شروع می کند. به طور پیش فرض، تمام اجزای یک برنامه کاربردی در یک فرآیند و رشته یکسان اجرا می شوند که به آن نخ اصلی می گویند.

اگر یک مؤلفه برنامه شروع شود و از قبل یک فرآیند برای آن برنامه وجود داشته باشد، زیرا مؤلفه دیگری از برنامه قبلاً شروع شده است، مؤلفه در آن فرآیند شروع می شود و از همان رشته اجرا استفاده می کند. با این حال، می‌توانید ترتیبی دهید که مؤلفه‌های مختلف در برنامه خود در فرآیندهای جداگانه اجرا شوند و می‌توانید رشته‌های اضافی برای هر فرآیندی ایجاد کنید.

این سند نحوه عملکرد پردازش ها و رشته ها در یک برنامه اندروید را مورد بحث قرار می دهد.

فرآیندها

به‌طور پیش‌فرض، همه اجزای برنامه در یک فرآیند اجرا می‌شوند و اکثر برنامه‌ها این را تغییر نمی‌دهند. با این حال، اگر متوجه شدید که باید کنترل کنید که یک جزء خاص به کدام فرآیند تعلق دارد، می توانید این کار را در فایل مانیفست انجام دهید.

ورودی مانیفست برای هر نوع عنصر مؤلفه - <activity> ، <service> ، <receiver> و <provider> —از ویژگی android:process پشتیبانی می‌کند که می‌تواند فرآیندی را مشخص کند که جزء اجرا می‌شود. می‌توانید این ویژگی را طوری تنظیم کنید که هر کامپوننت در فرآیند خود اجرا می شود یا به طوری که برخی از اجزای یک فرآیند را به اشتراک می گذارند در حالی که برخی دیگر این کار را نمی کنند.

همچنین می‌توانید android:process طوری تنظیم کنید که اجزای برنامه‌های مختلف در یک فرآیند اجرا شوند، مشروط بر اینکه برنامه‌ها شناسه کاربری لینوکس یکسانی داشته باشند و با گواهینامه‌های یکسانی امضا شده باشند.

عنصر <application> همچنین از ویژگی android:process پشتیبانی می‌کند که می‌توانید از آن برای تنظیم یک مقدار پیش‌فرض که برای همه مؤلفه‌ها اعمال می‌شود، استفاده کنید.

اندروید ممکن است زمانی تصمیم بگیرد که فرآیندی را خاموش کند، زمانی که سایر فرآیندها به منابع نیاز دارند که سریع‌تر در خدمت کاربر هستند. مؤلفه های برنامه در حال اجرا در فرآیندی که خاموش شده اند، در نتیجه از بین می روند. زمانی که کاری برای آن مؤلفه‌ها وجود داشته باشد، یک فرآیند دوباره شروع می‌شود.

هنگامی که تصمیم می گیرید کدام فرآیندها را خاموش کنید، سیستم اندروید اهمیت نسبی آنها را برای کاربر می سنجد. به عنوان مثال، در مقایسه با فرآیندی که میزبان فعالیت‌های قابل مشاهده است، با سهولت بیشتری فعالیت‌های میزبانی فرآیند را که دیگر روی صفحه قابل مشاهده نیستند، خاموش می‌کند. بنابراین، تصمیم در مورد خاتمه دادن به یک فرآیند به وضعیت اجزای در حال اجرا در آن فرآیند بستگی دارد.

جزئیات چرخه عمر فرآیند و ارتباط آن با حالت های برنامه در فرآیندها و چرخه عمر برنامه مورد بحث قرار گرفته است.

موضوعات

هنگامی که یک برنامه راه اندازی می شود، سیستم یک رشته اجرایی برای برنامه ایجاد می کند که به آن رشته اصلی می گویند. این موضوع بسیار مهم است، زیرا وظیفه ارسال رویدادها به ویجت های رابط کاربری مناسب از جمله رویدادهای ترسیمی را بر عهده دارد. همچنین تقریباً همیشه رشته‌ای است که در آن برنامه شما با مؤلفه‌های بسته‌های android.widget و android.view جعبه ابزار Android UI تعامل دارد. به همین دلیل، رشته اصلی گاهی اوقات رشته UI نامیده می شود. با این حال، تحت شرایط خاص، رشته اصلی برنامه ممکن است رشته رابط کاربری آن نباشد. برای اطلاعات بیشتر، حاشیه نویسی موضوع را ببینید.

سیستم یک رشته مجزا برای هر نمونه از یک جزء ایجاد نمی کند. تمام اجزایی که در یک فرآیند اجرا می شوند در رشته UI نمونه سازی می شوند و فراخوانی های سیستم به هر جزء از آن رشته ارسال می شود. در نتیجه، روش‌هایی که به تماس‌های سیستم پاسخ می‌دهند - مانند onKeyDown() برای گزارش اقدامات کاربر، یا یک روش بازگشت به تماس چرخه حیات - همیشه در رشته UI فرآیند اجرا می‌شوند.

به عنوان مثال، وقتی کاربر دکمه‌ای را روی صفحه لمس می‌کند، رشته رابط کاربری برنامه شما رویداد لمسی را به ویجت ارسال می‌کند، که به نوبه خود حالت فشرده آن را تنظیم می‌کند و یک درخواست نامعتبر را در صف رویداد ارسال می‌کند. رشته UI درخواست را از صف خارج می کند و به ویجت اطلاع می دهد تا خودش را دوباره ترسیم کند.

اگر برنامه خود را به درستی پیاده سازی نکنید، این مدل تک رشته ای زمانی که برنامه شما کار فشرده ای را در پاسخ به تعامل کاربر انجام می دهد، می تواند عملکرد ضعیفی داشته باشد. انجام عملیات طولانی در رشته UI، مانند دسترسی به شبکه یا جستجوهای پایگاه داده، کل UI را مسدود می کند. وقتی رشته مسدود می شود، هیچ رویدادی از جمله رویدادهای ترسیمی قابل ارسال نیست.

از منظر کاربر، به نظر می رسد که برنامه معلق است. حتی بدتر از آن، اگر رشته UI برای بیش از چند ثانیه مسدود شود، گفتگوی " برنامه پاسخ نمی دهد " (ANR) به کاربر نمایش داده می شود. ممکن است کاربر تصمیم بگیرد که برنامه شما را ترک کند یا حتی آن را حذف نصب کند.

به خاطر داشته باشید که جعبه ابزار Android UI امن نیست . بنابراین، رابط کاربری خود را از یک موضوع کارگر دستکاری نکنید. تمام دستکاری ها را در رابط کاربری خود از رشته UI انجام دهید. دو قانون برای مدل تک رشته ای اندروید وجود دارد:

  1. رشته رابط کاربری را مسدود نکنید.
  2. به جعبه ابزار Android UI از خارج از رشته UI دسترسی نداشته باشید.

نخ های کارگری

به دلیل این مدل تک رشته ای، برای پاسخگویی رابط کاربری برنامه شما ضروری است که رشته رابط کاربری را مسدود نکنید. اگر عملیاتی دارید که باید آنی انجام دهید، مطمئن شوید که آنها را در پس‌زمینه یا رشته‌های کارگری جداگانه انجام دهید. فقط به یاد داشته باشید که نمی‌توانید UI را از هیچ رشته‌ای غیر از UI یا رشته اصلی به‌روزرسانی کنید.

برای کمک به پیروی از این قوانین، Android راه‌های مختلفی برای دسترسی به رشته رابط کاربری از موضوعات دیگر ارائه می‌کند. در اینجا لیستی از روش هایی وجود دارد که می تواند کمک کند:

مثال زیر از View.post(Runnable) استفاده می کند:

کاتلین

fun onClick(v: View) {
    Thread(Runnable {
        // A potentially time consuming task.
        val bitmap = processBitMap("image.png")
        imageView.post {
            imageView.setImageBitmap(bitmap)
        }
    }).start()
}

جاوا

public void onClick(View v) {
    new Thread(new Runnable() {
        public void run() {
            // A potentially time consuming task.
            final Bitmap bitmap =
                    processBitMap("image.png");
            imageView.post(new Runnable() {
                public void run() {
                    imageView.setImageBitmap(bitmap);
                }
            });
        }
    }).start();
}

این پیاده‌سازی از نظر thread ایمن است، زیرا عملیات پس‌زمینه از یک رشته جداگانه انجام می‌شود در حالی که ImageView همیشه از رشته UI دستکاری می‌شود.

با این حال، با افزایش پیچیدگی عملیات، این نوع کد می تواند پیچیده و نگهداری آن دشوار باشد. برای مدیریت تعاملات پیچیده تر با یک رشته کاری، ممکن است از یک Handler در رشته کاری خود برای پردازش پیام های ارسال شده از رشته UI استفاده کنید. برای توضیح کامل نحوه زمان‌بندی کار بر روی رشته‌های پس‌زمینه و برقراری ارتباط مجدد با رشته رابط کاربری، به نمای کلی کار پس‌زمینه مراجعه کنید.

روش های ایمن با نخ

در برخی موقعیت‌ها، متدهایی که پیاده‌سازی می‌کنید از بیش از یک رشته فراخوانی می‌شوند، و بنابراین باید به گونه‌ای نوشته شوند که ایمن باشند.

این در درجه اول برای روش هایی که می توان از راه دور فراخوانی کرد، مانند روش هایی در یک سرویس محدود صادق است. هنگامی که یک فراخوانی روی یک متد پیاده‌سازی شده در یک IBinder از همان فرآیندی شروع می‌شود که در آن IBinder در حال اجرا است، متد در رشته تماس‌گیرنده اجرا می‌شود. با این حال، هنگامی که فراخوانی در فرآیند دیگری منشأ می‌گیرد، این متد در رشته‌ای انتخاب شده از مجموعه رشته‌هایی که سیستم در همان فرآیند IBinder نگهداری می‌کند، اجرا می‌شود. در رشته UI فرآیند اجرا نمی شود.

به عنوان مثال، در حالی که متد onBind() یک سرویس از رشته UI فرآیند سرویس فراخوانی می شود، متدهای پیاده سازی شده در شیئی که onBind() برمی گرداند، مانند زیر کلاسی که متدهای فراخوانی روش راه دور (RPC) را پیاده سازی می کند، از رشته ها فراخوانی می شوند. در استخر از آنجایی که یک سرویس می‌تواند بیش از یک کلاینت داشته باشد، بیش از یک رشته Pool می‌تواند به طور همزمان یک روش IBinder را درگیر کند، بنابراین روش‌های IBinder باید پیاده‌سازی شوند تا از نظر thread ایمن باشند.

به طور مشابه، ارائه‌دهنده محتوا می‌تواند درخواست‌های داده‌ای را دریافت کند که از فرآیندهای دیگر سرچشمه می‌گیرند. کلاس های ContentResolver و ContentProvider جزئیات نحوه مدیریت ارتباطات بین پردازشی (IPC) را پنهان می کنند، اما متدهای ContentProvider که به این درخواست ها پاسخ می دهند - متدهای query() , insert() , delete() , update() و getType() - از مجموعه ای از رشته ها در فرآیند ارائه دهنده محتوا فراخوانی می شوند، نه رشته رابط کاربر برای فرآیند. از آنجایی که این روش ها ممکن است از هر تعداد رشته به طور همزمان فراخوانی شوند، آنها نیز باید پیاده سازی شوند تا از نظر thread ایمن باشند.

ارتباطات بین فرآیندی

اندروید مکانیزمی را برای IPC با استفاده از RPC ارائه می‌کند، که در آن یک متد توسط یک اکتیویتی یا مؤلفه‌های دیگر برنامه فراخوانی می‌شود، اما در فرآیند دیگری از راه دور اجرا می‌شود و هر نتیجه‌ای به تماس‌گیرنده بازگردانده می‌شود. این امر مستلزم تجزیه یک فراخوانی متد و داده‌های آن به سطحی است که سیستم عامل می‌تواند آن را درک کند، آن را از فرآیند محلی و فضای آدرس به فرآیند و فضای آدرس راه دور منتقل می‌کند، و سپس مجدداً جمع‌آوری می‌شود و تماس را در آنجا برقرار می‌کند.

سپس مقادیر برگشتی در جهت مخالف منتقل می شوند. اندروید تمامی کدها را برای انجام این تراکنش های IPC ارائه می کند، بنابراین می توانید بر روی تعریف و پیاده سازی رابط برنامه نویسی RPC تمرکز کنید.

برای اجرای IPC، برنامه شما باید با استفاده از bindService() به یک سرویس متصل شود. برای اطلاعات بیشتر، به نمای کلی خدمات مراجعه کنید.