استفاده از ابزارهای بازرسی کد، مانند lint ، میتواند به شما در یافتن مشکلات و بهبود کدتان کمک کند، اما ابزارهای بازرسی فقط میتوانند موارد زیادی را استنتاج کنند. به عنوان مثال، شناسههای منبع Android از یک int
برای شناسایی رشتهها، گرافیکها، رنگها و سایر انواع منابع استفاده میکنند، بنابراین ابزارهای بازرسی نمیتوانند تشخیص دهند که چه زمانی یک منبع رشته را مشخص کردهاید که باید در آن رنگ مشخص میکردید. این وضعیت به این معنی است که برنامه شما ممکن است نادرست اجرا شود یا اصلا اجرا نشود، حتی اگر از بازرسی کد استفاده کنید.
حاشیه نویسی به شما امکان می دهد نکاتی را برای ابزارهای بازرسی کد مانند پرز ارائه کنید تا به شناسایی این مشکلات ظریف کد کمک کند. حاشیه نویسی ها به عنوان تگ های فراداده اضافه می شوند که به متغیرها، پارامترها، و مقادیر برگردانده می شوند تا مقادیر بازگشتی روش، پارامترهای ارسال شده، متغیرهای محلی و فیلدها را بررسی کنید. هنگامی که با ابزارهای بازرسی کد استفاده می شود، حاشیه نویسی می تواند به شما در تشخیص مشکلاتی مانند استثناهای اشاره گر تهی و تضاد نوع منبع کمک کند.
اندروید از طریق کتابخانه Jetpack Annotations از انواع حاشیه نویسی ها پشتیبانی می کند. از طریق بسته androidx.annotation
می توانید به کتابخانه دسترسی داشته باشید.
توجه: اگر یک ماژول به یک پردازشگر حاشیه نویسی وابستگی دارد، باید از پیکربندی وابستگی kapt
یا ksp
برای Kotlin یا پیکربندی وابستگی annotationProcessor
برای جاوا برای افزودن آن وابستگی استفاده کنید.
حاشیه نویسی را به پروژه خود اضافه کنید
برای فعال کردن حاشیه نویسی در پروژه خود، وابستگی androidx.annotation:annotation
به کتابخانه یا برنامه خود اضافه کنید. هر حاشیهنویسی که اضافه میکنید، هنگام اجرای بازرسی کد یا کار lint
بررسی میشود.
وابستگی کتابخانه Jetpack Annotations را اضافه کنید
کتابخانه Jetpack Annotations در Google's Maven Repository منتشر شده است. برای افزودن کتابخانه Jetpack Anotations به پروژه خود، خط زیر را در بلوک dependencies
های فایل build.gradle
یا build.gradle.kts
خود قرار دهید:
dependencies { implementation("androidx.annotation:annotation:1.9.1") }
dependencies { implementation 'androidx.annotation:annotation:1.9.1' }
اگر از حاشیه نویسی در ماژول کتابخانه خود استفاده می کنید، حاشیه نویسی به عنوان بخشی از آرتیفکت آرشیو Android (AAR) با فرمت XML در فایل annotations.zip
گنجانده می شود. افزودن وابستگی androidx.annotation
برای هیچ یک از کاربران پایین دستی کتابخانه شما وابستگی ایجاد نمی کند.
توجه: اگر از دیگر کتابخانههای Jetpack استفاده میکنید، ممکن است نیازی به اضافه کردن وابستگی androidx.annotation
نداشته باشید. از آنجا که بسیاری از کتابخانه های Jetpack دیگر به کتابخانه حاشیه نویسی وابسته هستند، ممکن است از قبل به حاشیه نویسی ها دسترسی داشته باشید.
برای فهرست کاملی از حاشیهنویسیهای موجود در مخزن Jetpack، یا به مرجع کتابخانه Jetpack Annotations مراجعه کنید یا از ویژگی تکمیل خودکار برای نمایش گزینههای موجود برای import androidx.annotation.
بیانیه
بازرسی کد را اجرا کنید
برای شروع بازرسی کد از Android Studio، که شامل اعتبار سنجی حاشیه نویسی و بررسی خودکار پرزها می شود، از منو گزینه Analyze > Inspect Code را انتخاب کنید. Android Studio پیامهای تضاد را نمایش میدهد تا مشکلات احتمالی را در مواردی که کد شما با حاشیهنویسی مغایرت دارد، پرچمگذاری کند و راهحلهای ممکن را پیشنهاد کند.
همچنین میتوانید با اجرای وظیفه lint
با استفاده از خط فرمان، حاشیهنویسی را اعمال کنید. اگرچه این ممکن است برای پرچمگذاری مشکلات با یک سرور یکپارچهسازی مداوم مفید باشد، اما وظیفه lint
حاشیهنویسی بیثبات را اعمال نمیکند (توضیح داده شده در بخش زیر). فقط اندروید استودیو این کار را انجام می دهد. برای کسب اطلاعات بیشتر در مورد فعال کردن و اجرای بازرسی پرز، به بهبود کد خود با بررسی لینت مراجعه کنید.
اگرچه تداخل حاشیه نویسی اخطار ایجاد می کند، اما این هشدارها مانع از کامپایل شدن برنامه شما نمی شود.
حاشیه نویسی پوچ
حاشیه نویسی پوچ می تواند در کد جاوا مفید باشد تا اینکه آیا مقادیر می توانند null باشند یا خیر. آنها در کد Kotlin کمتر مفید هستند، زیرا Kotlin قوانین پوچ پذیری را ایجاد کرده است که در زمان کامپایل اجرا می شوند. حاشیهنویسیهای @Nullable
و @NonNull
را برای بررسی پوچ بودن یک متغیر، پارامتر یا مقدار بازگشتی داده شده اضافه کنید. حاشیه نویسی @Nullable
یک متغیر، پارامتر یا مقدار بازگشتی را نشان می دهد که می تواند null باشد. @NonNull
یک متغیر، پارامتر یا مقدار بازگشتی را نشان می دهد که نمی تواند null باشد.
به عنوان مثال، اگر یک متغیر محلی که حاوی مقدار تهی است به عنوان پارامتر به یک متد با حاشیه نویسی @NonNull
به آن پارامتر ارسال شود، ساخت کد یک هشدار ایجاد می کند که نشان دهنده یک تضاد غیر تهی است. همچنین، تلاش برای ارجاع به نتیجه روشی که توسط @Nullable
مشخص شده است، بدون اینکه ابتدا بررسی کنید که آیا نتیجه صفر است یا خیر، یک هشدار تهی ایجاد می کند. فقط در صورتی @Nullable
در مقدار برگشتی یک متد استفاده کنید که هر استفاده از متد باید صریحاً تهی باشد.
مثال زیر پوچ بودن را در عمل نشان می دهد. کد مثال Kotlin از حاشیه نویسی @NonNull
استفاده نمی کند زیرا زمانی که یک نوع غیر قابل تهی مشخص شود به طور خودکار به بایت کد تولید شده اضافه می شود. مثال جاوا از حاشیه نویسی @NonNull
بر روی پارامترهای context
و attrs
استفاده می کند تا بررسی کند که مقادیر پارامترهای ارسال شده تهی نیستند. همچنین بررسی میکند که خود متد onCreateView()
null را برنگرداند:
... /** Annotation not used because of the safe-call operator(?)**/ override fun onCreateView( name: String?, context: Context, attrs: AttributeSet ): View? { ... } ...
import androidx.annotation.NonNull; ... /** Add support for inflating the <fragment> tag. **/ @NonNull @Override public View onCreateView(String name, @NonNull Context context, @NonNull AttributeSet attrs) { ... } ...
تجزیه و تحلیل پوچ پذیری
Android Studio از اجرای تجزیه و تحلیل پوچ پذیری برای استنباط خودکار و درج حاشیه نویسی پوچ در کد شما پشتیبانی می کند. تجزیه و تحلیل پوچ پذیری قراردادها را در سراسر سلسله مراتب روش در کد شما اسکن می کند تا تشخیص دهد:
- فراخوانی متدهایی که می توانند null برگردانند.
- روش هایی که نباید پوچ برگردانند.
- متغیرهایی مانند فیلدها، متغیرهای محلی و پارامترها که می توانند تهی باشند.
- متغیرهایی مانند فیلدها، متغیرهای محلی و پارامترهایی که نمی توانند مقدار تهی را نگه دارند.
سپس تجزیه و تحلیل به طور خودکار حاشیه نویسی تهی مناسب را در مکان های شناسایی شده درج می کند.
برای اجرای تجزیه و تحلیل پوچ پذیری در Android Studio، Analyze > Infer Nullity را انتخاب کنید. Android Studio حاشیه نویسی های Android @Nullable
و @NonNull
را در مکان های شناسایی شده در کد شما قرار می دهد. پس از اجرای تجزیه و تحلیل تهی، این یک تمرین خوب برای تأیید حاشیه نویسی تزریق شده است.
توجه: هنگام اضافه کردن حاشیهنویسیهای پوچ، تکمیل خودکار ممکن است حاشیهنویسی IntelliJ @Nullable
و @NotNull
را به جای حاشیهنویسی پوچ Android پیشنهاد دهد و ممکن است کتابخانه مربوطه را بهطور خودکار وارد کند. با این حال، جستجوگر پرز اندروید استودیو فقط به دنبال حاشیهنویسیهای پوچ اندروید است. هنگام تأیید حاشیهنویسیهای خود، تأیید کنید که پروژه شما از حاشیهنویسیهای پوچ Android استفاده میکند تا بررسیکننده پرز بتواند بهدرستی در حین بازرسی کد به شما اطلاع دهد.
حاشیه نویسی منابع
اعتبارسنجی انواع منابع می تواند مفید باشد زیرا ارجاعات Android به منابع، مانند منابع قابل ترسیم و رشته ، به عنوان اعداد صحیح ارسال می شوند.
کدی که انتظار دارد یک پارامتر به نوع خاصی از منبع، مانند یک String
ارجاع دهد، می تواند به نوع مرجع مورد انتظار int
ارسال شود، اما در واقع به نوع دیگری از منبع، مانند منبع R.string
اشاره می کند.
برای مثال، همانطور که در اینجا نشان داده شده است، حاشیهنویسیهای @StringRes
را اضافه کنید تا بررسی کنید که آیا یک پارامتر منبع دارای مرجع R.string
است یا خیر:
abstract fun setTitle(@StringRes resId: Int)
public abstract void setTitle(@StringRes int resId)
در حین بازرسی کد، اگر یک مرجع R.string
در پارامتر ارسال نشود، حاشیه نویسی یک هشدار ایجاد می کند.
حاشیه نویسی برای انواع دیگر منابع، مانند @DrawableRes
، @DimenRes
، @ColorRes
و @InterpolatorRes
، می تواند با استفاده از همان قالب حاشیه نویسی اضافه شود و در طول بازرسی کد اجرا شود.
اگر پارامتر شما از چندین نوع منبع پشتیبانی می کند، می توانید بیش از یک یادداشت نوع منبع را روی یک پارامتر مشخص قرار دهید. از @AnyRes
برای نشان دادن اینکه پارامتر مشروح شده می تواند هر نوع منبع R
باشد استفاده کنید.
اگرچه میتوانید از @ColorRes
برای تعیین اینکه یک پارامتر باید یک منبع رنگی باشد استفاده کنید، یک عدد صحیح رنگی (در قالب RRGGBB
یا AARRGGBB
) به عنوان یک منبع رنگی شناسایی نمیشود. در عوض، از حاشیه نویسی @ColorInt
برای نشان دادن اینکه یک پارامتر باید یک عدد صحیح رنگی باشد استفاده کنید. ابزارهای ساخت، کد نادرستی را که شناسه منبع رنگی مانند android.R.color.black
را به جای یک عدد صحیح رنگی، به روشهای حاشیهنویسی ارسال میکند، پرچمگذاری میکند.
حاشیه نویسی موضوع
حاشیه نویسی موضوع بررسی می کند که آیا یک روش از نوع خاصی از رشته فراخوانی شده است یا خیر. حاشیه نویسی موضوع زیر پشتیبانی می شود:
ابزارهای ساخت، حاشیهنویسیهای @MainThread
و @UiThread
را بهعنوان قابل تعویض در نظر میگیرند، بنابراین میتوانید روشهای @UiThread
را از روشهای @MainThread
و بالعکس فراخوانی کنید. با این حال، در مورد برنامههای سیستمی با چندین نمایش در رشتههای مختلف، ممکن است یک رشته رابط کاربری با رشته اصلی متفاوت باشد. بنابراین، باید روشهای مرتبط با سلسلهمراتب نمای برنامه را با @UiThread
و فقط روشهای مرتبط با چرخه عمر برنامه را با @MainThread
حاشیهنویسی کنید.
اگر همه متدهای یک کلاس دارای نیاز رشتهای یکسان هستند، میتوانید یک حاشیهنویسی رشته تکی به کلاس اضافه کنید تا تأیید کنید که همه متدهای کلاس از یک نوع رشته فراخوانی میشوند.
استفاده متداول از حاشیه نویسی رشته، اعتبارسنجی این است که روش ها یا کلاس های حاشیه نویسی شده با @WorkerThread
فقط از یک رشته پس زمینه مناسب فراخوانی می شوند.
حاشیه نویسی محدودیت ارزش
از حاشیه نویسی @IntRange
، @FloatRange
و @Size
برای اعتبارسنجی مقادیر پارامترهای ارسال شده استفاده کنید. هر دو @IntRange
و @FloatRange
زمانی که روی پارامترهایی اعمال میشوند که کاربران احتمالاً محدوده را اشتباه میگیرند، بسیار مفید هستند.
حاشیه نویسی @IntRange
تأیید می کند که یک مقدار پارامتر عدد صحیح یا طولانی در یک محدوده مشخص قرار دارد. مثال زیر نشان می دهد که پارامتر alpha
باید دارای یک عدد صحیح از 0 تا 255 باشد:
fun setAlpha(@IntRange(from = 0, to = 255) alpha: Int) { ... }
public void setAlpha(@IntRange(from=0,to=255) int alpha) { ... }
حاشیه نویسی @FloatRange
بررسی می کند که آیا یک مقدار پارامتر شناور یا دوگانه در محدوده مشخصی از مقادیر ممیز شناور قرار دارد. مثال زیر نشان می دهد که پارامتر alpha
باید دارای مقدار شناور از 0.0 تا 1.0 باشد:
fun setAlpha(@FloatRange(from = 0.0, to = 1.0) alpha: Float) {...}
public void setAlpha(@FloatRange(from=0.0, to=1.0) float alpha) {...}
حاشیه نویسی @Size
اندازه یک مجموعه یا آرایه یا طول یک رشته را بررسی می کند. از حاشیه نویسی @Size
می توان برای تأیید کیفیت های زیر استفاده کرد:
- حداقل اندازه، مانند
@Size(min=2)
- حداکثر اندازه، مانند
@Size(max=2)
- اندازه دقیق، مانند
@Size(2)
- عددی که اندازه آن باید مضربی از آن باشد، مانند
@Size(multiple=2)
برای مثال @Size(min=1)
خالی نبودن یک مجموعه را بررسی میکند و @Size(3)
تأیید میکند که یک آرایه دقیقاً دارای سه مقدار است.
مثال زیر نشان می دهد که آرایه location
باید حداقل یک عنصر داشته باشد:
fun getLocation(button: View, @Size(min=1) location: IntArray) { button.getLocationOnScreen(location) }
void getLocation(View button, @Size(min=1) int[] location) { button.getLocationOnScreen(location); }
حاشیه نویسی مجوز
از حاشیه نویسی @RequiresPermission
برای تایید مجوزهای تماس گیرنده یک روش استفاده کنید. برای بررسی یک مجوز از لیست مجوزهای معتبر، از ویژگی anyOf
استفاده کنید. برای بررسی مجموعه ای از مجوزها، از ویژگی allOf
استفاده کنید. مثال زیر به متد setWallpaper()
اشاره می کند تا نشان دهد که فراخوان متد باید مجوز permission.SET_WALLPAPERS
داشته باشد:
@RequiresPermission(Manifest.permission.SET_WALLPAPER) @Throws(IOException::class) abstract fun setWallpaper(bitmap: Bitmap)
@RequiresPermission(Manifest.permission.SET_WALLPAPER) public abstract void setWallpaper(Bitmap bitmap) throws IOException;
مثال زیر به تماس گیرنده متد copyImageFile()
نیاز دارد که هم دسترسی خواندن به حافظه خارجی و هم دسترسی خواندن به ابرداده مکان در تصویر کپی شده داشته باشد:
@RequiresPermission(allOf = [ Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.ACCESS_MEDIA_LOCATION ]) fun copyImageFile(dest: String, source: String) { ... }
@RequiresPermission(allOf = { Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.ACCESS_MEDIA_LOCATION}) public static final void copyImageFile(String dest, String source) { //... }
برای مجوزهای intent، الزامات مجوز را در فیلد رشته ای که نام اقدام intent را تعریف می کند، قرار دهید:
@RequiresPermission(android.Manifest.permission.BLUETOOTH) const val ACTION_REQUEST_DISCOVERABLE = "android.bluetooth.adapter.action.REQUEST_DISCOVERABLE"
@RequiresPermission(android.Manifest.permission.BLUETOOTH) public static final String ACTION_REQUEST_DISCOVERABLE = "android.bluetooth.adapter.action.REQUEST_DISCOVERABLE";
برای مجوزهای ارائهدهندگان محتوا که به مجوزهای جداگانه برای دسترسی خواندن و نوشتن نیاز دارند، هر مورد نیاز مجوز را در @RequiresPermission.Read
یا @RequiresPermission.Write
حاشیهنویسی بنویسید:
@RequiresPermission.Read(RequiresPermission(READ_HISTORY_BOOKMARKS)) @RequiresPermission.Write(RequiresPermission(WRITE_HISTORY_BOOKMARKS)) val BOOKMARKS_URI = Uri.parse("content://browser/bookmarks")
@RequiresPermission.Read(@RequiresPermission(READ_HISTORY_BOOKMARKS)) @RequiresPermission.Write(@RequiresPermission(WRITE_HISTORY_BOOKMARKS)) public static final Uri BOOKMARKS_URI = Uri.parse("content://browser/bookmarks");
مجوزهای غیر مستقیم
هنگامی که یک مجوز به مقدار خاصی که به پارامتر یک متد ارائه می شود بستگی دارد، از @RequiresPermission
بر روی خود پارامتر بدون فهرست کردن مجوزهای خاص استفاده کنید. به عنوان مثال، متد startActivity(Intent)
از یک مجوز غیرمستقیم در intent ارسال شده به متد استفاده می کند:
abstract fun startActivity(@RequiresPermission intent: Intent, bundle: Bundle?)
public abstract void startActivity(@RequiresPermission Intent intent, @Nullable Bundle)
هنگامی که از مجوزهای غیرمستقیم استفاده می کنید، ابزارهای ساخت، تجزیه و تحلیل جریان داده را انجام می دهند تا بررسی کنند که آیا آرگومان ارسال شده به متد دارای حاشیه نویسی @RequiresPermission
است یا خیر. سپس هر گونه حاشیه نویسی موجود را از پارامتر روی خود متد اعمال می کنند. در مثال startActivity(Intent)
، حاشیهنویسیها در کلاس Intent
هنگامی که یک intent بدون مجوزهای مناسب به متد ارسال میشود، هشدارهای ناشی از استفادههای نامعتبر از startActivity(Intent)
را ایجاد میکنند، همانطور که در شکل 1 نشان داده شده است.
ابزارهای ساخت اخطار را در startActivity(Intent)
از حاشیه نویسی روی نام اقدام intent مربوطه در کلاس Intent
ایجاد می کنند:
@RequiresPermission(Manifest.permission.CALL_PHONE) const val ACTION_CALL = "android.intent.action.CALL"
@RequiresPermission(Manifest.permission.CALL_PHONE) public static final String ACTION_CALL = "android.intent.action.CALL";
در صورت لزوم، میتوانید @RequiresPermission
برای @RequiresPermission.Read
یا @RequiresPermission.Write
در هنگام حاشیهنویسی پارامتر یک روش جایگزین کنید. با این حال، برای مجوزهای غیرمستقیم @RequiresPermission
نباید همراه با یادداشتهای مجوز خواندن یا نوشتن استفاده شود.
حاشیه نویسی ارزش بازگشتی
از حاشیه نویسی @CheckResult
برای تأیید اینکه نتیجه یا مقدار برگشتی یک روش واقعاً استفاده شده است استفاده کنید. به جای حاشیه نویسی هر روش غیر خالی با @CheckResult
، حاشیه نویسی را اضافه کنید تا نتایج روش های بالقوه گیج کننده را روشن کنید.
به عنوان مثال، توسعه دهندگان جدید جاوا اغلب به اشتباه فکر می کنند که < String >.trim()
فضای خالی را از رشته اصلی حذف می کند. حاشیه نویسی متد با پرچم های @CheckResult
از < String >.trim()
استفاده می کند که در آن تماس گیرنده با مقدار برگشتی متد کاری انجام نمی دهد.
مثال زیر متد checkPermissions()
را برای بررسی اینکه آیا مقدار بازگشتی متد واقعاً ارجاع داده شده است یا خیر توضیح می دهد. همچنین متد enforcePermission()
به عنوان روشی که باید به عنوان جایگزین به توسعهدهنده پیشنهاد شود نام میبرد:
@CheckResult(suggest = "#enforcePermission(String,int,int,String)") abstract fun checkPermission(permission: String, pid: Int, uid: Int): Int
@CheckResult(suggest="#enforcePermission(String,int,int,String)") public abstract int checkPermission(@NonNull String permission, int pid, int uid);
حاشیه نویسی CallSuper
از حاشیه نویسی @CallSuper
استفاده کنید تا تأیید کنید که یک روش فراخوان اجرای فوق العاده متد را فراخوانی می کند.
مثال زیر متد onCreate()
را حاشیه نویسی می کند تا اطمینان حاصل شود که هر پیاده سازی متد برتری super.onCreate()
را فراخوانی می کند:
@CallSuper override fun onCreate(savedInstanceState: Bundle?) { }
@CallSuper protected void onCreate(Bundle savedInstanceState) { }
حاشیه نویسی Typedef
حاشیه نویسی Typedef بررسی می کند که آیا یک پارامتر خاص، مقدار بازگشتی یا فیلد به مجموعه خاصی از ثابت ها ارجاع می دهد یا خیر. آنها همچنین تکمیل کد را برای ارائه خودکار ثابت های مجاز فعال می کنند.
از حاشیه نویسی @IntDef
و @StringDef
برای ایجاد حاشیه نویسی شمارش شده از مجموعه اعداد صحیح و رشته ها برای اعتبارسنجی انواع دیگر منابع کد استفاده کنید.
حاشیه نویسی Typedef از @interface
برای اعلام نوع حاشیه نویسی شمارش شده جدید استفاده می کند. حاشیه نویسی @IntDef
و @StringDef
، همراه با @Retention
، حاشیه نویسی جدید را نشان می دهد و برای تعریف نوع شمارش شده ضروری است. حاشیه نویسی @Retention(RetentionPolicy.SOURCE)
به کامپایلر می گوید که داده های حاشیه نویسی شمارش شده را در فایل .class
ذخیره نکند.
مثال زیر مراحل ایجاد حاشیه نویسی را نشان می دهد که بررسی می کند آیا مقداری که به عنوان پارامتر متد ارسال می شود به یکی از ثابت های تعریف شده ارجاع می دهد یا خیر:
import androidx.annotation.IntDef //... // Define the list of accepted constants and declare the NavigationMode annotation. @Retention(AnnotationRetention.SOURCE) @IntDef(NAVIGATION_MODE_STANDARD, NAVIGATION_MODE_LIST, NAVIGATION_MODE_TABS) annotation class NavigationMode // Declare the constants. const val NAVIGATION_MODE_STANDARD = 0 const val NAVIGATION_MODE_LIST = 1 const val NAVIGATION_MODE_TABS = 2 abstract class ActionBar { // Decorate the target methods with the annotation. // Attach the annotation. @get:NavigationMode @setparam:NavigationMode abstract var navigationMode: Int }
import androidx.annotation.IntDef; //... public abstract class ActionBar { //... // Define the list of accepted constants and declare the NavigationMode annotation. @Retention(RetentionPolicy.SOURCE) @IntDef({NAVIGATION_MODE_STANDARD, NAVIGATION_MODE_LIST, NAVIGATION_MODE_TABS}) public @interface NavigationMode {} // Declare the constants. public static final int NAVIGATION_MODE_STANDARD = 0; public static final int NAVIGATION_MODE_LIST = 1; public static final int NAVIGATION_MODE_TABS = 2; // Decorate the target methods with the annotation. @NavigationMode public abstract int getNavigationMode(); // Attach the annotation. public abstract void setNavigationMode(@NavigationMode int mode); }
وقتی این کد را میسازید، اگر پارامتر mode
به یکی از ثابتهای تعریفشده ( NAVIGATION_MODE_STANDARD
، NAVIGATION_MODE_LIST
، یا NAVIGATION_MODE_TABS
) اشاره نکند، هشداری ایجاد میشود.
@IntDef
و @IntRange
را ترکیب کنید تا نشان دهید که یک عدد صحیح می تواند مجموعه ای از ثابت ها یا مقداری در یک محدوده باشد.
ترکیب ثابت ها با پرچم ها را فعال کنید
اگر کاربران می توانند ثابت های مجاز را با یک پرچم ترکیب کنند (مانند |
، &
، ^
، و غیره)، می توانید یک حاشیه نویسی با ویژگی flag
تعریف کنید تا بررسی کنید که آیا یک پارامتر یا مقدار بازگشتی به یک الگوی معتبر ارجاع می دهد یا خیر.
مثال زیر حاشیه نویسی DisplayOptions
با لیستی از ثابت های معتبر DISPLAY_
ایجاد می کند:
import androidx.annotation.IntDef ... @IntDef(flag = true, value = [ DISPLAY_USE_LOGO, DISPLAY_SHOW_HOME, DISPLAY_HOME_AS_UP, DISPLAY_SHOW_TITLE, DISPLAY_SHOW_CUSTOM ]) @Retention(AnnotationRetention.SOURCE) annotation class DisplayOptions ...
import androidx.annotation.IntDef; ... @IntDef(flag=true, value={ DISPLAY_USE_LOGO, DISPLAY_SHOW_HOME, DISPLAY_HOME_AS_UP, DISPLAY_SHOW_TITLE, DISPLAY_SHOW_CUSTOM }) @Retention(RetentionPolicy.SOURCE) public @interface DisplayOptions {} ...
وقتی کدی را با یک پرچم حاشیهنویسی میسازید، اگر پارامتر تزئین شده یا مقدار بازگشتی به یک الگوی معتبر اشاره نکند، یک هشدار ایجاد میشود.
حاشیه نویسی را حفظ کنید
حاشیه نویسی @Keep
تضمین می کند که یک کلاس یا روش حاشیه نویسی زمانی که کد در زمان ساخت کوچک می شود حذف نمی شود. این حاشیهنویسی معمولاً به متدها و کلاسهایی اضافه میشود که از طریق بازتاب به آنها دسترسی پیدا میکنند تا مانع از تلقی کامپایلر به عنوان استفادهنشده از کد شود.
احتیاط: کلاسها و روشهایی که با استفاده از @Keep
حاشیهنویسی میکنید، همیشه در APK برنامهتان ظاهر میشوند، حتی اگر هرگز به این کلاسها و روشها در منطق برنامهتان مراجعه نکنید.
برای کوچک نگه داشتن اندازه برنامه، در نظر بگیرید که آیا لازم است هر حاشیه نویسی @Keep
را در برنامه خود حفظ کنید. اگر از بازتاب برای دسترسی به یک کلاس یا روش مشروح استفاده میکنید، از یک -if
شرطی در قوانین ProGuard خود استفاده کنید و کلاسی را که بازتاب را فراخوانی میکند، مشخص کنید.
برای اطلاعات بیشتر در مورد نحوه کوچک کردن کد و تعیین اینکه کدام کد حذف نشود، به کوچک کردن، مبهم کردن، و بهینه سازی برنامه خود مراجعه کنید.
حاشیه نویسی قابل مشاهده کد
از حاشیه نویسی های زیر برای نشان دادن قابلیت مشاهده بخش های خاصی از کد مانند متدها، کلاس ها، فیلدها یا بسته ها استفاده کنید.
کد را برای آزمایش قابل مشاهده کنید
حاشیه نویسی @VisibleForTesting
نشان می دهد که یک روش حاشیه نویسی بیشتر از حد معمول برای آزمایش پذیر بودن روش قابل مشاهده است. این حاشیهنویسی یک آرگومان اختیاری otherwise
دارد که به شما امکان میدهد مشخص کنید که اگر روش برای آزمایش قابل مشاهده نباشد، چقدر قابل مشاهده است. Lint از آرگومان otherwise
برای اعمال دید مورد نظر استفاده می کند.
در مثال زیر، myMethod()
معمولاً private
است، اما برای آزمایشها package-private
است. با نام VisibleForTesting.PRIVATE
، اگر این روش از خارج از زمینه مجاز توسط دسترسی private
، مانند یک واحد کامپایل متفاوت فراخوانی شود، lint پیامی را نمایش می دهد.
@VisibleForTesting(otherwise = VisibleForTesting.PRIVATE) fun myMethod() { ... }
@VisibleForTesting(otherwise = VisibleForTesting.PRIVATE) void myMethod() { ... }
همچنین میتوانید @VisibleForTesting(otherwise = VisibleForTesting.NONE)
را مشخص کنید تا نشان دهید که یک روش فقط برای آزمایش وجود دارد. این فرم مانند استفاده از @RestrictTo(TESTS)
است. هر دو همان بررسی پرز را انجام می دهند.
یک API را محدود کنید
حاشیه نویسی @RestrictTo
نشان می دهد که دسترسی به API مشروح (بسته، کلاس یا روش) محدود است، به شرح زیر:
زیر کلاس ها
از فرم حاشیهنویسی @RestrictTo(RestrictTo.Scope.SUBCLASSES)
استفاده کنید تا دسترسی API را فقط به زیر کلاسها محدود کنید.
فقط کلاس هایی که کلاس حاشیه نویسی را گسترش می دهند می توانند به این API دسترسی داشته باشند. اصلاح کننده protected
جاوا به اندازه کافی محدود کننده نیست، زیرا امکان دسترسی از کلاس های نامرتبط در یک بسته را فراهم می کند. همچنین، مواردی وجود دارد که میخواهید یک متد را برای انعطافپذیری آینده public
بگذارید، زیرا هرگز نمیتوانید یک متد protected
و لغو شده قبلی را public
کنید، اما میخواهید یک اشاره ارائه دهید که کلاس برای استفاده در کلاس یا از زیر کلاسها در نظر گرفته شده است. فقط
کتابخانه ها
از فرم حاشیه نویسی @RestrictTo(RestrictTo.Scope.LIBRARY_GROUP_PREFIX)
استفاده کنید تا دسترسی API را فقط به کتابخانه های خود محدود کنید.
فقط کد کتابخانه شما می تواند به API مشروح شده دسترسی داشته باشد. این به شما امکان می دهد نه تنها کد خود را در هر سلسله مراتب بسته ای که می خواهید سازماندهی کنید، بلکه کد را در میان گروهی از کتابخانه های مرتبط به اشتراک بگذارید. این گزینه در حال حاضر برای کتابخانههای Jetpack در دسترس است که کدهای پیادهسازی زیادی دارند که برای استفاده خارجی در نظر گرفته نشده است، اما باید public
باشد تا آن را در کتابخانههای مختلف Jetpack مکمل به اشتراک بگذارد.
تست کردن
از فرم حاشیه نویسی @RestrictTo(RestrictTo.Scope.TESTS)
برای جلوگیری از دسترسی سایر توسعه دهندگان به APIهای آزمایشی شما استفاده کنید.
فقط کد آزمایشی می تواند به API مشروح شده دسترسی داشته باشد. این باعث میشود دیگر توسعهدهندگان نتوانند از APIهایی برای توسعهای که فقط برای اهداف آزمایشی در نظر دارید استفاده کنند.