在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);
}
注意mUpdateSuggestionsTask是Runnable对象。如果不使用第一个removeCallbacks可能会导致该Runnable执行多次。
确保某方法只在主线程中调用
比如某些会更新UI的方法,就需要在该方法开始的地方判断当前线程是否是主线程。判断逻辑如下所示:
protected void checkThread() {
if (Looper.myLooper() != Looper.getMainLooper()) {
throw new IllegalStateException("Accessed Application object from thread "
+ Thread.currentThread().getName());
}
}
AndroidManifest中,每个tag中如何声明android:name
如果是在程序包下面还有子文件夹,那么使用如下方式声明:<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>
Logcat用法
可以在命令行,运行下面的命令:
adb logcat -help
来显示所有的命令说明。但是一般我们需要用到的是具体的命令,最好是直接就能用的,比较适合我这种懒人。
比如要监控WindowsManager这个TAG,那么运行一下命令即可:
adb logcat -b system -s "WindowManager"
-b是将默认监听的main缓冲切换到system(这两个数值都在命令说明里面可以找到),并且增加一个过滤器只看tag为“
WindowManager”的log。这个知识参考这篇文章。
判断手机中是否有SIM卡
可以通过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;
可以通过返回值的判断,来确定SIM卡的状态。
如何在应用的PreferenceScreen中应用其他模块的偏好菜单
最简单的就是如下方式
<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>
可以看到,设置了intent内容,也就是点击该菜单会跳转到哪里去。该点击事件会导致从Settings模块跳转到Phone模块的MobileNetWorkSettings这个PreferenceActivity子类中来。
是否有卡以及是否处于飞行模式
参考的高通代码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;
}
注意,实际上SIM_STATE_UNKNOWN并不能判断是否处于飞行模式,该状态的定义可以知道,关闭飞行模式,搜索网络过程中,也是处于该状态,如果对于飞行状态有严格的要求的话,最好不要用状态是否非UNKOWN来判断是否处于飞行模式。
判断飞行模式:
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中,这样修改的代码就可以用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
如何启动Settings模块中某个Fragment
在一个应用开发中碰到这个问题,各种组合终于找到了启动方法:
该应用在包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
记住要跳转到的标签exitBAT前面加冒号,而不是加到后面。
GOTO exitBAT
.
.
.
:exitBAT
exit
如何禁用ActionBar的向上导航功能
参考代码
getActionBar().setDisplayHomeAsUpEnabled(false);
如果是在一个Fragment里面,可以参考:
getActivity().getActionBar().setDisplayHomeAsUpEnabled(false);