NSFileManager *m = [NSFileManager defaultManager]; NSArray *fileList = [m directoryContentsAtPath:[[[[NSHomeDirectory() stringByAppendingPathComponent:@"Documents"] stringByAppendingPathComponent:@"163"] stringByAppendingPathComponent:@"songzi"] stringByAppendingPathComponent:@"pictures"]]; for (NSString *s in fileList){ NSLog(s); }
最近在做一个防火墙应用,需要从数据库中得到数据,显示在ListView中,同时每一个Item还有几个CheckBox,可以直接点击CheckBox来更新数据库的内容。效果图是这样的:
首先提一下,Android设计ListView主要是用来展示数据,在其中进行各种操作的效率不高,应该尽可能的避免在其中进行过多操作。
下面是在实现过程中遇到了一些问题和解决的思路。
最初的解决思路:使用自定义的CursorAdapter,将数据库中的数据展示在ListView中,然后在adapter中对CheckBox添加setOnCheckChangedListener,来监听点击并更新到数据库。
1.在成功添加CheckBox,并setOnCheckChangedListener来更新数据库之后,出现了一个问题:当点击一个CheckBox以后,查看数据库发现正常更新,但是向下滚动屏幕直到该Item被隐藏,然后再向上滚动重新显示该条目时,CheckBox又回到原来的状态。
经过分析,发现问题在于更新数据库之后,CursorAdapter所使用的cursor没有更新,仍旧是旧的数据,因此在重新bindView时CheckBox回到原来的状态。
简单的是给数据库设置一个ContentObserver,并在数据库改变时重新查询和更新界面。当然,如果api level在11以上时,可以直接使用 public CursorAdapter (Context context, Cursor c, int flags)
中的 FLAG_REGISTER_CONTENT_OBSERVER 标志。
这种方法比较正规,但是有一定的开销。而且要谨慎使用cursor.requery和 CursorAdapter (Context context, Cursor c, boolean autoRequery),autoRequery传入true。
这些已经被废弃的方法会在UI线程中执行数据查询和更新,有可能导致应用显示缓慢甚至ANR。
还有一种轻巧的方法,当该数据库只由你自己操作时,可以在adapter中使用一个数组boolean[]或者集合list<boolean>来手动管理ListView的CheckBox状态,不必在每次改变后重新查询数据并刷新界面,开支会减小很多。
2.我在这里选择了第一种方法,创建了一个ContentObserver来监听数据库,并在每次改变时刷新ListView的adapter。这样导致了第二个问题:当点击CheckBox之后,程序不停地刷新界面,进入死循环。
这个bug原因相当的简单:当改变数据库刷新ListView时会对每一个CheckBox进行setchecked(boolean),而setChecked(boolean)又触发OnCheckChangedListener从而再次更新数据库,导致死循环。
一个简单避开这个bug的方法就是改变监听的方式。当监听CheckChanged时,刷新列表时setChecked(boolean)会触发监听,可以改为监听click事件来避开这个问题,使用setOnClickListener,在每次点击CheckBox后直接反转数据库中该位置的数据(0,1互换)即可。
如果有任何需要交流的问题,请给我留言。
Android 2.3.3 Eclipse Version: 3.7.0 LogCat
LogCat 报错:
02-03 15:52:46.363: INFO/ActivityManager(61): Starting: Intent { flg=0x10000000 cmp=com.taobao.iphone4s/.Start } from pid 736 02-03 15:52:46.405: DEBUG/AndroidRuntime(736): Shutting down VM 02-03 15:52:46.405: WARN/dalvikvm(736): threadid=1: thread exiting with uncaught exception (group=0x40015560) 02-03 15:52:46.405: ERROR/CrashHandler(736): Unable to start receiver com.taobao.iphone4s.BootBroadcastReceiver: android.content.ActivityNotFoundException: Unable to find explicit activity class {com.taobao.iphone4s/com.taobao.iphone4s.Start}; have you declared this activity in your AndroidManifest.xml? 02-03 15:52:46.672: DEBUG/CrashHandler(736): BOARD : unknown 02-03 15:52:46.672: DEBUG/CrashHandler(736): BOOTLOADER : unknown 02-03 15:52:46.672: DEBUG/CrashHandler(736): BRAND : generic 02-03 15:52:46.672: DEBUG/CrashHandler(736): CPU_ABI : armeabi 02-03 15:52:46.672: DEBUG/CrashHandler(736): CPU_ABI2 : unknown 02-03 15:52:46.672: DEBUG/CrashHandler(736): DEVICE : generic 02-03 15:52:46.683: DEBUG/CrashHandler(736): DISPLAY : sdk-eng 2.3.3 GRI34 101070 test-keys 02-03 15:52:46.683: DEBUG/CrashHandler(736): FINGERPRINT : generic/sdk/generic:2.3.3/GRI34/101070:eng/test-keys 02-03 15:52:46.683: DEBUG/CrashHandler(736): HARDWARE : goldfish 02-03 15:52:46.683: DEBUG/CrashHandler(736): HOST : android-test-26.mtv.corp.google.com 02-03 15:52:46.693: DEBUG/CrashHandler(736): ID : GRI34 02-03 15:52:46.693: DEBUG/CrashHandler(736): MANUFACTURER : unknown 02-03 15:52:46.703: DEBUG/CrashHandler(736): MODEL : sdk 02-03 15:52:46.703: DEBUG/CrashHandler(736): PRODUCT : sdk 02-03 15:52:46.712: DEBUG/CrashHandler(736): RADIO : unknown 02-03 15:52:46.712: DEBUG/CrashHandler(736): SERIAL : unknown 02-03 15:52:46.712: DEBUG/CrashHandler(736): TAGS : test-keys 02-03 15:52:46.712: DEBUG/CrashHandler(736): TIME : 1296773036000 02-03 15:52:46.712: DEBUG/CrashHandler(736): TYPE : eng 02-03 15:52:46.712: DEBUG/CrashHandler(736): UNKNOWN : unknown 02-03 15:52:46.724: DEBUG/CrashHandler(736): USER : android-build 02-03 15:52:49.905: INFO/Process(736): Sending signal. PID: 736 SIG: 9 02-03 15:52:49.924: INFO/ActivityManager(61): Process com.taobao.iphone4s (pid 736) has died. 02-03 15:52:49.924: INFO/WindowManager(61): WIN DEATH: Window{405917c8 Toast paused=false} 02-03 15:52:50.252: WARN/NotificationService(61): Object died trying to hide notification android.app.ITransientNotification$Stub$Proxy@406031c8 in package com.taobao.iphone4s 02-03 15:52:50.252: WARN/ActivityManager(61): setProcessForeground called on unknown pid: 736 02-03 15:52:51.622: DEBUG/dalvikvm(340): GC_EXPLICIT freed 7K, 55% free 2530K/5511K, external 1625K/2137K, paused 45ms 02-03 15:52:56.762: DEBUG/dalvikvm(252): GC_EXPLICIT freed 7K, 54% free 2536K/5511K, external 1625K/2137K, paused 55ms
主要错误:
02-03 15:52:46.405: ERROR/CrashHandler(736): Unable to start receiver com.taobao.iphone4s.BootBroadcastReceiver: android.content.ActivityNotFoundException: Unable to find explicit activity class {com.taobao.iphone4s/com.taobao.iphone4s.Start}; have you declared this activity in your AndroidManifest.xml?
原因:AndroidManifest.xml未添加对应Activity配置,无法找到明确的Activity。
解决办法:在AndroidManifest.xml添加对应Activity配置内容。
<activity android:name=".Start" android:label="@string/app_name" android:theme="@android:style/Theme.Dialog" android:screenOrientation="user" android:configChanges="orientation|keyboardHidden"> <intent-filter> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity>