在Android中,有些方法非常有用,但是一时想不起来怎么用。所以将这些方法积累在这里,自己可以时常上来看看熟悉一下,如果能够帮到朋友们就更好了。
前置条件:
import android.os.SystemClock;
使用方法:
long startTime = SystemClock.uptimeMillis();
扩展用法:
将这个时间转换成人类可读的时间。也就是转换成年月日的形式。(代码参考自:xiaanming酱的博客。)
/** * 将毫秒数转换成yyyy-MM-dd-HH-mm-ss的格式 * @param milliseconds * @return */ private String paserTime(long milliseconds) { System.setProperty("user.timezone", "Asia/Shanghai"); TimeZone tz = TimeZone.getTimeZone("Asia/Shanghai"); TimeZone.setDefault(tz); SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd-HH-mm-ss"); String times = format.format(new Date(milliseconds)); return times; }
前置条件:
定义了Config类,并且在类中对获取数值进行了包装。详情见QuickSearchBox源码。
使用方法:
long delay = getConfig().getTypingUpdateSuggestionsDelayMillis();
详细方法实现:
public boolean showSuggestionsForZeroQuery() { return mContext.getResources().getBoolean(R.bool.show_zero_query_suggestions); }
该数值在values/config.xml里面设定。
前置条件:
应用中需要频繁发送一些信息。
应用场景:
字串匹配等。
private void updateSuggestionsBuffered() { if (DBG) Log.d(TAG, "updateSuggestionsBuffered()"); mHandler.removeCallbacks(mUpdateSuggestionsTask); long delay = getConfig().getTypingUpdateSuggestionsDelayMillis(); mHandler.postDelayed(mUpdateSuggestionsTask, delay); }
比如某些会更新UI的方法,就需要在该方法开始的地方判断当前线程是否是主线程。判断逻辑如下所示:
protected void checkThread() { if (Looper.myLooper() != Looper.getMainLooper()) { throw new IllegalStateException("Accessed Application object from thread " + Thread.currentThread().getName()); } }
<activity android:name="<span style="color:#ff0000;">.preferences.SearchSettingsActivity</span>" android:label="@string/search_settings" android:excludeFromRecents="true"> <intent-filter> <action android:name="android.search.action.SEARCH_SETTINGS" /> <category android:name="android.intent.category.DEFAULT" /> </intent-filter> <intent-filter> <action android:name="android.search.action.WEB_SEARCH_SETTINGS" /> <category android:name="android.intent.category.DEFAULT" /> </intent-filter> </activity>原始包名是com.android.quicksearchbox。而该类在这个路径下面的子文件夹里面,所以声明的时候用"."来进行子路径的表示。还有另外一种方式,如果在某一个类里面声明了另外一个内部类,在tag里面需要引用,那么就如下所示(注意$符号):
<activity android:name="<span style="color:#ff0000;">Settings$WirelessSettingsActivity</span>" android:taskAffinity="com.android.settings" android:label="@string/wireless_networks_settings_title" android:parentActivityName="Settings"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.DEFAULT" /> </intent-filter> </activity>
可以在命令行,运行下面的命令:
adb logcat -help
来显示所有的命令说明。但是一般我们需要用到的是具体的命令,最好是直接就能用的,比较适合我这种懒人。
比如要监控WindowsManager这个TAG,那么运行一下命令即可:
adb logcat -b system -s "WindowManager"
WindowManager”的log。这个知识参考这篇文章。
可以通过TelephonyManager中提供的方法来判断,代码如下:
import android.telephony.TelephonyManager;
TelephonyManager mTm = (TelephonyManager) mContext.getSystemService(Context.TELEPHONY_SERVICE); int simState = mTm.getSimState();
下面simState的类型包括如下几类:
/** 当处于状态间跳转时. */ public static final int SIM_STATE_UNKNOWN = 0; /** 设备中无SIM卡 */ public static final int SIM_STATE_ABSENT = 1; /** 处于锁定状态 */ public static final int SIM_STATE_PIN_REQUIRED = 2; /** PUK锁定状态 */ public static final int SIM_STATE_PUK_REQUIRED = 3; /** 网络PIN锁定状态 */ public static final int SIM_STATE_NETWORK_LOCKED = 4; /** SIM卡已就绪 */ public static final int SIM_STATE_READY = 5; /** SIM卡存在,但是有错误。废卡之类的 *@hide */ public static final int SIM_STATE_CARD_IO_ERROR = 6;
最简单的就是如下方式
<PreferenceScreen android:key="mobile_network_settings" android:title="@string/network_settings_title" android:dependency="toggle_airplane"> <intent android:action="android.intent.action.MAIN" android:targetPackage="com.android.phone" android:targetClass="com.android.phone.MobileNetworkSettings" /> </PreferenceScreen>
参考的高通代码WirelessSettings.java里面的函数:
判断是否有卡:
private boolean hasSimCard() { MSimTelephonyManager multiSimManager = MSimTelephonyManager.getDefault(); if (multiSimManager.isMultiSimEnabled()) { int numPhones = multiSimManager.getPhoneCount(); for (int i = 0; i < numPhones; i++) { // Because the status of slot1/2 will return // SIM_STATE_UNKNOWN under airplane mode. if (multiSimManager.getSimState(i) != TelephonyManager.SIM_STATE_ABSENT) { return true; } } } else { TelephonyManager singleSimManager = TelephonyManager.getDefault(); if (singleSimManager.getSimState() != TelephonyManager.SIM_STATE_ABSENT) { return true; } } return false; }
判断飞行模式:
public static boolean getAirplaneMode(Context context){ int isAirplaneMode = Settings.System.getInt(context.getContentResolver(),Settings.System.AIRPLANE_MODE_ON, 0) ; return (isAirplaneMode == 1)?true:false; }
我们有时候会需要将系统应用导入到Eclipse中,这样修改的代码就可以用Eclipse的自动检查和提示功能,减少花在代码上的精力,这样可以更多着眼于业务逻辑上。需要的包在alps\out\target\common\obj\JAVA_LIBRARIES\(一般系统模块的包在这里,比如framework、telephony-common等)和alps\out\target\common\obj\APP。一般导入之后,出现的找不到对应类的错误都可以通过导入系统jar包或者应用jar包解决。导入方法为:右击项目,Build Path------>Configure Build Path。在Libraries标签里面点击Add Library...选择User Library--->User Libraries---->New-->填写库名称(注意勾选上System Library选项)--->Add External JARs...将拷贝到本地的那些jar包(最好根据模块把将class.jar重命名)导入进去。确定之后在Java Build Path的Order and Export将刚才新建的库顺序放到最前面。
有些类不知道怎么找到底在哪个模块的class.jar里面?
在out/target/common里面调用grep搜索,例如:grep -w "Telephony" ./ -r
在一个应用开发中碰到这个问题,各种组合终于找到了启动方法:
该应用在包com.android.settings.fuelgauge;中添加了一个名为:PowerRankingFragment的类。如果要通过intent启动,那么intent各项设置如下:
ComponentName c = new ComponentName("com.android.settings", "com.android.settings.fuelgauge.PowerRankingFragment"); packageName = c.getPackageName(); className = "com.android.settings.fuelgauge.PowerRankingFragment";
Intent intent = new Intent(); intent.setClassName(packageName, className); startActivity(intent);这样就可以启动目标fragment了。这里要注意的是设定的包名和类名。
记住要跳转到的标签exitBAT前面加冒号,而不是加到后面。
GOTO exitBAT . . . :exitBAT exit
参考代码
getActionBar().setDisplayHomeAsUpEnabled(false);
getActivity().getActionBar().setDisplayHomeAsUpEnabled(false);