第一种,容易想到的方法。
要实现双击,你需要保存第一次点击时的时间,需要使用到变量,之后便是与第二次点击时的时间比较,看时间间隔是否在你设定的时间内(比如500ms)。
long firstClickTime = 0;
long secondClickTime = 0;
public void click1(View view) {
if (firstClickTime > 0) {
secondClickTime = SystemClock.uptimeMillis();
if (secondClickTime - firstClickTime < 500) {
Toast.makeText(this, "第一种双击方式", 0).show();
}
firstClickTime = 0;
return ;
}
firstClickTime = SystemClock.uptimeMillis();
new Thread(new Runnable() {
@Override
public void run() {
//
try {
Thread.sleep(500);
firstClickTime = 0;
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}).start();
}
双击事件(多击事件)中有两个重要的问题需要考虑:第一次点击的时间,最后(如何知道是“最后”)点击的时间;点击一次后,等待一段时间(比如2s),再次双击(多击)如何保证得到正确的响应。
上面的代码解决了这两个问题,但不够效率,也很复杂。它是通过一个变量存储第一次点击的时间,通过判断这个变量的大小来判断是不是第二次点击,解决第一个问题;再加上子线程的sleep方法,500ms内如果没有第二次点击,变量就重置了,解决了第二个问题。
但是这种方式解决双击事件还算凑合,那么三击、四击……如何解决了?
第二种,换种方式存储变量
<pre name="code" class="java"> List<Long> times = new ArrayList<Long>();
public void click2(View view) {
times.add(SystemClock.uptimeMillis());
if (times.size() == 2) {
//已经完成了一次双击,list可以清空了
if (times.get(times.size()-1)-times.get(0) < 500) {
times.clear();
Toast.makeText(this, "第二种双击方式", 0).show();
} else {
//这种情况下,第一次点击的时间已经没有用处了,第二次就是“第一次”
times.remove(0);
}
}
}
在第二种方式中,使用List存放点击时的时间。现在来分析一下这个实现的原理:
1、如何判断是第二次点击?
通过链表的长度,每次点击list的长度加1,当为2时,表示点击了两次。
2、如何消除间隔一段时间的点击事件的影响?
如果是正常的双击,点击两次就完成一次“轮回”,之前保存的时间数据已经使用完毕了,需要清除掉,具体操作就是将List清空。而如果是点击一次后,第二次点击相隔时间较长,那个第一次点击的时间已经没有用处了,就直接将第二次点击视为“第一次”点击,具体而言就是去掉第1次的点击事件。
这个方法,比起第一种方法效率要好得多,而且非常容易扩展到三击、四击……事件。比如:
<pre name="code" class="java"> List<Long> times = new ArrayList<Long>();
public void click2(View view) {
times.add(SystemClock.uptimeMillis());
if (times.size() == 3) {
if (times.get(times.size()-1)-times.get(0) < 500) {
times.clear();
Toast.makeText(this, "三击方式", 0).show();
} else {
times.remove(0);
}
}
}
改变的只是判断条件。
第三种,谷歌程序员的写法。
下面是谷歌所写的三击方法,我改写成了双击的方法
/**
* 双击事件、多击事件
*/
//存储时间的数组
long[] mHits = new long[2];
public void doubleClick() {
// 双击事件响应
/**
* arraycopy,拷贝数组
* src 要拷贝的源数组
* srcPos 源数组开始拷贝的下标位置
* dst 目标数组
* dstPos 开始存放的下标位置
* length 要拷贝的长度(元素的个数)
*
*/
//实现数组的移位操作,点击一次,左移一位,末尾补上当前开机时间(cpu的时间)
System.arraycopy(mHits, 1, mHits, 0, mHits.length - 1);
mHits[mHits.length - 1] = SystemClock.uptimeMillis();
//双击事件的时间间隔500ms
if (mHits[0] >= (SystemClock.uptimeMillis() - 500)) {
//双击后具体的操作
//do
}
}
非常简洁,思想差不多,不过谷歌工程师是利用数组移位操作来消除第二个问题的影响的。而要实现多击事件,只需修改数组长度即可。