Android系统中,默认的单位是像素(px)。也就是说,在没有明确说明的情况下,所有的大小设置都是以像素为单位。
如果以像素设置大小,会导致不同分辨率下出现不同的效果。那么,如何将应用中所有大小的单位都设置为’dp’呢?
实际上TextView.setTextSize()重载了根据单位设置大小的方法。
笔者在此基础上实现了以下方法:
/**
* 获取当前分辨率下指定单位对应的像素大小(根据设备信息)
* px,dip,sp -> px
*
* Paint.setTextSize()单位为px
*
* 代码摘自:TextView.setTextSize()
*
* @param unit TypedValue.COMPLEX_UNIT_*
* @param size
* @return
*/
public float getRawSize(int unit, float size) {
Context c = getContext();
Resources r;
if (c == null)
r = Resources.getSystem();
else
r = c.getResources();
return TypedValue.applyDimension(unit, size, r.getDisplayMetrics());
}
下面是网友提供的方法:
/**
* 根据手机的分辨率从 dp 的单位 转成为 px(像素)
*/
public static int dip2px(Context context, float dpValue) {
final float scale = context.getResources().getDisplayMetrics().density;
return (int) (dpValue * scale + 0.5f);
}
/**
* 根据手机的分辨率从 px(像素) 的单位 转成为 dp
*/
public static int px2dip(Context context, float pxValue) {
final float scale = context.getResources().getDisplayMetrics().density;
return (int) (pxValue / scale + 0.5f);
}
Android使用BitmapFactory.Options获取图片文件类型(mime)
Android系统中在读取图片时可通过BitmapFactory.Options的outMimeType来直接读取其图片类型。如果要知道一个文件的类型,最好方式是直接读取文件头信息,可查看Android中Java根据文件头获取文件类型。
参考代码:
BitmapFactory.Options opts = new BitmapFactory.Options();
opts.inJustDecodeBounds = true; //确保图片不加载到内存
BitmapFactory.decodeResource(getResources(), R.drawable.a, opts);
System.out.println(opts.outMimeType);
Android系统的“程序异常退出”,给应用的用户体验造成不良影响。为了捕获应用运行时异常并给出友好提示,便可继承UncaughtExceptionHandler类来处理。通过Thread.setDefaultUncaughtExceptionHandler()方法将异常处理类设置到线程上即可。
1、异常处理类,代码如下:
public class CrashHandler implements UncaughtExceptionHandler {
public static final String TAG = "CrashHandler";
private static CrashHandler INSTANCE = new CrashHandler();
private Context mContext;
private Thread.UncaughtExceptionHandler mDefaultHandler;
private CrashHandler() {
}
public static CrashHandler getInstance() {
return INSTANCE;
}
public void init(Context ctx) {
mContext = ctx;
mDefaultHandler = Thread.getDefaultUncaughtExceptionHandler();
Thread.setDefaultUncaughtExceptionHandler(this);
}
@Override
public void uncaughtException(Thread thread, Throwable ex) {
// if (!handleException(ex) && mDefaultHandler != null) {
// mDefaultHandler.uncaughtException(thread, ex);
// } else {
// android.os.Process.killProcess(android.os.Process.myPid());
// System.exit(10);
// }
System.out.println("uncaughtException");
new Thread() {
@Override
public void run() {
Looper.prepare();
new AlertDialog.Builder(mContext).setTitle("提示").setCancelable(false)
.setMessage("程序崩溃了...").setNeutralButton("我知道了", new OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
System.exit(0);
}
})
.create().show();
Looper.loop();
}
}.start();
}
/**
* 自定义错误处理,收集错误信息 发送错误报告等操作均在此完成. 开发者可以根据自己的情况来自定义异常处理逻辑
*
* @param ex
* @return true:如果处理了该异常信息;否则返回false
*/
private boolean handleException(Throwable ex) {
if (ex == null) {
return true;
}
// new Handler(Looper.getMainLooper()).post(new Runnable() {
// @Override
// public void run() {
// new AlertDialog.Builder(mContext).setTitle("提示")
// .setMessage("程序崩溃了...").setNeutralButton("我知道了", null)
// .create().show();
// }
// });
return true;
}
}
2、线程绑定异常处理类
public class CrashHandlerActivity extends Activity {
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
CrashHandler crashHandler = CrashHandler.getInstance();
crashHandler.init(this); //传入参数必须为Activity,否则AlertDialog将不显示。
// 创建错误
throw new NullPointerException();
}
}
Android系统中TextView实现跑马灯效果,必须具备以下几个条件:
1、android:ellipsize=”marquee”
2、TextView必须单行显示,即内容必须超出TextView大小
3、TextView要获得焦点才能滚动
XML代码:
android:ellipsize="marquee", android:singleLine="true"
1
|
android:ellipsize="marquee", android:singleLine="true"
|
Java代码:
mTVText.setText("哼唱接撒砥砺风节雷锋精神http://orgcent.com/,很长很长很长很长很长很长的数据");
mTVText.setSingleLine(true);
mTVText.setEllipsize(TruncateAt.MARQUEE);
1
2 3 |
mTVText.
setText
(
"哼唱接撒砥砺风节雷锋精神http://orgcent.com/,很长很长很长很长很长很长的数据"
)
;
mTVText. setSingleLine ( true ) ; mTVText. setEllipsize (TruncateAt. MARQUEE ) ; |
PS: TextView.setHorizontallyScrolling(true); //让文字可以水平滑动
TextView还可以设置跑马灯效果的滚动次数,如下:
XML代码设置:
android:marqueerepeatlimit="1"。1代表1次,-1代表无限循环。
1
|
android
:marqueerepeatlimit
=
"1"。
1代表
1次,
-
1代表无限循环。
|
Java代码设置:
1
|
mTVText.
setMarqueeRepeatLimit
(
-
1
)
;
|
mTVText.setMarqueeRepeatLimit(-1);
Android闹钟程序周期循环提醒源码(AlarmManager)
Android系统提供了AlarmManager类来管理闹钟定时提醒任务。通过AlarmManager实现定时提醒及定时循环提醒。那么,AlarmManager类可以应用到以下场景:
1、定时循环启动组件(Component,如Activity、BroadcastReceiver),这样能替代在后台启动Service进行定时提醒任务
2、实现闹钟的按小时、天、周等形式的定时循环提醒功能。
定时启动组件很简单,下面贴出闹钟按天、周形式的定时循环提醒功能的核心代码。此功能核心的是计算出下一次闹钟提醒时间,代码如下:
/**
* 闹钟三种设置模式(dateMode):
* 1、DATE_MODE_FIX:指定日期,如20120301 , 参数dateValue格式:2012-03-01
* 2、DATE_MODE_WEEK:按星期提醒,如星期一、星期三 , 参数dateValue格式:1,3
* 3、DATE_MODE_MONTH:按月提醒,如3月2、3号,4月2、3号, 参数dateValue格式:3,4|2,3
*
* startTime:为当天开始时间,如上午9点, 参数格式为09:00
*/
public static long getNextAlarmTime(int dateMode, String dateValue,
String startTime) {
final SimpleDateFormat fmt = new SimpleDateFormat();
final Calendar c = Calendar.getInstance();
final long now = System.currentTimeMillis();
// 设置开始时间
try {
if(Task.DATE_MODE_FIX == dateMode) {
fmt.applyPattern("yyyy-MM-dd");
Date d = fmt.parse(dateValue);
c.setTimeInMillis(d.getTime());
}
fmt.applyPattern("HH:mm");
Date d = fmt.parse(startTime);
c.set(Calendar.HOUR_OF_DAY, d.getHours());
c.set(Calendar.MINUTE, d.getMinutes());
c.set(Calendar.SECOND, 0);
c.set(Calendar.MILLISECOND, 0);
} catch (Exception e) {
e.printStackTrace();
}
long nextTime = 0;
if (Task.DATE_MODE_FIX == dateMode) { // 按指定日期
nextTime = c.getTimeInMillis();
// 指定日期已过
if (now >= nextTime) nextTime = 0;
} else if (Task.DATE_MODE_WEEK == dateMode) { // 按周
final long[] checkedWeeks = parseDateWeeks(dateValue);
if (null != checkedWeeks) {
for (long week : checkedWeeks) {
c.set(Calendar.DAY_OF_WEEK, (int) (week + 1));
long triggerAtTime = c.getTimeInMillis();
if (triggerAtTime <= now) { // 下周
triggerAtTime += AlarmManager.INTERVAL_DAY * 7;
}
// 保存最近闹钟时间
if (0 == nextTime) {
nextTime = triggerAtTime;
} else {
nextTime = Math.min(triggerAtTime, nextTime);
}
}
}
} else if (Task.DATE_MODE_MONTH == dateMode) { // 按月
final long[][] items = parseDateMonthsAndDays(dateValue);
final long[] checkedMonths = items[0];
final long[] checkedDays = items[1];
if (null != checkedDays && null != checkedMonths) {
boolean isAdd = false;
for (long month : checkedMonths) {
c.set(Calendar.MONTH, (int) (month - 1));
for (long day : checkedDays) {
c.set(Calendar.DAY_OF_MONTH, (int) day);
long triggerAtTime = c.getTimeInMillis();
if (triggerAtTime <= now) { // 下一年
c.add(Calendar.YEAR, 1);
triggerAtTime = c.getTimeInMillis();
isAdd = true;
} else {
isAdd = false;
}
if (isAdd) {
c.add(Calendar.YEAR, -1);
}
// 保存最近闹钟时间
if (0 == nextTime) {
nextTime = triggerAtTime;
} else {
nextTime = Math.min(triggerAtTime, nextTime);
}
}
}
}
}
return nextTime;
}
public static long[] parseDateWeeks(String value) {
long[] weeks = null;
try {
final String[] items = value.split(",");
weeks = new long[items.length];
int i = 0;
for (String s : items) {
weeks[i++] = Long.valueOf(s);
}
} catch (Exception e) {
e.printStackTrace();
}
return weeks;
}
public static long[][] parseDateMonthsAndDays(String value) {
long[][] values = new long[2][];
try {
final String[] items = value.split("\\|");
final String[] monthStrs = items[0].split(",");
final String[] dayStrs = items[1].split(",");
values[0] = new long[monthStrs.length];
values[1] = new long[dayStrs.length];
int i = 0;
for (String s : monthStrs) {
values[0][i++] = Long.valueOf(s);
}
i = 0;
for (String s : dayStrs) {
values[1][i++] = Long.valueOf(s);
}
} catch (Exception e) {
e.printStackTrace();
}
return values;
}
1、异常处理类,代码如下:
1
2 3 4 |
BitmapFactory.
Options opts
=
new BitmapFactory.
Options
(
)
;
opts. inJustDecodeBounds = true ; //确保图片不加载到内存 BitmapFactory. decodeResource (getResources ( ), R. drawable. a, opts ) ; System. out. println (opts. outMimeType ) ; |