android项目打包成jar
随时随地阅读更多技术实战干货,获取项目源码、学习资料,请关注源代码社区公众号(ydmsq666)
在前面android 项目打包成Library一文中简要介绍了如何将android项目打包成库供第三方调用,但这种方式有一个缺陷,需要将源码提供给第三方,下面介绍将android项目打包成jar的方法,如果项目中没有引用资源文件,那就非常简单,直接打包src即可。如果项目引用了资源文件,则可以参考下面的方式,利用反射去动态获取资源ID。
下面示例代码转自xiaanming的博客:Android将Activity打成jar包供第三方调用(解决资源文件不能打包的问题)_xiaanming的博客-CSDN博客_android将activity打成jar包供第三方调用
需要打包的库项目:
LibraryActivity:
package com.home.library;
import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;
import android.widget.Toast;
public class LibraryActivity extends Activity {
String msg = "我是来自Jar中的Activity";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(MResource.getIdByName(getApplication(), "layout",
"activity_library"));
TextView mTextView = (TextView) findViewById(MResource.getIdByName(
getApplication(), "id", "textView1"));
mTextView.setText(msg);
Button mButton = (Button) findViewById(MResource.getIdByName(
getApplication(), "id", "button1"));
mButton.setText(msg);
mButton.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Toast.makeText(getApplication(), msg, Toast.LENGTH_SHORT)
.show();
}
});
}
}
MResource这个类很重要,主要是它的作用,利用反射根据资源名字获取资源ID(其实系统也自带了根据资源名字获取资源ID的方法getResources().getIdentifier("main_activity", "layout", getPackageName());第一个参数是资源的名字,第二个参数是资源的类型,例如layout, string等,第三个是包名字)
package com.home.library;
import android.content.Context;
/**
* 根据资源的名字获取其ID值
*
* @author mining
*
*/
public class MResource {
public static int getIdByName(Context context, String className, String name) {
String packageName = context.getPackageName();
Class r = null;
int id = 0;
try {
r = Class.forName(packageName + ".R");
Class[] classes = r.getClasses();
Class desireClass = null;
for (int i = 0; i < classes.length; ++i) {
if (classes[i].getName().split("\\$")[1].equals(className)) {
desireClass = classes[i];
break;
}
}
if (desireClass != null)
id = desireClass.getField(name).getInt(desireClass);
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (SecurityException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (NoSuchFieldException e) {
e.printStackTrace();
}
return id;
}
public static int[] getIdsByName(Context context, String className,
String name) {
String packageName = context.getPackageName();
Class r = null;
int[] ids = null;
try {
r = Class.forName(packageName + ".R");
Class[] classes = r.getClasses();
Class desireClass = null;
for (int i = 0; i < classes.length; ++i) {
if (classes[i].getName().split("\\$")[1].equals(className)) {
desireClass = classes[i];
break;
}
}
if ((desireClass != null)
&& (desireClass.getField(name).get(desireClass) != null)
&& (desireClass.getField(name).get(desireClass).getClass()
.isArray()))
ids = (int[]) desireClass.getField(name).get(desireClass);
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (SecurityException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (NoSuchFieldException e) {
e.printStackTrace();
}
return ids;
}
}
库项目的布局:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity" >
<Button
android:id="@+id/button1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentLeft="true"
android:layout_alignParentRight="true"
android:layout_alignParentTop="true" />
<TextView
android:id="@+id/textView1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentLeft="true"
android:layout_alignParentRight="true"
android:layout_below="@+id/button1" />
<ImageView
android:id="@+id/imageView1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:layout_alignParentLeft="true"
android:layout_alignParentRight="true"
android:layout_below="@+id/textView1"
android:layout_marginTop="28dp"
android:src="@drawable/ic_launcher" />
</RelativeLayout>
需要引用库项目的工程:
MainActivity:
package com.home.mainactivity;
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Button btn = (Button) findViewById(R.id.main_btn);
btn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent();
intent.setClassName(getApplication(), "com.home.library.LibraryActivity");
startActivity(intent);
}
});
}
}
布局文件:activity_main.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<Button
android:id="@+id/main_btn"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="跳转至LibraryActivity" />
</LinearLayout>
这里有三点需要注意:
1、把库项目打包成jar时,只需打包src。
2、除了需要将jar拷贝到libs目录下外,还需要将库项目中的资源文件拷贝到引用项目对应位置。因为库项目打包时只打了src下面的文件。
3、需要在引用项目的AndroidManifest.xml中声明注册引用到的activity,在上面示例中就是com.home.library.LibraryActivity
详细的配图解说参见原文:Android将Activity打成jar包供第三方调用(解决资源文件不能打包的问题)_xiaanming的博客-CSDN博客_android将activity打成jar包供第三方调用
另外有一种简单的方式,可以不使用反射。就是把所有用到的资源的ID(R.layout.xxx,R.id.xxx...)放在一个单独的类里面,然后把这个类也提供给第三方,第三方使用时,只需重新导一下R的包即可。