当前位置:   article > 正文

Android Studio 进行NDK开发,实现JNI,以及编写C++与Java交互(Java调用本地函数)并编译出本地so动态库_android studio ndk

android studio ndk

1.首先认识一下NDK

(1)什么是NDK?
NDK全称是Native Development Kit,NDK提供了一系列的工具,帮助开发者快速开发C/C++的动态库,并能自动将so和java应用一起打包成apk。NDK集成了交叉编译器(交叉编译器需要UNIX或LINUX系统环境),并提供了相应的mk文件隔离CPU、平台、ABI等差异,开发人员只需要简单修改mk文件(指出“哪些文件需要编译”、“编译特性要求”等),就可以创建出so。

(2)为什么使用NDK?
1)代码的保护:由于apk的java层代码很容易被反编译,而C/C++库反汇难度较大。
2)可以方便地使用现存的开源库:大部分现存的开源库都是用C/C++代码编写的。
3)提高程序的执行效率:将要求高性能的应用逻辑使用C开发,从而提高应用程序的执行效率。
4)便于移植:用C/C++写得库可以方便在其他的嵌入式平台上再次使用。

(3)什么是JNI?
JNI全称为:Java Native Interface。JNI是本地编程接口,它使得在 Java 虚拟机内部运行的 Java代码能够与用其它语言(如 C、C++)编写的代码进行交互。

(4)为什么使用JNI?
JNI的目的是使java方法能够调用C/C++实现的一些函数。

(5)安卓中的so文件是什么?
Android中用到的so文件是一个C/C++的函数库。在android的JNI中,要先将相应的C/C++打包成so库,然后导入到lib文件夹中供java调用。

2.Android Studio 配置NDK(使用Android Studio 4.2.2之后的稳定版本)

(1) 步骤一:点击红圈处(这是Mac配置流程,Windows对应的按钮是Settings),如下图一

(2)步骤二:下载下图一中第3步红圈中的一个NDK和一个CMake,下载成功后如下图一所示(建议下载前先配置Android Studio 国内镜像代理,详见:Android Studio 国内镜像代理设置(如果设置之后还是远程仓库下载失败,请仔细阅读其内容就可以解决了)_android studio 镜像_ErwinNakajima的博客-CSDN博客)。

3.开始开发,在main文件夹下面创建一个cpp文件夹,如下图一和下圖二,然后在cpp文件夹下创建native-lib.cpp、return-data.cpp和CMakeLists.txt,然后添加具体内容;

native-lib.cpp文件的内容;

  1. //
  2. // Created by ErwinNakashima on 2022/12/24.
  3. //
  4. #include <jni.h>
  5. #include <stdio.h>
  6. #include <string.h>
  7. #include <string>
  8. static jclass contextClass;
  9. static jclass signatureClass;
  10. static jclass packageNameClass;
  11. static jclass packageInfoClass;
  12. /**
  13. 之前生成好的签名字符串
  14. */
  15. const char *SIGN_RELEASE = "308203b130820299a00302010202045fa239d4300d06092a864886f70d01010b0500308188310c300a06035504061303555341311c301a060355040813135374617465206f662043616c69666f726e69613112301006035504071309437570657274696e6f310e300c060355040a13056170706c65310e300c060355040b13056170706c653126302406035504030c1d72786a617661325f616e645f726574726f666974325f72656c65617365301e170d3232313232313133333232365a170d3437313231353133333232365a308188310c300a06035504061303555341311c301a060355040813135374617465206f662043616c69666f726e69613112301006035504071309437570657274696e6f310e300c060355040a13056170706c65310e300c060355040b13056170706c653126302406035504030c1d72786a617661325f616e645f726574726f666974325f72656c6561736530820122300d06092a864886f70d01010105000382010f003082010a02820101009f09581713e084950d06b016bf86063343864fe59952c335f3bb44e14ed67686749dcd372f075d2d74757c18dff9082941cbc7c83c60c3ad51d69ca503826775b2b703a6c347d70b8364b5f140cf7e1025590c8dcfa1469d2b5af242323b6c7bcddf92fb44aaa82a7c735597112964432fc16bb6dbf360f2a44d0e9e9722295b070582ea72310c674aed8ef8aa24ec06e972edafcae51d93c7370cfbc3e804fda3cc6f22ec89f98dac2ceb5607ef564fd3151091d2e0c142b984a21c61bc63b75e0bdc931dca1a9cc76f1ee326d59ef6edeb1d5dd07dd12fa32e55de3572784e8adc67388d643b310560f77e75ec944e00e6ae62d283c90c89eae5ce71747a9f0203010001a321301f301d0603551d0e041604147387f3952464b66ce5bb906d56845bc4410d8bc1300d06092a864886f70d01010b050003820101003ca9530822c2f272bcfb94dc2552045db8d4038385fbac917e08266f6f47b00ba36a735fc62da0c2d4bbe8fcfaec0d87c7c6223c257a22240f69057f954d90fda7c7dfe4daa2f3fa482aa1a35b56c1220b449115a670324408ae9f4f6dfa3af40b9c55275c27785bcfeb1337c7228ca2deac5c9e5b4fabd33e77f3fab0c18df0facfd23980a037907acd215c11a450d98789f002081379a688686b23b3aec1fbf4e3bf1db0e34daac5140e60ad412c11c1717c3befb83ca5878d1f5b199f6f4fee89591c9dbcec13a340c7aa817ab4d68b19598f57e60b08e950ba2843d5400b576511d8b4a0ac45accf92d5c82f0d9afc11bce5c2d58ae4f3f8e9da604e83c7";
  16. const char *SIGN_DEBUG = "308202e4308201cc020101300d06092a864886f70d010105050030373116301406035504030c0d416e64726f69642044656275673110300e060355040a0c07416e64726f6964310b30090603550406130255533020170d3231303830363136353432345a180f32303531303733303136353432345a30373116301406035504030c0d416e64726f69642044656275673110300e060355040a0c07416e64726f6964310b300906035504061302555330820122300d06092a864886f70d01010105000382010f003082010a02820101008b2616bebb0dd79ea6e0688ce89e5ae221e3132ee9b6cc96514119557f0be3e79371e22c3c91250988a833942714ae562e8a86d6fff4290cb7b7bf718aeacd480d2cffe38c9787218e6391e562843b95dd26642b24e2106694501f0fe39186bf5fcc5c3cca91b9d86c113ffb0acf6e0e6ac9a4cda01110c5f18729d8f2f091b9fb604595a492ddcad6ae71dd190672cd8a675483563d5a734f9ec040890456ee02a32b61ccfc61811c8311b61eb90ffb15fae0db04f52c562b3713781fd772331619a4670065ed574e96da2cf47e7c4b29af30d5bbc1e271f23f3ea33b1085bb228e44d948d1f2adb0c71ee1c2652fe5b554d5e8e430c68f35b090f7d6dccbb10203010001300d06092a864886f70d010105050003820101004f1fd6247b615c2216e23eb8fe38e20282e9d5742b9485fec941fa541c97203eb60e3419fd6742d50bd2d60274d8489d1c03ab87f604aa2632aebdb2c7cc46e42f9f6dfec32155cca601fcf4abb3724068ccda637aa11c22d361afe9ec91b0d15209a9121c849aef791ceb670052e943891c34c0d380947f442ff93a93e8c6ac594d003f40ee0880dd0a0742ad1aa5c18f692b6480c3cf3baf42f5bacd8f31e811e88e98c187da52d4ed74aeaadb5f5f2c8b99c63612ce5abf4532151bcc4f3cab9b320c12b5c5e2c7fb6a69e72d6d1acdb43415dcecf9737ed124f28850d9e691cdb03a17c6c62a51fbd5c460067f3f890df085c4a849c05b061062d2aab16c";
  17. std::string hello = "(*^__^*) 嘻嘻……~Hello from C++ 特朗普的头发是黄色的";
  18. /*
  19. 根据context对象,获取签名字符串
  20. */
  21. const char *getSignString(JNIEnv *env, jobject contextObject) {
  22. jmethodID getPackageManagerId = (env)->GetMethodID(contextClass, "getPackageManager",
  23. "()Landroid/content/pm/PackageManager;");
  24. jmethodID getPackageNameId = (env)->GetMethodID(contextClass, "getPackageName",
  25. "()Ljava/lang/String;");
  26. jmethodID signToStringId = (env)->GetMethodID(signatureClass, "toCharsString",
  27. "()Ljava/lang/String;");
  28. jmethodID getPackageInfoId = (env)->GetMethodID(packageNameClass, "getPackageInfo",
  29. "(Ljava/lang/String;I)Landroid/content/pm/PackageInfo;");
  30. jobject packageManagerObject = (env)->CallObjectMethod(contextObject, getPackageManagerId);
  31. jstring packNameString = (jstring) (env)->CallObjectMethod(contextObject, getPackageNameId);
  32. jobject packageInfoObject = (env)->CallObjectMethod(packageManagerObject, getPackageInfoId,
  33. packNameString, 64);
  34. jfieldID signaturefieldID = (env)->GetFieldID(packageInfoClass, "signatures",
  35. "[Landroid/content/pm/Signature;");
  36. jobjectArray signatureArray = (jobjectArray) (env)->GetObjectField(packageInfoObject,
  37. signaturefieldID);
  38. jobject signatureObject = (env)->GetObjectArrayElement(signatureArray, 0);
  39. return (env)->GetStringUTFChars(
  40. (jstring) (env)->CallObjectMethod(signatureObject, signToStringId), 0);
  41. }
  42. extern "C"
  43. JNIEXPORT jstring JNICALL
  44. Java_com_phone_library_1common_JavaGetData_nativeGetString(JNIEnv *env, jclass clazz,
  45. jobject context, jboolean is_release) {
  46. const char *signStrng = getSignString(env, context);
  47. bool isRelease = is_release;
  48. const char *SIGN;
  49. if (isRelease) {
  50. SIGN = SIGN_RELEASE;
  51. } else {
  52. SIGN = SIGN_DEBUG;
  53. }
  54. if (strcmp(signStrng, SIGN) == 0)//签名一致 返回合法的 api key,否则返回错误
  55. {
  56. return (env)->NewStringUTF(hello.c_str());
  57. } else {
  58. return (env)->NewStringUTF("error");
  59. }
  60. }
  61. //bool toCppBool(jboolean value) {
  62. // return value == JNI_TRUE;
  63. //}
  64. /**
  65. 利用OnLoad钩子,初始化需要用到的Class类.
  66. */
  67. JNIEXPORT jint
  68. JNICALL JNI_OnLoad(JavaVM *vm, void *reserved) {
  69. JNIEnv *env = NULL;
  70. jint result = -1;
  71. if (vm->GetEnv((void **) &env, JNI_VERSION_1_4) != JNI_OK)
  72. return result;
  73. contextClass = (jclass) env->NewGlobalRef((env)->FindClass("android/content/Context"));
  74. signatureClass = (jclass) env->NewGlobalRef((env)->FindClass("android/content/pm/Signature"));
  75. packageNameClass = (jclass) env->NewGlobalRef(
  76. (env)->FindClass("android/content/pm/PackageManager"));
  77. packageInfoClass = (jclass) env->NewGlobalRef(
  78. (env)->FindClass("android/content/pm/PackageInfo"));
  79. return JNI_VERSION_1_4;
  80. }

return-data.cpp文件的内容;

  1. #include <jni.h>
  2. #include <stdio.h>
  3. #include <string.h>
  4. #ifdef __cplusplus
  5. extern "C" {
  6. #endif
  7. static jclass contextClass;
  8. static jclass signatureClass;
  9. static jclass packageNameClass;
  10. static jclass packageInfoClass;
  11. /**
  12. 之前生成好的签名字符串
  13. */
  14. const char *SIGN_RELEASE = "308203b130820299a00302010202045fa239d4300d06092a864886f70d01010b0500308188310c300a06035504061303555341311c301a060355040813135374617465206f662043616c69666f726e69613112301006035504071309437570657274696e6f310e300c060355040a13056170706c65310e300c060355040b13056170706c653126302406035504030c1d72786a617661325f616e645f726574726f666974325f72656c65617365301e170d3232313232313133333232365a170d3437313231353133333232365a308188310c300a06035504061303555341311c301a060355040813135374617465206f662043616c69666f726e69613112301006035504071309437570657274696e6f310e300c060355040a13056170706c65310e300c060355040b13056170706c653126302406035504030c1d72786a617661325f616e645f726574726f666974325f72656c6561736530820122300d06092a864886f70d01010105000382010f003082010a02820101009f09581713e084950d06b016bf86063343864fe59952c335f3bb44e14ed67686749dcd372f075d2d74757c18dff9082941cbc7c83c60c3ad51d69ca503826775b2b703a6c347d70b8364b5f140cf7e1025590c8dcfa1469d2b5af242323b6c7bcddf92fb44aaa82a7c735597112964432fc16bb6dbf360f2a44d0e9e9722295b070582ea72310c674aed8ef8aa24ec06e972edafcae51d93c7370cfbc3e804fda3cc6f22ec89f98dac2ceb5607ef564fd3151091d2e0c142b984a21c61bc63b75e0bdc931dca1a9cc76f1ee326d59ef6edeb1d5dd07dd12fa32e55de3572784e8adc67388d643b310560f77e75ec944e00e6ae62d283c90c89eae5ce71747a9f0203010001a321301f301d0603551d0e041604147387f3952464b66ce5bb906d56845bc4410d8bc1300d06092a864886f70d01010b050003820101003ca9530822c2f272bcfb94dc2552045db8d4038385fbac917e08266f6f47b00ba36a735fc62da0c2d4bbe8fcfaec0d87c7c6223c257a22240f69057f954d90fda7c7dfe4daa2f3fa482aa1a35b56c1220b449115a670324408ae9f4f6dfa3af40b9c55275c27785bcfeb1337c7228ca2deac5c9e5b4fabd33e77f3fab0c18df0facfd23980a037907acd215c11a450d98789f002081379a688686b23b3aec1fbf4e3bf1db0e34daac5140e60ad412c11c1717c3befb83ca5878d1f5b199f6f4fee89591c9dbcec13a340c7aa817ab4d68b19598f57e60b08e950ba2843d5400b576511d8b4a0ac45accf92d5c82f0d9afc11bce5c2d58ae4f3f8e9da604e83c7";
  15. const char *SIGN_DEBUG = "308202e4308201cc020101300d06092a864886f70d010105050030373116301406035504030c0d416e64726f69642044656275673110300e060355040a0c07416e64726f6964310b30090603550406130255533020170d3231303830363136353432345a180f32303531303733303136353432345a30373116301406035504030c0d416e64726f69642044656275673110300e060355040a0c07416e64726f6964310b300906035504061302555330820122300d06092a864886f70d01010105000382010f003082010a02820101008b2616bebb0dd79ea6e0688ce89e5ae221e3132ee9b6cc96514119557f0be3e79371e22c3c91250988a833942714ae562e8a86d6fff4290cb7b7bf718aeacd480d2cffe38c9787218e6391e562843b95dd26642b24e2106694501f0fe39186bf5fcc5c3cca91b9d86c113ffb0acf6e0e6ac9a4cda01110c5f18729d8f2f091b9fb604595a492ddcad6ae71dd190672cd8a675483563d5a734f9ec040890456ee02a32b61ccfc61811c8311b61eb90ffb15fae0db04f52c562b3713781fd772331619a4670065ed574e96da2cf47e7c4b29af30d5bbc1e271f23f3ea33b1085bb228e44d948d1f2adb0c71ee1c2652fe5b554d5e8e430c68f35b090f7d6dccbb10203010001300d06092a864886f70d010105050003820101004f1fd6247b615c2216e23eb8fe38e20282e9d5742b9485fec941fa541c97203eb60e3419fd6742d50bd2d60274d8489d1c03ab87f604aa2632aebdb2c7cc46e42f9f6dfec32155cca601fcf4abb3724068ccda637aa11c22d361afe9ec91b0d15209a9121c849aef791ceb670052e943891c34c0d380947f442ff93a93e8c6ac594d003f40ee0880dd0a0742ad1aa5c18f692b6480c3cf3baf42f5bacd8f31e811e88e98c187da52d4ed74aeaadb5f5f2c8b99c63612ce5abf4532151bcc4f3cab9b320c12b5c5e2c7fb6a69e72d6d1acdb43415dcecf9737ed124f28850d9e691cdb03a17c6c62a51fbd5c460067f3f890df085c4a849c05b061062d2aab16c";
  16. const char *aesKey = "rxjava_and_re_ro";
  17. const char *databaseEncryptKey = "Aa123456";
  18. /*
  19. 根据context对象,获取签名字符串
  20. */
  21. const char *getSignString(JNIEnv *env, jobject contextObject) {
  22. jmethodID getPackageManagerId = (env)->GetMethodID(contextClass, "getPackageManager",
  23. "()Landroid/content/pm/PackageManager;");
  24. jmethodID getPackageNameId = (env)->GetMethodID(contextClass, "getPackageName",
  25. "()Ljava/lang/String;");
  26. jmethodID signToStringId = (env)->GetMethodID(signatureClass, "toCharsString",
  27. "()Ljava/lang/String;");
  28. jmethodID getPackageInfoId = (env)->GetMethodID(packageNameClass, "getPackageInfo",
  29. "(Ljava/lang/String;I)Landroid/content/pm/PackageInfo;");
  30. jobject packageManagerObject = (env)->CallObjectMethod(contextObject, getPackageManagerId);
  31. jstring packNameString = (jstring) (env)->CallObjectMethod(contextObject, getPackageNameId);
  32. jobject packageInfoObject = (env)->CallObjectMethod(packageManagerObject, getPackageInfoId,
  33. packNameString, 64);
  34. jfieldID signaturefieldID = (env)->GetFieldID(packageInfoClass, "signatures",
  35. "[Landroid/content/pm/Signature;");
  36. jobjectArray signatureArray = (jobjectArray) (env)->GetObjectField(packageInfoObject,
  37. signaturefieldID);
  38. jobject signatureObject = (env)->GetObjectArrayElement(signatureArray, 0);
  39. return (env)->GetStringUTFChars(
  40. (jstring) (env)->CallObjectMethod(signatureObject, signToStringId), 0);
  41. }
  42. extern "C"
  43. JNIEXPORT jstring JNICALL
  44. Java_com_phone_library_1common_JavaGetData_nativeAesKey(JNIEnv *env, jclass clazz, jobject context,
  45. jboolean is_release) {
  46. const char *signStrng = getSignString(env, context);
  47. bool isRelease = is_release;
  48. const char *SIGN;
  49. if (isRelease) {
  50. SIGN = SIGN_RELEASE;
  51. } else {
  52. SIGN = SIGN_DEBUG;
  53. }
  54. if (strcmp(signStrng, SIGN) == 0)//签名一致 返回合法的 api key,否则返回错误
  55. {
  56. return (env)->NewStringUTF(aesKey);
  57. } else {
  58. return (env)->NewStringUTF("error");
  59. }
  60. }
  61. extern "C"
  62. JNIEXPORT jstring JNICALL
  63. Java_com_phone_library_1common_JavaGetData_nativeDatabaseEncryptKey(JNIEnv *env, jclass clazz,
  64. jobject context,
  65. jboolean is_release) {
  66. const char *signStrng = getSignString(env, context);
  67. bool isRelease = is_release;
  68. const char *SIGN;
  69. if (isRelease) {
  70. SIGN = SIGN_RELEASE;
  71. } else {
  72. SIGN = SIGN_DEBUG;
  73. }
  74. if (strcmp(signStrng, SIGN) == 0)//签名一致 返回合法的 api key,否则返回错误
  75. {
  76. return (env)->NewStringUTF(databaseEncryptKey);
  77. } else {
  78. return (env)->NewStringUTF("error");
  79. }
  80. }
  81. /**
  82. 利用OnLoad钩子,初始化需要用到的Class类.
  83. */
  84. JNIEXPORT jint
  85. JNICALL JNI_OnLoad(JavaVM *vm, void *reserved) {
  86. JNIEnv *env = NULL;
  87. jint result = -1;
  88. if (vm->GetEnv((void **) &env, JNI_VERSION_1_4) != JNI_OK)
  89. return result;
  90. contextClass = (jclass) env->NewGlobalRef((env)->FindClass("android/content/Context"));
  91. signatureClass = (jclass) env->NewGlobalRef((env)->FindClass("android/content/pm/Signature"));
  92. packageNameClass = (jclass) env->NewGlobalRef(
  93. (env)->FindClass("android/content/pm/PackageManager"));
  94. packageInfoClass = (jclass) env->NewGlobalRef(
  95. (env)->FindClass("android/content/pm/PackageInfo"));
  96. return JNI_VERSION_1_4;
  97. }
  98. #ifdef __cplusplus
  99. }
  100. #endif

CMakeLists.txt的內容。

  1. # For more information about using CMake with Android Studio, read the
  2. # documentation: https://d.android.com/studio/projects/add-native-code.html
  3. # Sets the minimum version of CMake required to build the native library.
  4. cmake_minimum_required(VERSION 3.10.2)
  5. # Creates and names a library, sets it as either STATIC
  6. # or SHARED, and provides the relative paths to its source code.
  7. # You can define multiple libraries, and CMake builds them for you.
  8. # Gradle automatically packages shared libraries with your APK.
  9. #设置so库的输出路径
  10. set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${PROJECT_SOURCE_DIR}/libs/${ANDROID_ABI})
  11. add_library( # Sets the name of the library.
  12. native-lib
  13. # Sets the library as a shared library.
  14. SHARED
  15. # Provides a relative path to your source file(s).
  16. native-lib.cpp)
  17. add_library( # Sets the name of the library.
  18. return-data
  19. # Sets the library as a shared library.
  20. SHARED
  21. # Provides a relative path to your source file(s).
  22. return-data.cpp)
  23. # Searches for a specified prebuilt library and stores the path as a
  24. # variable. Because CMake includes system libraries in the search path by
  25. # default, you only need to specify the name of the public NDK library
  26. # you want to add. CMake verifies that the library exists before
  27. # completing its build.
  28. find_library( # Sets the name of the path variable.
  29. log-lib
  30. # Specifies the name of the NDK library that
  31. # you want CMake to locate.
  32. log)
  33. # Specifies libraries CMake should link to your target library. You
  34. # can link multiple libraries, such as libraries you define in this
  35. # build script, prebuilt third-party libraries, or system libraries.
  36. target_link_libraries( # Specifies the target library.
  37. native-lib
  38. return-data
  39. # Links the target library to the log library
  40. # included in the NDK.
  41. ${log-lib})

4.配置ndk,在module的build.gradle文件下的android下配置;

  1. sourceSets {
  2. main {
  3. jniLibs.srcDirs = ['libs']
  4. }
  5. }
  6. ndkVersion '16.1.4479499'
  7. externalNativeBuild {
  8. cmake {
  9. path file('src/main/cpp/CMakeLists.txt')
  10. version '3.10.2'
  11. }
  12. }

module的build.gradle文件下的defaultConfig下配置;

  1. ndk {
  2. //选择要添加的对应 cpu 类型的 .so 库。
  3. abiFilters 'armeabi-v7a', 'arm64-v8a'
  4. // abiFilters 'armeabi-v7a', 'arm64-v8a'
  5. // 还可以添加 'x86', 'x86_64', 'mips', 'mips64'
  6. }

項目的gradle.properties文件下配置,如下图一;

android.useDeprecatedNdk=true

 項目的local.properties 文件下配置,如下图一。

ndk.dir=/Users/erwinnakashima/Library/Android/sdk/ndk/16.1.4479499

 5.添加JavaGetData文件,JavaGetData文件内容;

  1. package com.phone.library_common;
  2. import android.content.Context;
  3. public class JavaGetData {
  4. static {
  5. System.loadLibrary("return-data");
  6. System.loadLibrary("native-lib");
  7. }
  8. public static native String nativeAesKey(Context context, boolean isRelease);
  9. public static native String nativeDatabaseEncryptKey(Context context, boolean isRelease);
  10. public static native String nativeGetString(Context context, boolean isRelease);
  11. }

在Application的onCreate方法中调用JavaGetData.nativeAesKey(),然后ReBuild Project。就会生成两个文件这几个so文件,如下图一,然后就能正常获取到C++文件(也就是cpp文件)中的数据了,还可以把so动态库提供给第三方使用,具体方式详见Android 项目调用第三方库so动态库_ErwinNakajima的博客-CSDN博客

如对此有疑问,请联系qq1164688204。

推荐Android开源项目

项目功能介绍:原本是RxJava2 和Retrofit2 项目,现已更新使用Kotlin+RxJava2+Retrofit2+MVP架构+组件化和
Kotlin+Retrofit2+协程+MVVM架构+组件化, 添加自动管理token 功能,添加RxJava2 生命周期管理,集成极光推送、阿里云Oss对象存储和高德地图定位功能。

项目地址:https://gitee.com/urasaki/RxJava2AndRetrofit2


 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/繁依Fanyi0/article/detail/109816?site
推荐阅读
相关标签
  

闽ICP备14008679号