使用永无止境的处理程序消息开始更新视图是一种不好的做法(尝试查看进程的 CPU 级别)。更好更优雅的方法是注册一个BroadcastReceiver
将触发更新的。
public class Clock extends LinearLayout {
private Calendar mCalendar;
private LinearLayout mLayoutTime;
private TextView mAMPMText;
private TextView mDateText;
private TextView mTimeText;
private View mSendFeedback;
private boolean mAttached;
private final Handler mHandler = new Handler();
SimpleDateFormat dateFormatter = new SimpleDateFormat("EEE, MMM d, yyyy");
SimpleDateFormat timeFormatter = new SimpleDateFormat("h:mm");
public Clock(final Context context, int layoutResourceID, int dateResId, int meResId,int amPmResId) {
super(context);
LayoutInflater layoutInflater = LayoutInflater.from(context);
View view = layoutInflater.inflate(layoutResourceID, null);
addView(view, new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT));
mAMPMText = (TextView) view.findViewById(amPmResId);
mTimeText = (TextView) view.findViewById(timeResId);
mDateText = (TextView) view.findViewById(dateResId);
}
@Override
protected void onAttachedToWindow() {
super.onAttachedToWindow();
if (!mAttached) {
mAttached = true;
IntentFilter filter = new IntentFilter();
filter.addAction(Intent.ACTION_TIME_TICK);
filter.addAction(Intent.ACTION_TIME_CHANGED);
filter.addAction(Intent.ACTION_TIMEZONE_CHANGED);
getContext().registerReceiver(mIntentReceiver, filter, null, mHandler);
}
// NOTE: It's safe to do these after registering the receiver since the receiver always runs
// in the main thread, therefore the receiver can't run before this method returns.
// The time zone may have changed while the receiver wasn't registered, so update the Time
mCalendar = Calendar.getInstance();
// Make sure we update to the current time
onTimeChanged();
updateView();
}
@Override
protected void onDetachedFromWindow() {
super.onDetachedFromWindow();
if (mAttached) {
getContext().unregisterReceiver(mIntentReceiver);
mAttached = false;
}
}
private void updateView(){
mTimeText.setText(timeFormatter.format(mCalendar.getTime()));
mDateText.setText(dateFormatter.format(mCalendar.getTime()));
mAMPMText.setText(mCalendar.get(Calendar.AM_PM) == 0 ? "AM" : "PM");
}
private void onTimeChanged() {
mCalendar.setTime(new Date());
updateContentDescription(mCalendar);
}
private final BroadcastReceiver mIntentReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
if (intent.getAction().equals(Intent.ACTION_TIMEZONE_CHANGED)) {
String tz = intent.getStringExtra("time-zone");
mCalendar.setTimeZone(TimeZone.getTimeZone(tz));
}
onTimeChanged();
updateView();
}
};
private void updateContentDescription(Calendar calendar) {
setContentDescription(calendar.toString());
}
}
布局(可以做得更好)
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="wrap_content"
android:layout_height="match_parent"
android:gravity="center_vertical"
android:orientation="horizontal" >
<LinearLayout
android:id="@+id/layout_time"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="horizontal" >
<TextView
android:id="@+id/time_txt"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:includeFontPadding="true" />
<TextView
android:id="@+id/am_pm_txt"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="AM" />
</LinearLayout>
<TextView
android:id="@+id/date_txt"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
</RelativeLayout>