当前位置:  编程技术>移动开发
本页文章导读:
    ▪关于QQ游戏大厅的架构小弟我也想说几句        关于QQ游戏大厅的架构我也想说几句先抛砖引玉一下,接个例子说明,现在的例子是要实现: 一个安装的apk里面的一个activity A要去调用一个放在sd卡上的没安装的apk里面的activity B,点返回键.........
    ▪ onWindowFocus-onAttachedToWindow-onWindowVisibilityChanged-onWindowFocusChanged        onWindowFocus--onAttachedToWindow--onWindowVisibilityChanged--onWindowFocusChanged1:   entry:        onStart---->onResume---->onAttachedToWindow----------->onWindowVisibilityChanged--visibility=0---------->onWindowFocusChanged(true)--.........
    ▪ ListView小知识点及技艺       ListView小知识点及技巧小知识点: ListView添加header以后.ListView中的item的position变为从1开始而不再是从0开始. 需求描述: ListView原本有数据.先将ListView滑动到底部,再重新加载数据.加载完成后显示.........

[1]关于QQ游戏大厅的架构小弟我也想说几句
    来源: 互联网  发布时间: 2014-02-18
关于QQ游戏大厅的架构我也想说几句

先抛砖引玉一下,接个例子说明,现在的例子是要实现:

一个安装的apk里面的一个activity A要去调用一个放在sd卡上的没安装的apk里面的activity B,点返回键又回到Activity A,而不是直接退出。

ok,我这里已经有现成的方法了,先上代码:

主动去加载的SD上类的activity

import java.lang.reflect.Constructor;
import java.lang.reflect.Method;

import android.app.Activity;
import android.content.Intent;
import android.content.pm.PackageInfo;
import android.os.Bundle;
import android.util.Log;
import android.view.KeyEvent;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.Toast;
import dalvik.system.DexClassLoader;

public class MainActivity
    extends Activity {
    private static final String TAG = "MainActivity";
    private boolean isJumped = false;
    private Class localClass;
    private Object instance;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        Button btn = (Button) findViewById(R.id.btn);
        btn.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View v) {
            	isJumped = true;
                Bundle paramBundle = new Bundle();
                paramBundle.putBoolean("KEY_START_FROM_OTHER_ACTIVITY", true);
                String dexpath = "/mnt/sdcard/ToolA.apk";
                String dexoutputpath = "/mnt/sdcard/";
                LoadAPK(paramBundle, dexpath, dexoutputpath);
            }
        });
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
    }

    @Override
    public boolean onKeyDown(int keyCode, KeyEvent event) {
        if (keyCode == KeyEvent.KEYCODE_BACK) {
			if (isJumped) {
				Bundle paramBundle = new Bundle();
                paramBundle.putBoolean("KEY_START_FROM_OTHER_ACTIVITY", true);
				closeAPK(paramBundle);
				isJumped = false;
				return true;
			}
        }
        return super.onKeyDown(keyCode, event);
    }
    
    public void LoadAPK(Bundle paramBundle, String dexpath, String dexoutputpath) {
        ClassLoader localClassLoader = ClassLoader.getSystemClassLoader();
        DexClassLoader localDexClassLoader =
            new DexClassLoader(dexpath, dexoutputpath, null, localClassLoader);
        try {
            PackageInfo plocalObject = getPackageManager().getPackageArchiveInfo(dexpath, 1);

            if ((plocalObject.activities != null) && (plocalObject.activities.length > 0)) {
                String activityname = plocalObject.activities[0].name;
                Log.d(TAG, "activityname = " + activityname);

                localClass = localDexClassLoader.loadClass(activityname);
                Constructor localConstructor = localClass.getConstructor(new Class[] {});
                instance = localConstructor.newInstance(new Object[] {});
                Log.d(TAG, "instance = " + instance);

                Method localMethodSetActivity =
                    localClass.getDeclaredMethod("setActivity", new Class[] {Activity.class});
                localMethodSetActivity.setAccessible(true);
                localMethodSetActivity.invoke(instance, new Object[] {this});

                Method methodonCreate = localClass.getDeclaredMethod("onCreate", new Class[] {Bundle.class});
                methodonCreate.setAccessible(true);
                methodonCreate.invoke(instance, new Object[] {paramBundle});
            }
            return;
        }
        catch (Exception ex) {
            ex.printStackTrace();
        }
    }
    
	public void closeAPK(Bundle paramBundle) {
		try {
			Method methodonCreate = localClass.getDeclaredMethod("onGoBack");
			methodonCreate.setAccessible(true);
			methodonCreate.invoke(instance);
		} catch (Exception ex) {
			ex.printStackTrace();
		}
	}
}

另外一个是sd卡上待加载的activity:

package mobi.thinkchange.android.toola;

import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.view.KeyEvent;
import android.widget.Toast;

public class ToolAActivity
    extends Activity {
    private static final String TAG = "ToolAActivity";
    private Activity otherActivity;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        boolean b = false;
        if (savedInstanceState != null) {
            b = savedInstanceState.getBoolean("KEY_START_FROM_OTHER_ACTIVITY", false);
            if (b) {
                this.otherActivity.setContentView(new TBSurfaceView(this.otherActivity));
            }
        }
        if (!b) {
            super.onCreate(savedInstanceState);
            setContentView(new TBSurfaceView(this));
        }
    }

    public void setActivity(Activity paramActivity) {
        Log.d(TAG, "setActivity..." + paramActivity);
        this.otherActivity = paramActivity;
    }

    public void onGoBack() {
    	otherActivity.setContentView(R.layout.main);
    }
}

上面是实现我说的例子最核心的两个类,全部代码详见:http://download.csdn.net/detail/androidzhaoxiaogang/4903804

(代码只是片段,如果要实际应用还需要完善,比如在返回的时候终端执行的runnable可参考我的另外一篇文章如何终端执行的java线程)

也许网上有很多关于QQ游戏大厅的分析或者类似上面的例子,但是我想做进一步的分析。

1.我们知道一个activity具有严格意义上的生命周期必须要在ActivityThread里面去创建,所以通过DexClassLoader加载的ActivityB是没有生命周期的,它也就是一个普通的类;

2.有人会问,为什么还要让ActivityA 来加载ActivityB,而不是直接ActivityA去load 另外一个类比如classC呢? 这个问题很简单,因为我们在打包成APK的时候,有activity会更方便和直观(自己去体会)。简单的游戏,往往一个activity就够了。那么ActivityA 也仅仅只需要加载一个类实例。


1楼zhouyuanjing前天 18:16涨见识了

    
[2] onWindowFocus-onAttachedToWindow-onWindowVisibilityChanged-onWindowFocusChanged
    来源: 互联网  发布时间: 2014-02-18
onWindowFocus--onAttachedToWindow--onWindowVisibilityChanged--onWindowFocusChanged

1:   entry:        onStart---->onResume---->onAttachedToWindow----------->onWindowVisibilityChanged--visibility=0---------->onWindowFocusChanged(true)------->


2.   exit:      onPause---->onStop---->onWindowFocusChanged(false)  ----------------------         (lockscreen)


3,  exit : onPause----->onWindowFocusChanged--false-------->Launcher.Workspace, onWindowVisibilityChanged--visibility=8------------>onStop  (to another activity)


4, 


    
[3] ListView小知识点及技艺
    来源: 互联网  发布时间: 2014-02-18
ListView小知识点及技巧

小知识点:
ListView添加header以后.ListView中的item的position变为从1开始而不再是从0开始.


需求描述:
ListView原本有数据.先将ListView滑动到底部,再重新加载数据.加载完成后显示ListView的头部而不是仍然在底部

解决办法:
adapter.notifyDataSetChanged();
listView.setAdapter(adapter);

 


    
最新技术文章:
▪Android开发之登录验证实例教程
▪Android开发之注册登录方法示例
▪Android获取手机SIM卡运营商信息的方法
▪Android实现将已发送的短信写入短信数据库的...
▪Android发送短信功能代码
▪Android根据电话号码获得联系人头像实例代码
▪Android中GPS定位的用法实例
▪Android实现退出时关闭所有Activity的方法
▪Android实现文件的分割和组装
▪Android录音应用实例教程
▪Android双击返回键退出程序的实现方法
▪Android实现侦听电池状态显示、电量及充电动...
▪Android获取当前已连接的wifi信号强度的方法
▪Android实现动态显示或隐藏密码输入框的内容
▪根据USER-AGENT判断手机类型并跳转到相应的app...
▪Android Touch事件分发过程详解
▪Android中实现为TextView添加多个可点击的文本
▪Android程序设计之AIDL实例详解
▪Android显式启动与隐式启动Activity的区别介绍
▪Android按钮单击事件的四种常用写法总结
▪Android消息处理机制Looper和Handler详解
▪Android实现Back功能代码片段总结
▪Android实用的代码片段 常用代码总结
▪Android实现弹出键盘的方法
▪Android中通过view方式获取当前Activity的屏幕截...
▪Android提高之自定义Menu(TabMenu)实现方法
▪Android提高之多方向抽屉实现方法
▪Android提高之MediaPlayer播放网络音频的实现方法...
▪Android提高之MediaPlayer播放网络视频的实现方法...
▪Android提高之手游转电视游戏的模拟操控
 


站内导航:


特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

©2012-2021,,E-mail:www_#163.com(请将#改为@)

浙ICP备11055608号-3