1

我希望我的应用程序支持任何 Android 手机 api 3 及更高版本。

为什么?因为我不喜欢浪费,而这些旧手机完全可以胜任手头的任务。

遗憾的是,事实证明,获取有关基本 pre-api-9 SDK 构建的信息远比让 NDK 正常工作要难得多。

4

1 回答 1

2

我找不到使用 developer.android.com 上提供的任何 Android Studio 版本构建 Android 应用程序 api < 9 的机制。

但是,可以使用较旧的命令行工具来构建它们。(我没有研究过 api < 9 使用 ActivityCompat 库或 ndk-builds 来调试 JNI 应用程序——这些留给读者作为练习:-)

注意:使用这些技术需要熟悉当前的命令行构建,否则就没有什么意义了。

该任务分为四个:

api < 9 的 Android 示例项目

https://android.googlesource.com/platform/development/+refs

拥有从 donut-release 到 android-s-beta-4 的一切。
每个都有一个“samples”子目录,这是一个很好的起点。

用于构建 api < 9 个项目的 Android 工具

https://dl.google.com/android/repository/repository-10.xml

列出所有下载及其摘要/校验和,位于:

https://dl.google.com/android/repository/(repo-filename)

您需要找到、下载并安装:

  • 平台 SDK 3
  • build-tools 17.0.0(包含 aapt 的最新版本)
  • build-tools 24.0.3(包含 apksigner 的最旧版本)
  • ndk R11C(支持 JNI for apis 3 到 8 的最新版本)

为 api < 9 构建 Android JNI 库

请注意,api < 4 的设备似乎更喜欢从设备操作系统链接共享符号,而不是从您的 JNI 库中链接,因此请避免从您的库中发布冲突符号并静态链接任何第三方库(许可允许)。

示例生成文件:

API = 3
NDK = ~/Android/Sdk/ndk/r11c
CC = $(NDK)/toolchains/arm-linux-androideabi-4.9/prebuilt/linux-x86_64/bin/arm-linux-androideabi-gcc
SR = $(NDK)/platforms/android-$(API)/arch-arm
INC = -I$(JAVA_HOME)/include -I$(JAVA_HOME)/include/linux

objs = obj/thing1.o obj/thing2.o ...

libexample.so : $(objs)
    $(CC) --sysroot=$(SR) -fPIC -Wall -shared -o libexample.so -O $^

obj/%.o : %.c
    $(CC) --sysroot=$(SR) -fPIC -Wall -mthumb -c $< -o $@ $(INC)

如果您正在构建 openssl,您将需要一个额外的配置标志:

--with-rand-seed=devrandom

以及无线程无asm。

为 api < 9 构建 Android Java 应用程序

遗憾的是,我无法找到一种方法来使用当今的 Android Studio 来构建这些低 api 项目,所以下面是一些讨厌的命令行东西来代替它。

您需要找到、下载并安装:

  • 平台 SDK 3
  • build-tools 17.0.0(包含 aapt 的最新版本)
  • build-tools 24.0.3(包含 apksigner 的最旧版本)

(有关如何获得这些内容,请参见上文。)

项目树

project/
    Makefile
    app/
        build/
        src/
            main/
                AndroidManifest.xml
                assets/
                jniLibs/
                    armeabi/
                        libexample.so
                res/
                    layout/
                        activity_main.xml
                java/
                    com/
                        example/
                            project/
                                MainActivity.java

AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.project">
    <application
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        >
        <uses-sdk android:minSdkVersion="3" />
        <activity android:name="com.example.project.MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN"/>
                <category android:name="android.intent.category.LAUNCHER"/>
            </intent-filter>
        </activity>
    </application>
</manifest>

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    tools:context="com.example.project.MainActivity"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:id="@+id/outer"
    android:orientation="vertical"    
    >
    <Button
        android:id="@+id/btn"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="run"
        /><!--note: no onClick-->
    ...

MainActivity.java

package com.example.project;
import android.os.Bundle;
import android.app.Activity;
import android.widget.Button;
public class MainActivity extends Activity {
    @Override protected void onCreate(Bundle bdl) {
        super.onCreate(bdl);
        setContentView(R.layout.activity_main);        
        final Button btn = (Button)findViewById(R.id.btn);
        btn.setOnClickListener(new Button.OnClickListener() {
            public void onClick(View v) {
                ...
            }
        });
    }
};

生成文件

API = 3
KEY_PATH = ~/jkeystore
DROID_HOME = ~/Android/Sdk
JAVA_HOME = /usr/lib/jvm/java-8-openjdk-amd64
BLD_TOOLS=17.0.0
APK_SIGNR=24.0.3

PKG_PATH = com/example/project
JVA_ROOT = app/src/main/java
RES_ROOT = app/src/main/res
AST_ROOT = app/src/main/assets
AMF_PNAM = app/src/main/AndroidManifest.xml
LIB_REAL = app/src/main/jniLibs

CLS_ROOT = app/build/intermediates/javac/debug/classes
DEX_PATH = app/build/intermediates/dex/debug/mergeDexDebug
APK_PATH = app/build/outputs/apk/debug
APK_NAME = app-debug
LIB_TEMP = lib

JVA_PRJT = $(JVA_ROOT)/$(PKG_PATH)
CLS_PRJT = $(CLS_ROOT)/$(PKG_PATH)
APK_PNAM = $(APK_PATH)/$(APK_NAME)
DRD_JAR =  $(DROID_HOME)/platforms/android-$(API)/android.jar

jopt  = -proc:none -Xlint:all -Xlint:-fallthrough -Werror -Xmaxerrs 5
jopt += -implicit:none -target 1.6 -source 1,6 -bootclasspath ~/jre1.6.0_45/lib/rt.jar

# wildcard matches symlinks awa directories. dir filters, sort dedupes.
JVA_DIRS := $(dir $(wildcard $(JVA_PRJT)/*/) )
JVA_DIRS := $(sort $(JVA_DIRS) )
JVA_DIRS += $(JVA_PRJT)/

# all in java tree except R.java (and thus R.class)
temp_jva := $(foreach pth, $(JVA_DIRS), $(wildcard $(pth)*.java) )
JVA_LIST := $(patsubst $(JVA_PRJT)/R.java, , $(temp_jva))

temp_cls := $(patsubst $(JVA_PRJT)/%.java,$(CLS_PRJT)/%.class, $(JVA_LIST))
CLS_LIST := $(patsubst $(JVA_PRJT)/%,$(CLS_PRJT)/%, $(temp_cls))

LIB_LIST := $(shell find $(LIB_REAL) -type f 2> /dev/null)
LIB_TLST := $(patsubst $(LIB_REAL)/%, $(LIB_TEMP)/%, $(LIB_LIST))

AST_LIST := $(shell find $(AST_ROOT) -type f 2> /dev/null)
ifneq ($(strip $(AST_LIST)),)
    AST_OPTN = -A $(AST_ROOT)
endif

RES_LIST := $(shell find $(RES_ROOT) -type f)

.DELETE_ON_ERROR:

assembleDebug: $(APK_PNAM).apk
    
$(APK_PNAM).apk : $(APK_PNAM).nosig.apk $(AMF_PNAM) $(RES_LIST) $(AST_LIST) $(LIB_LIST)
    #****************************** SIGN **********************************
    $(DROID_HOME)/build-tools/$(APK_SIGNR)/apksigner sign -v --verbose \
        --ks $(KEY_PATH) --ks-pass pass:debug_pw --min-sdk-version $(API) \
        --out $@ $(APK_PNAM).nosig.apk

$(APK_PNAM).nosig.apk : $(DEX_PATH)/classes.dex $(AMF_PNAM) $(RES_LIST) $(AST_LIST) $(LIB_LIST)
    #****************************** PACKAGE *******************************
    @-mkdir -p $(APK_PATH)
    $(DROID_HOME)/build-tools/$(BLD_TOOLS)/aapt package -v -f -M $(AMF_PNAM) \
        -S $(RES_ROOT) $(AST_OPTN) -I $(DRD_JAR) -F $(APK_PNAM).nosig.apk $(DEX_PATH)/
ifneq ($(strip $(LIB_LIST)),)
    #****************************** LIBS **********************************
    cp -r $(LIB_REAL) $(LIB_TEMP)/
    $(DROID_HOME)/build-tools/$(BLD_TOOLS)/aapt add -v $(APK_PNAM).nosig.apk $(LIB_TLST)
    @-rm -r $(LIB_TEMP)
endif

$(DEX_PATH)/classes.dex : $(CLS_LIST) $(CLS_PRJT)/R.class
    #****************************** LINK **********************************
    $(JAVA_HOME)/bin/javac $(jopt) -sourcepath $(JVA_ROOT)/ -cp $(DRD_JAR) \
        -d $(CLS_ROOT)/ $(JVA_PRJT)/$(START).java
    @-mkdir -p $(DEX_PATH)
    $(DROID_HOME)/build-tools/$(BLD_TOOLS)/dx --dex --verbose --output=$@ $(CLS_ROOT)

$(CLS_PRJT)/%.class : $(JVA_PRJT)/%.java $(CLS_PRJT)/R.class
    @-mkdir -p $(CLS_PRJT)
    $(JAVA_HOME)/bin/javac $(jopt) -sourcepath $(JVA_ROOT)/ -cp $(DRD_JAR) -d $(CLS_ROOT)/ $<
    
$(CLS_PRJT)/R.class : $(JVA_PRJT)/R.java
    #****************************** R.CLASS *******************************
    @-mkdir -p $(CLS_PRJT)
    $(JAVA_HOME)/bin/javac $(jopt) -sourcepath $(JVA_ROOT)/ -cp $(DRD_JAR) -d $(CLS_ROOT)/ $<

$(JVA_PRJT)/R.java : $(RES_LIST) $(AMF_PNAM)
    #****************************** R.JAVA ********************************
    $(DROID_HOME)/build-tools/$(BLD_TOOLS)/aapt package -v -f -M $(AMF_PNAM) \
        -S $(RES_ROOT) -m -J $(JVA_ROOT) -I $(DRD_JAR)

你可能需要一个 32 位的 zlib:

sudo apt install zlib1g:i386

...并且应该真正获得用于引导的 java 1.6:

https://www.oracle.com/uk/java/technologies/javase-java-archive-javase6-downloads.html

下载 jre-6u45-linux-x64.bin(需要注册)

您还需要为 apk 签名配置密钥库:

/usr/lib/jvm/java-8-openjdk-amd64/bin/keytool \
    -genkeypair -validity 1000 -dname "CN=some company,O=Android,C=JPN" -keystore ~/jkeystore \
    -storepass debug_pw -keypass debug_pw -alias cert -keyalg RSA -v

重要的:

我应该指出,最好认为 api < 9 的设备极易受到移动数据、wifi、网络共享、adb、蓝牙、GSM 等的攻击。

抱歉,这篇文章太长了,但是当我尝试分部分发布时,我遇到了版主/重复问题...

于 2021-08-20T02:41:34.453 回答