介绍
前边两篇介绍两种方案都是基于Android基础知识实现的,其中方案二中的缺点是:工具类要维护所有的跳转和数据传递,这里边包含Fragment启动Activity的方式,而且还要管理所有的逻辑,当跳转完成后要记得从内存中移出去。但是当没有登录的时候跳转到登录页面,用户没有进行登录而是按下返回键,那么刚才添加的跳转和数据就没有去向和消费了,就会一直滞留在缓存中了。一般情况用户也不会累计太多这种操作的。当然针对这种情况也有处理方案:当退出APP的时候清除缓存,或者使用LRU缓存,当内存满的时候,清除掉那些旧的无用的数据。
那么还有没有更好点的方案呢?
当然有了,我们学习的AOP思想就可以解决这个问题,关于AOP的介绍不是本文的重点,网上有很多不错博文,请大家自行解决。
基于AOP的思想我们使用动态代理的思想处理登录成功后再跳转到目标页的逻辑。
面向切面的思想很简单如上图,我们在所有横向跳转的页面中间插一个纵向的切面,进行拦截处理,当没有登录的时候,进行登录。
还和第二篇前面一样,这里不在讲述不传递数据的情况,直接讲述传递数据的情况。
定义动态代理需要的接口
public interface ILaunchManagerService {
public void startActivity(Intent intent);
}
这里ILaunchManagerService命名偷学的ActivityManagerService的方式,里边只是定义一个startActivity抽象方法。
定义InvocationHandler的实现类
public class LaunchInvocationHandler implements InvocationHandler {
private ILaunchManagerService launchManagerService;
private Context context;
public LaunchInvocationHandler(Context context, ILaunchManagerService launchManagerService) {
this.context = context;
this.launchManagerService = launchManagerService;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if (!Constant.isLogin) {
Intent intent = (Intent) args[0];
ComponentName component = intent.getComponent();
String className = component.getClassName();
Bundle bundleExtra = intent.getBundleExtra(Constant.KEY_DATA);
bundleExtra.putString(Constant.KEY_CLASS_NAME, className);
intent.putExtra(Constant.KEY_DATA, bundleExtra);
intent.setClass(context, LoginActivity.class);
args[0] = intent;
}
method.invoke(launchManagerService, args);
return null;
}
}
invoke方法内部判断没有登录的时候,修改实现方法跳转到登录页,并且把目标页的className传递给登录页。
使用
MainActivity.java
public class MainActivity extends AppCompatActivity implements ILaunchManagerService {
private Button myGoodsBtn, myFavoritesBtn;
private ILaunchManagerService launchManagerService;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
myGoodsBtn = findViewById(R.id.myGoodsBtn);
myFavoritesBtn = findViewById(R.id.myFavoriteBtn);
launchManagerService = (ILaunchManagerService) Proxy.newProxyInstance(this.getClassLoader(), new Class[]{ILaunchManagerService.class}, new LaunchInvocationHandler(this,this));
myGoodsBtn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
lookMyGoods();
}
});
myFavoritesBtn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
lookMyFavorite();
}
});
}
private void lookMyGoods() {
Intent intent = new Intent(this, MyGoodsActivity.class);
Bundle bundle = new Bundle();
bundle.putString(Constant.KEY_MSG, "通过登录传递的产品");
intent.putExtra(Constant.KEY_DATA, bundle);
launchManagerService.startActivity(intent);
}
private void lookMyFavorite() {
Intent intent = new Intent(this, MyGoodsActivity.class);
Bundle bundle = new Bundle();
bundle.putString(Constant.KEY_MSG, "通过登录传递的收藏");
intent.putExtra(Constant.KEY_DATA, bundle);
launchManagerService.startActivity(intent);
}
}
MainActivity 采用类适配器的方式,使用代理对象实现页面的跳转。launchManagerService.startActivity,会调用到代理类的invoke的方法,进行切面拦截处理。
登录页
public class LoginActivity extends AppCompatActivity {
......
private void handleLoginSuccess3(){
String className = bundle.getString(Constant.KEY_CLASS_NAME);
if (TextUtils.isEmpty(className)) {
return;
}
Intent intent = new Intent();
intent.setComponent(new ComponentName(this, className));
bundle.remove(Constant.KEY_CLASS_NAME);
intent.putExtra(Constant.KEY_DATA, bundle);
startActivity(intent);
finish();
}
}
登录成功后根据接收的目标页 className 进行页面跳转
商品和收藏页面
public class MyFavoriteActivity extends AppCompatActivity {
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_favorite);
Bundle bundle = getIntent().getBundleExtra(Constant.KEY_DATA);
if (bundle != null) {
String msg = bundle.getString(Constant.KEY_MSG);
if (!TextUtils.isEmpty(msg)) {
Toast.makeText(this, "我的收藏--》msg="+msg,Toast.LENGTH_LONG).show();
}
}
}
}
public class MyGoodsActivity extends AppCompatActivity {
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_goods);
Bundle bundle = getIntent().getBundleExtra(Constant.KEY_DATA);
if (bundle != null) {
String msg = bundle.getString(Constant.KEY_MSG);
if (!TextUtils.isEmpty(msg)) {
Toast.makeText(this, "我的产品--》msg="+msg,Toast.LENGTH_LONG).show();
}
}
}
}
这两个页面就是正常的接收数据进行处理,不再详述。
总结
目前来说这种方案是最简单,容易维护的方案。