Yearly Archives: 2010

线程问题引起的内存泄露

最近解决的一个内存泄露,竟然是由于线程问题引起的 看看下面代码,如何? new Thread(new Runnable() {             public void run() {                 MessagingNotification.updateSendFailedNotificationForThread(ComposeMessageActivity.this, threadId);             }         }).run(); 有什么问题吗? 对,这段代码生成了一个thread的对象,竟然调用了run方法,这引起了内存泄露。 必须改为start才行。

Posted in Android Tagged with: , , , , , ,

activity的启动方式(launch mode)

在android里,有4种activity的启动模式,分别为: “standard” (默认) “singleTop” “singleTask” “singleInstance” 它们主要有如下不同: 1. 如何决定所属task “standard”和”singleTop”的activity的目标task,和收到的Intent的发送者在同一个task内,除非intent包括参数FLAG_ACTIVITY_NEW_TASK。 如果提供了FLAG_ACTIVITY_NEW_TASK参数,会启动到别的task里。 “singleTask”和”singleInstance”总是把activity作为一个task的根元素,他们不会被启动到一个其他task里。 2. 是否允许多个实例 “standard”和”singleTop”可以被实例化多次,并且存在于不同的task中,且一个task可以包括一个activity的多个实例; “singleTask”和”singleInstance”则限制只生成一个实例,并且是task的根元素。 singleTop要求如果创建intent的时候栈顶已经有要创建 的Activity的实例,则将intent发送给该实例,而不发送给新的实例。 3. 是否允许其它activity存在于本task内 “singleInstance”独占一个task,其它activity不能存在那个task里;如果它启动了一个新的activity,不管新的activity的launch mode 如何,新的activity都将会到别的task里运行(如同加了FLAG_ACTIVITY_NEW_TASK参数)。 而另外三种模式,则可以和其它activity共存。 4. 是否每次都生成新实例 “standard”对于没一个启动Intent都会生成一个activity的新实例; “singleTop”的activity如果在task的栈顶的话,则不生成新的该activity的实例,直接使用栈顶的实例,否则,生成该activity的实例。 比如现在task栈元素为A-B-C-D(D在栈顶),这时候给D发一个启动intent,如果D是 “standard”的,则生成D的一个新实例,栈变为A-B-C-D-D。 如果D是singleTop的话,则不会生产D的新实例,栈状态仍为A-B-C-D 如果这时候给B发Intent的话,不管B的launchmode是”standard” 还是 “singleTop” ,都会生成B的新实例,栈状态变为A-B-C-D-B。 “singleInstance”是其所在栈的唯一activity,它会每次都被重用。 “singleTask”如果在栈顶,则接受intent,否则,该intent会被丢弃,但是该task仍会回到前台。 当已经存在的activity实例处理新的intent时候,会调用onNewIntent()方法 如果收到intent生成一个activity实例,那么用户可以通过back键回到上一个状态;如果是已经存在的一个activity来处理这个intent的话,用户不能通过按back键返回到这之前的状态。

Posted in Android Tagged with: , , , , ,

Android内存泄漏问题的排查

稍好效果,请点击这里下载PDF版本 首先先检查下你有没有犯这些错误(内存泄露的主要原因): 生命周期过长的对象(static),尤其是集合对象(List/Map等)作为cache等使用,如果没有将某个对象主动的从中清除的话,这个集合就会占用越来越多的内存,可以用WeakReference,如WeakHashMap,使得它持有的对象不增加对象的引用数。 Scope定义不对,方法的局部变量定义成类的变量,类的静态变量等。尽量使得变量作用域别太大。 异常时没有加finally{}来释放某些资源,比如Cursor。 Listener没有显式remove;内部类持有外部对象的隐式引用,不论是什么,如果有add方法,一定要想想是否需要remove方法。 图像对象是否在不用的时候释放了(((BitmapDrawable) d).getBitmap().recycle();) 以下是正文 1. 获取hprof信息 1.1. 使用脚本get_hprof.sh ./get_hprof.sh --脚本内容见附录 运行此脚本之前,需要根据你的实际情况修改脚本,设置一下几个变量: # where is you sdk SDKPATH=. 这个是需要制定sdk/tools目录的位置,需要执行其下的adb和hprof-conv命令 # where will you want to save the log file TARGET_LOG_DIR=./hproflog log的保存位置 # which process will you

Posted in Android Tagged with: , , , , ,

android内存泄露(memory leak)之进程HProf (Heap Profile) 信息

因为性能问题(主要是内存泄漏),需要采集程序堆栈等信息,代码如下,只是简单说几个地方: 1.默认的HProf (Heap Profile) 放在/data/misc,需要改写权限为777 2.kill -10 pid 即可生成.prof文件 3.需要用android sdk的hprof-conf将android的hprof转化为标准的(能在MAT中用的) #!/bin/sh   # author liubin (http://liubin.nanshapo.com) # get_hprof.sh 2010/12/17   ######################################################## ###########                              ############### ###########   you

Posted in Android Tagged with: ,

在pc上操作实机

实机屏幕花了,没法操作,还好有 androidscreencast 这个软件。 这个软件,可以在pc上模拟操作真机,不仅是menu,back键,屏幕操作也可以,很方便。 我手工装的jdk1.5好像不太好用,启动出错;安装了openjdk-6-jdk之后,用它打开,就可以启动了: sudo apt-get install openjdk-6-jdk /usr/lib/jvm/java-6-openjdk/bin/javaws androidscreencast.jrlp

Posted in Android

手动启动媒体扫描

adb shell am broadcast -a android.intent.action.MEDIA_MOUNTED -d file:///sdcard Development\src\com\android\development\MediaScannerActivity.java中: @Override     public void onResume() {         super.onResume();           ……..           sendBroadcast(new Intent(Intent.ACTION_MEDIA_MOUNTED, Uri.parse("file://" + Environment.getExternalStorageDirectory())));         ……..

Posted in Android

logcat tips

写程序少不了调试,相对与debug,打log可能最简单。 adb 的logcat能查看所有的log,然后,很多时候可能不是我们想看的,所以,特总结了一些小窍门 1. 只看某一TAG的log adb logcat TWE:* *:S 在运行logcat的时候在前两列的信息中你就可以看到 logcat 的标签列表和优先级别, 它是这样标出的:/ . 下面是一个logcat输出的例子,它的优先级就似乎I,标签就是ActivityManage: I/ActivityManager( 585): Starting activity: Intent { action=android.intent.action…} 标签就是log.d的第一个参数了。 logcat过滤器的语法是tag:priority,所以,TWE:*,就表示标签TAW的所有级别的log。 *:S 的用处是确保日志输出的时候按照过滤器的进行限制。 或者logcat -s “TAG:PRIORITY” 2.清楚所有log logcat -c 3.将log保存到指定的文件中 logcat -f [filename] 4.在log中显示时间 默认的情况下是不显示打印时间的,需要用-v参数来设置格式: logcat -v

Posted in Android

从命令行给模拟器拨打电话

从Eclips的DDMS视图虽然可以给模拟器打电话,发短信,但是毕竟要去编辑源代码的换还要切换模式。 其实可以从命令行进行。 首先,登录到模拟器: telnet localhost 5554 进入后,按help可以得到帮助 help Android console command help: help|h|? print a list of commands event simulate hardware events geo Geo-location commands gsm GSM related commands kill kill the emulator instance gsm2 GSM2 related commands network manage

Posted in Android

sqlite的PHONE_NUMBERS_EQUAL函数

怎么才叫两个电话号码相等?用=肯定不行,86 138xxx 和0138xxx应该是相等的 android里在c实现了PHONE_NUMBERS_EQUAL函数,这个函数有3个参数。 前两个自然是需要比较的电话号码了,第三个参数有些特别,可选值为0或者1。 查看源代码可见,这个参数名为accept_thailand_case,看来和泰国有关了 各个国家的电话号码规则都不一样,看来移植的时候这个文件是必须被修改的。 在我们这里,第三个参数用0就可以了 代码位置在external\sqlite

Posted in Android

创建android home程序

需要在manifest.xml里加入: <intent-filter>      <action android:name="android.intent.action.MAIN" />      <category android:name="android.intent.category.HOME" />      <category android:name="android.intent.category.DEFAULT" />  </intent-filter>

Posted in Android, Tech

无觅相关文章插件,快速提升流量