首先我们需要明白,实现动态加载就是要解决两个问题:(如果使用Fragments实现,则是一个问题)
1、Activity生命周期的管理。
2、动态加载的apk的资源如何获取。
第一个问题是因为在java中任何一个程序要运行起来,必须通过类加载器将某个类加入内存,当我们通过一个类加载器将Activity加入内存时,其实这个Activity就是一个普通的类,它已经没有生命周期的概念了,在Android系统中,Activity的生命周期是通过ActivityManager来控制的,如果我们通过动态加载的方式加载这个Activity,那么ActivityManager根本就不知道这个Activity的存在,所以我们必须处理好这个Activity的生命周期,至于第二个问题,在Android中,我们获取资源都是通过Context拿到的,而动态加载的APK是没有Context的,所以我们不能和以前一样那样来拿。前面的两篇文章推荐的方法已经能够很好的解决以上两个问题,因此实现了APK的动态加载。 我先来描述一下大牛博客中实现动态加载的思路吧:创建一个ProxyActivity,通过名字知道,它就是一个代理Activity,我们调用任何一个Activity都是通过调用ProxyActivity实现的,我只需要传入动态加载apk的路径和需要动态加载的类名,比如加载了一个Activity之后,通过反射机制读取到Activity的所有的生命周期函数以及onActivityResult等函数,并保存在一个列表中,在ProxyActivity的onCreate中通过反射调用动态加载的Activity的onCreate,由于ProxyActivity是一个正常的Activity,它的生命周期是正常的,所以在ProxyActivity的生命周期函数中调用动态加载Activity的生命周期函数就ok了,从而实现动态加载的Activity也有生命周期了。同时尽然是代理,那么就代理彻底一点,就干脆把动态加载的Activity中的所有的逻辑都转入到ProxyActivity中。那么这就要求被加载的Activity有一个ProxyActivity的引用,这个可以让所有动态加载的Activity继承一个BaseActivity,这个BaseActivity中有一个setProxy方法,用来设置ProxyActivity。所以不是任何APK,都可以动态加载的,一般只有动态加载自己编写的apk,动态加载别人的apk不太现实。 看了上面的思路,是不是有点借腹生子的感觉,其实就是把动态加载的Activity的逻辑转移到了ProxyActivity 解决资源访问的问题方法就是造ProxyActivity中重载者两个函数 public abstract AssetManager getAssets(); public abstract Resources getResources(); 至于为什么能解决资源的问题,我还是推荐几篇文章大家去学习一下吧: 本人的另外一篇文章:http://blog.csdn.net/yuanzeyao/article/details/12955459 讲解Android资源加载机制的一篇文章:http://blog.csdn.net/singwhatiwanna/article/details/24532419 好了,上面就是通过Activity实现的动态加载apk,下面看看我是怎么通过Fragment来实现动态加载的,如果熟悉Fragment的同学们应该知道,Fragment就相当于一个有生命周期的View,它的生命周期被所在的Activity的生命周期管理,即使我们通过类加载器把一个Fragment加入到内存,它和以前我们使用的Fragment没有什么两样,只要我们将这个Fragment加入到ProxyActivity,ProxyActivity就会自动的管理好这个Fragment的生命周期。所以我们就不需要担心Fragment的生命周期,下面就来看看代码实现吧:
1、BaseFragment.java
[java]view plaincopy
publicclassBaseFragmentextendsFragmentimplementsIConstant
{
privatestaticfinalStringTAG="BaseFragment";
protectedStringmDexPath;
@Override
publicvoidonCreate(BundlesavedInstanceState)
{
super.onCreate(savedInstanceState);
Bundlebundle=this.getArguments();
//动态加载apk的路径
mDexPath=bundle.getString(DEX_PATH);
}
//在Fragment中启动另外一个Fragment
protectedvoidreplaceFragmentByProxy(Stringname)
{
if(mDexPath==null)
return;
//PROXY_VIEW_ACTION是ProxyActivity的action
Intentintent=newIntent(PROXY_VIEW_ACTION);
//传递apk路径
intent.putExtra(DEX_PATH,mDexPath);
//是启动Fragment还是启动Fragment,这里启动的是Fragment
intent.putExtra(START_TYPE,TYPE_FRAGMENT);
//需要加载的fragment的类名
intent.putExtra(CLASS_NAME,name);
this.startActivity(intent);
}
}
所有需要动态加载的Fragment都需要继承这个BaseFragment,每次启动一个Fragment,只需要传递apk的路径即可。下面是我写的一个MyFragment,用来使用BitmapFun加载网络图片的,这里仅仅是加载并显示图片,没有考虑其他的,如果想深入了解BitmapFun的使用,请看我的另外一篇文章:http://blog.csdn.net/yuanzeyao/article/details/38355719
[java]view plaincopy
publicclassMyFragmentextendsBaseFragment
{
privatestaticfinalStringTAG="MyFragment";
privatestaticfinalStringIMAGE_CACHE_DIR="thumbs";
privateImageFetchermImageFetcher;
privateGridViewmGridView;
privateContextcontext;
privateButtonbtn;
@Override
publicvoidonCreate(BundlesavedInstanceState)
{
super.onCreate(savedInstanceState);
ImageCacheParamscacheParams=newImageCacheParams(getActivity(),IMAGE_CACHE_DIR);
cacheParams.setMemCacheSizePercent(0.25f);//Setmemorycacheto25%ofappmemory
//TheImageFetchertakescareofloadingimagesintoourImageViewchildrenasynchronously
mImageFetcher=newImageFetcher(getActivity(),200);
mImageFetcher.setLoadingImage(R.drawable.empty_photo);
mImageFetcher.addImageCache(getActivity().getSupportFragmentManager(),cacheParams);
}
@Override
publicViewonCreateView(LayoutInflaterinflater,ViewGroupcontainer,BundlesavedInstanceState)
{
//这里其实可以直接使用R.layout.fragment
ResourcesmResources=this.getActivity().getResources();
returninflater.inflate(mResources.getIdentifier("fragment","layout","com.dl.client"),container,false);
}
@Override
publicvoidonViewCreated(Viewview,BundlesavedInstanceState)
{
super.onViewCreated(view,savedInstanceState);
mGridView=(GridView)view.findViewById(R.id.gridView);
btn=(Button)view.findViewById(R.id.btn_fragment);
btn.setOnClickListener(newView.OnClickListener()
{
@Override
publicvoidonClick(Viewarg0)
{
//在Fragment中动态加载另外一个Fragment
replaceFragmentByProxy("com.dl.client.TempFragment");
}
});
context=this.getActivity();
mGridView.setAdapter(newBaseAdapter()
{
@Override
publicViewgetView(intposition,ViewcontentView,ViewGrouparg2)
{
ImageViewmImg;
if(contentView==null)
{
contentView=LayoutInflater.from(context).inflate(R.layout.item,null);
}
mImg=(ImageView)contentView.findViewById(R.id.img_11);
//mImg.setImageResource(R.drawable.empty_photo);
mImageFetcher.loadImage(Images.imageThumbUrls[position],mImg);
returncontentView;
}
@Override
publiclonggetItemId(intarg0)
{
return0;
}
@Override
publicObjectgetItem(intarg0)
{
returnImages.imageThumbUrls[arg0];
}
@Override
publicintgetCount()
{
returnImages.imageThumbUrls.length;
}
});
}
}
下面看看这个应用的效果吧:
最后需要注意的一点就是动态加载的apk不能和宿主应用包含相同的jar包,不然会报错的。。。