我希望我的应用程序支持任何Android手机api 3及以上。
为什么?因为我不喜欢浪费,而且这些旧手机对于手头的任务来说已经足够好了。
可悲的是,获得基本的api-9前SDK构建的信息被证明比获得NDK工作要困难得多。
我无法找到一种机制来构建Android应用程序api
但是,可以使用旧的命令行工具构建它们。(我没有研究api <使用ActivityCompat库或ndk-build来调试JNI应用程序-这些留给读者作为练习:-)>
注意:使用这些技术需要熟悉当前的命令行构建,否则就没有什么意义了。
这个任务分为四个部分:
Android api样例项目<9
https://android.googlesource.com/platform/development/+refs
从甜甜圈到android-s-beta-4应有尽有。
每个目录都有一个'samples'子目录,这是一个开始的好地方。
Android工具构建api <9项目
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的api 3到8)
为api构建Android JNI库;9
注意api <设备似乎更喜欢从设备操作系统中链接共享符号,而不是从你的JNI库中链接,所以避免从你的库中发布冲突的符号,并静态链接任何第三方库(许可允许)。>
Makefile例子:
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
以及no-threads no-asm.
为api构建Android Java应用<9
遗憾的是,我无法找到一种方法来使用现在的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) {
...
}
});
}
};
Makefile
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
download 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, tethering, adb,蓝牙,GSM等。
唷
对不起,这篇文章太长了,但是当我试图在章节中发布它时,我遇到了版主/重复问题…