إذا كنت مستخدمًا جديدًا لتطبيق Bazel، يُرجى بدء البرنامج التعليمي الخاص بـ Building Android with Bazel.
نظرة عامة
يمكن تشغيل Bazel بالعديد من عمليات الإعداد المختلفة، بما في ذلك العديد من تلك الأدوات التي تستخدم سلسلة أدوات Android Native Development Kit (NDK). وهذا يعني أنّه يمكن تجميع قواعد cc_library وcc_binary العادية لنظام التشغيل Android مباشرةً ضمن Bazel. تحقّق Bazel هذا باستخدام قاعدة مستودع android_ndk_repository.
المتطلّبات الأساسية
يُرجى التأكّد من تثبيت حزمة تطوير البرامج (SDK) لنظام التشغيل Android وNDK.
لإعداد حزمة تطوير البرامج (SDK) وNDK، أضِف المقتطف التالي إلى WORKSPACE:
android_sdk_repository(
name = "androidsdk", # Required. Name *must* be "androidsdk".
path = "/path/to/sdk", # Optional. Can be omitted if `ANDROID_HOME` environment variable is set.
)
android_ndk_repository(
name = "androidndk", # Required. Name *must* be "androidndk".
path = "/path/to/ndk", # Optional. Can be omitted if `ANDROID_NDK_HOME` environment variable is set.
)
لمزيد من المعلومات عن قاعدة android_ndk_repository، راجِع إدخال
موسوعة موسوعة.
البدء بسرعة
لإنشاء +C أو Android، ما عليك سوى إضافة تبعيات cc_library إلى قواعد android_binary أو android_library.
على سبيل المثال، في حال استخدام ملف BUILD التالي لتطبيق Android:
# In <project>/app/src/main/BUILD.bazel
cc_library(
name = "jni_lib",
srcs = ["cpp/native-lib.cpp"],
)
android_library(
name = "lib",
srcs = ["java/com/example/android/bazel/MainActivity.java"],
resource_files = glob(["res/**/*"]),
custom_package = "com.example.android.bazel",
manifest = "LibraryManifest.xml",
deps = [":jni_lib"],
)
android_binary(
name = "app",
deps = [":lib"],
manifest = "AndroidManifest.xml",
)
يؤدي ملف BUILD هذا إلى الرسم البياني المستهدف التالي:

الشكل 1. إنشاء رسم بياني لمشروع Android باستخدام تبعيات cc_library.
لإنشاء التطبيق، ما عليك سوى تشغيل ما يلي:
bazel build //app/src/main:app
يُجمِّع الأمر bazel build ملفات Java وملفات موارد Android
وقواعد cc_library عن طريق تجميع كل البيانات في ملف APK:
$ zipinfo -1 bazel-bin/app/src/main/app.apk
nativedeps
lib/armeabi-v7a/libapp.so
classes.dex
AndroidManifest.xml
...
res/...
...
META-INF/CERT.SF
META-INF/CERT.RSA
META-INF/MANIFEST.MF
تُجمِّع Bazel جميع مكتبات cc_libras في ملف كائن واحد مشترك (.so)،
ويستهدف واجهة برمجة تطبيقات armeabi-v7a واجهة برمجة التطبيقات تلقائيًا. لتغيير هذا أو إنشاء عدة واجهات تطبيق ثنائية (ABI) في الوقت نفسه، يُرجى الاطّلاع على القسم الذي يتناول ضبط الهدف من واجهة التطبيق الثنائية (ABI).
مثال على الإعداد
يتوفّر هذا المثال في مستودع "مثلج".
في ملف BUILD.bazel، يتم تحديد ثلاثة استهدافات باستخدام القواعد android_binary
android_library وcc_library.
يعمل هدف المستوى الأعلى android_binary على إنشاء حزمة APK.
يحتوي الهدف cc_library على ملف مصدر C++ واحد مع تنفيذ دالة JNI:
#include <jni.h>
#include <string>
extern "C"
JNIEXPORT jstring
JNICALL
Java_com_example_android_bazel_MainActivity_stringFromJNI(
JNIEnv *env,
jobject /* this */) {
std::string hello = "Hello from C++";
return env->NewStringUTF(hello.c_str());
}
يحدّد الهدف android_library مصادر Java وملفات الموارد والاعتمادية على الهدف cc_library. في هذا المثال، يحمِّل MainActivity.java ملف العنصر المشترك libapp.so، ويحدّد توقيع الطريقة لدالة JNI:
public class MainActivity extends AppCompatActivity {
static {
System.loadLibrary("app");
}
@Override
protected void onCreate(Bundle savedInstanceState) {
// ...
}
public native String stringFromJNI();
}
ضبط STL
لضبط لغة C++ على STL، يجب استخدام العلامة --android_crosstool_top.
bazel build //:app --android_crosstool_top=target label
في ما يلي بنود الخدمة المتوفّرة في @androidndk:
| لغة الاستعلامات البنيوية (STL) | التصنيف المستهدف |
|---|---|
| منفذ | @androidndk//:toolchain-stlport |
| libc++ | @androidndk//:toolchain-libcpp |
| ويستنل | @androidndk//:toolchain-gnu-libstdcpp |
بالنسبة إلى الإصدار r16 والإصدارات الأقدم، يكون إصدار STL التلقائي هو gnustl. بالنسبة إلى الإصدار r17 والإصدارات الأحدث، تكون
libc++. ولتسهيل الأمر، تم تحويل الاسم @androidndk//:default_crosstool المستهدف إلى خدمات STL التلقائية المعنية.
يُرجى ملاحظة أنه اعتبارًا من r18 فصاعدًا، ستتم إزالة STLport وnuntl،
ما يجعل libc++ هو STL الوحيد في NDK.
يمكنك الاطّلاع على مستندات NDK للحصول على مزيد من المعلومات حول هذه القواعد.
ضبط واجهة تطبيق ثنائية (ABI) المستهدفة
لضبط واجهة التطبيق الثنائية (ABI) المُستهدَفة، استخدِم علامة --fat_apk_cpu على النحو التالي:
bazel build //:app --fat_apk_cpu=comma-separated list of ABIs
بشكل تلقائي، ينشئ Bazel رمز Android الأصلي للإصدار armeabi-v7a. للإصدار x86
(على سبيل المثال للمحاكيات)، مرِّر --fat_apk_cpu=x86. لإنشاء ملف APK يتضمن الكثير من البُنى الأساسية، يمكنك تحديد وحدات المعالجة المركزية (CPU) متعددة: --fat_apk_cpu=armeabi-v7a,x86.
في حال تحديد أكثر من واجهة تطبيق ثنائية (ABI)، ستُنشئ Bazel ملف APK يحتوي على عنصر مشترك لكل واجهة تطبيق ثنائية (ABI).
واستنادًا إلى إصدار NDK ومستوى واجهة برمجة تطبيقات Android، تتوفر واجهات التطبيق الثنائية (ABI) التالية:
| مراجعة NDK | قيم ABI |
|---|---|
| 16 وأقل | armeabi وarmeabi-v7a وar64-v8a وmips وmips64 وx86 وx86_64 |
| 17 سنة وما فوق | armeabi-v7a وar64-v8a وx86 وx86_64 |
راجع مستندات NDK لمزيد من المعلومات حول واجهات التطبيق الثنائية هذه.
ولا يُنصح باستخدام ملفات APK للدهون المتعددة واجهة التطبيق الثنائية (ABI) لإصدارات الإصدار لأنها تزيد حجم حزمة APK، ولكنها قد تكون مفيدة لإصدارات تأكيد الجودة والتطوير.
اختيار معيار C++
ويمكنك استخدام العلامات التالية لإنشاء المحتوى وفقًا لمعيار C++:
| C++ Standard | إبلاغ |
|---|---|
| C++98 | تلقائيًا، لا حاجة إلى الإبلاغ |
| C++11 | --cxxopt=-std=c++11 |
| C++14 | --cxxopt=-std=c++14 |
مثلاً:
bazel build //:app --cxxopt=-std=c++11
يمكنك الاطّلاع على المزيد من المعلومات عن تمرير علامات برامج التجميع والربط باستخدام --cxxopt و--copt
و--linkopt في دليل المستخدم.
يمكن أيضًا تحديد علامات برامج التجميع والربط كسمات في cc_library
باستخدام copts وlinkopts. مثلاً:
cc_library(
name = "jni_lib",
srcs = ["cpp/native-lib.cpp"],
copts = ["-std=c++11"],
linkopts = ["-ldl"], # link against libdl
)
التكامل مع المنصات وسلاسل الأدوات
ينتقل نموذج إعداد Bazel's إلى
المنصات و
سلاسل الأدوات. إذا كان الإصدار الذي تستخدمه يستخدم علامة --platforms لتحديد البنية أو نظام التشغيل الذي سيتم إنشاء العلامة من أجله، ستحتاج إلى تمرير علامة --extra_toolchains إلى Bazel لاستخدام NDK.
على سبيل المثال، للدمج مع سلسلة الأدوات android_arm64_cgo التي توفّرها قواعد Go، عليك اجتياز --extra_toolchains=@androidndk//:all بالإضافة إلى علامة --platforms.
bazel build //my/cc:lib \
--platforms=@io_bazel_rules_go//go/toolchain:android_arm64_cgo \
--extra_toolchains=@androidndk//:all
ويمكنك أيضًا تسجيلها مباشرةً في ملف WORKSPACE:
android_ndk_repository(name = "androidndk")
register_toolchains("@androidndk//:all")
يؤدّي تسجيل سلاسل الأدوات هذه إلى مطالبة شركة Bazel بالبحث عنها في ملف BUILD NDK (بالنسبة إلى NDK 20) عند حلّ المشاكل في البنية ونظام التشغيل:
toolchain(
name = "x86-clang8.0.7-libcpp_toolchain",
toolchain_type = "@bazel_tools//tools/cpp:toolchain_type",
target_compatible_with = [
"@platforms//os:android",
"@platforms//cpu:x86_32",
],
toolchain = "@androidndk//:x86-clang8.0.7-libcpp",
)
toolchain(
name = "x86_64-clang8.0.7-libcpp_toolchain",
toolchain_type = "@bazel_tools//tools/cpp:toolchain_type",
target_compatible_with = [
"@platforms//os:android",
"@platforms//cpu:x86_64",
],
toolchain = "@androidndk//:x86_64-clang8.0.7-libcpp",
)
toolchain(
name = "arm-linux-androideabi-clang8.0.7-v7a-libcpp_toolchain",
toolchain_type = "@bazel_tools//tools/cpp:toolchain_type",
target_compatible_with = [
"@platforms//os:android",
"@platforms//cpu:arm",
],
toolchain = "@androidndk//:arm-linux-androideabi-clang8.0.7-v7a-libcpp",
)
toolchain(
name = "aarch64-linux-android-clang8.0.7-libcpp_toolchain",
toolchain_type = "@bazel_tools//tools/cpp:toolchain_type",
target_compatible_with = [
"@platforms//os:android",
"@platforms//cpu:aarch64",
],
toolchain = "@androidndk//:aarch64-linux-android-clang8.0.7-libcpp",
)
كيفية تنفيذ ذلك: تقديم عمليات انتقال إعداد Android
يمكن أن تطلب قاعدة android_binary صراحةً من Bazel إنشاء تبعياتها في ضبط متوافق مع Android بحيث يعمل إصدار Bazel فقط بدون أي علامات خاصة، باستثناء --fat_apk_cpu و--android_crosstool_top لإعداد ABI وSTL.
وراء الكواليس، يستخدم هذا الإعداد التلقائي نظام التشغيل Android انتقالات الضبط.
تُغيِّر القاعدة المتوافقة، مثل android_binary، تلقائيًا ضبط تبعياتها باعتماد إصدار Android، بحيث تتأثّر فقط الإصدارات الفرعية المخصّصة للإصدار من Android. تتم معالجة الأجزاء الأخرى من الرسم البياني للإصدار باستخدام إعداد الاستهداف ذي المستوى الأعلى. ويمكنه أيضًا معالجة استهداف واحد في كلا الإعدادين، إذا كانت هناك مسارات خلال الرسم البياني للإصدار لدعم ذلك.
بعد ضبط Bazel لإعدادات الضبط المتوافق مع Android، سواء تم تحديدها على المستوى العلوي أو بسبب نقطة انتقال على مستوى أعلى، لن تجري نقاط النقل الإضافية التي تم إجراء تعديلات عليها عملية الضبط.
والموقع الوحيد المدمَج الذي يؤدي إلى الانتقال إلى إعداد Android
هو سمة deps android_binary.
على سبيل المثال، إذا كنت تحاول إنشاء استهداف android_library باستخدام تبعية cc_library بدون أي علامات، قد يظهر لك خطأ بشأن عنوان JNI غير متوفر:
ERROR: project/app/src/main/BUILD.bazel:16:1: C++ compilation of rule '//app/src/main:jni_lib' failed (Exit 1)
app/src/main/cpp/native-lib.cpp:1:10: fatal error: 'jni.h' file not found
#include <jni.h>
^~~~~~~
1 error generated.
Target //app/src/main:lib failed to build
Use --verbose_failures to see the command lines of failed build steps.
من المفترض أن تؤدي عمليات النقل التلقائية هذه إلى اتخاذ Bazel الإجراء المناسب في معظم الحالات. وإذا كان الهدف في سطر أوامر Bazel مضمّنًا فعلاً أسفل أي من قواعد النقل هذه، مثل مطوّري برامج C++ الذين يختبرون cc_library، يجب استخدام --crosstool_top مخصّص.
إنشاء cc_library لنظام التشغيل Android بدون استخدام android_binary
لإنشاء علامة cc_binary أو cc_library مستقلّة لنظام التشغيل Android بدون استخدام
android_binary، يمكنك استخدام علامات --crosstool_top و--cpu و--host_crosstool_top.
مثلاً:
bazel build //my/cc/jni:target \
--crosstool_top=@androidndk//:default_crosstool \
--cpu=<abi> \
--host_crosstool_top=@bazel_tools//tools/cpp:toolchain
في هذا المثال، تم إنشاء هدفَي المستوى cc_library وcc_binary من المستوى الأعلى باستخدام سلسلة أدوات NDK. مع ذلك، يؤدّي ذلك إلى إنشاء أدوات مضيفة لـ Bazel&hl=ar باستخدام سلسلة أدوات NDK (وبالتالي لنظام التشغيل Android)، لأنّ سلسلة أدوات المضيف يتم نسخها من سلسلة الأدوات المستهدفة. وللتغلب على ذلك، حدد قيمة
--host_crosstool_top لتكون @bazel_tools//tools/cpp:toolchain
لتحديد سلسلة أدوات المضيف +C بشكل صريح.
بهذه الطريقة، تتأثر شجرة الإصدار بالكامل.
ويمكن وضع هذه العلامات في إعداد bazelrc (واحدة لكل واجهة تطبيق ثنائية)، في
project/.bazelrc:
common:android_x86 --crosstool_top=@androidndk//:default_crosstool
common:android_x86 --cpu=x86
common:android_x86 --host_crosstool_top=@bazel_tools//tools/cpp:toolchain
common:android_armeabi-v7a --crosstool_top=@androidndk//:default_crosstool
common:android_armeabi-v7a --cpu=armeabi-v7a
common:android_armeabi-v7a --host_crosstool_top=@bazel_tools//tools/cpp:toolchain
# In general
common:android_<abi> --crosstool_top=@androidndk//:default_crosstool
common:android_<abi> --cpu=<abi>
common:android_<abi> --host_crosstool_top=@bazel_tools//tools/cpp:toolchain
بعد ذلك، لإنشاء cc_library للسمة x86، على سبيل المثال، يمكنك تشغيل:
bazel build //my/cc/jni:target --config=android_x86
بشكلٍ عام، استخدِم هذه الطريقة للأهداف المنخفضة المستوى (مثل cc_library) أو عندما تكون على دراية تامة بكيفية إنشاء المحتوى، ويمكنك الاعتماد على عمليات النقل التلقائية للإعدادات من android_binary للأهداف العالية المستوى التي تتوقّع إنشاء عدد كبير من الاستهدافات التي لا تتحكم فيها.