Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱
MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina.com

倒计时 总结 Timer Handler RxJava


目录

利用系统API的几种实现方式

使用 CountDownTimer 实现 - 最简洁【推荐】

CountDownTimer 简介

文档

Schedule安排、清单 a countdown until a time in the future, with regular规律的 notifications on intervals间隔 along the way过程.

在文本字段中显示一个30秒倒计时的示例:

@BindView(R.id.send) Button send;//发送验证码
new CountDownTimer(60000, 1000) {
@Override
public void onTick(long millisUntilFinished) {
send.setText(millisUntilFinished / 1000 + "S");
}
@Override
public void onFinish() {
send.setEnabled(true);
send.setText("重新发送");
}
}.start();

The calls to onTick(long) are synchronized同步 to this object so that one call to onTick(long) won't ever occur before the previous callback is complete.

This is only relevant相应、相关 when the implementation of onTick(long) takes an amount of一定数量的 time to execute执行 that is significant重大 compared to the countdown interval间隔.

API数量非常少,但各个都极其有用

构造方法

CountDownTimer(long millisInFuture, long countDownInterval)
  • millisInFuture: The number of millis in the future from the call to start() until the countdown is done and onFinish() is called.
  • countDownInterval: The interval along the way to receive onTick(long) callbacks.

开启和结束方法

final void cancel()
final CountDownTimer start()

抽象(回调)方法

abstract void onFinish():Callback fired when the time is up.
abstract void onTick(long millisUntilFinished):Callback fired on regular interval. millisUntilFinished: The amount of time until finished.

使用案例

@BindView(R.id.send) Button send;//发送验证码
private CountDownTimer timer;//使用CountDownTimer

@OnClick({R.id.send, R.id.next})
public void onClickIv(View v) {
switch (v.getId()) {
case R.id.send:
setTimer();//实际是要在点击之后判断如果符合倒计时条件才调用 setTimer 方法,要在失败后调用 destoryTimer
break;
}
}

@Override
protected void onDestroy() {
super.onDestroy();
destoryTimer();
}

private void setTimer() {
send.setEnabled(false);
timer = new CountDownTimer(60 * 1000, 1000) {
@Override
public void onTick(long millisUntilFinished) {
int time = (int) (millisUntilFinished / 1000);
send.setText(time + "s");
} @Override
public void onFinish() {
destoryTimer();
}
};
timer.start();
}

private void destoryTimer() {
send.setEnabled(true);
send.setText("获取验证码");
if (timer != null) {
timer.cancel();
timer = null;
}
}

使用 RxJava 实现 - 方便强大【推荐】

可以使用 intervalRange 很方便的实现这个功能,也可以使用 repeat、repeatUntil、repeatWhen 间接实现类似功能。

@BindView(R.id.send) Button send;//发送验证码
private Disposable disposable; @OnClick({R.id.send, R.id.next})
public void onClickIv(View v) {
switch (v.getId()) {
case R.id.send:
startCountDown();
break;
}
}

@Override
protected void onDestroy() {
super.onDestroy();
if (disposable != null && !disposable.isDisposed()) {
disposable.dispose();
}
} private void startCountDown() {
send.setEnabled(false);
disposable = Observable.intervalRange(0, 10, 0, 1, TimeUnit.SECONDS) //起始值,发送总数量,初始延迟,固定延迟
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(time -> send.setText((10 - time) + "s"),
Throwable::printStackTrace,
() -> {
send.setEnabled(true);
send.setText("获取验证码");
}
);
}

使用 Timer + Handler 实现 - 麻烦【不推荐】

Timer + 普通 Handler - 麻烦

@BindView(R.id.send) Button send;//发送验证码
private int time = 60;//倒计时时间
private Timer timer;

@OnClick({R.id.send, R.id.next})
public void onClickIv(View v) {
switch (v.getId()) {
case R.id.send:
setTimer();//实际是要在点击之后判断如果符合倒计时条件才调用 setTimer 方法,要在失败后调用 destoryTimer
break;
}
}

@Override
protected void onDestroy() {
super.onDestroy();
destoryTimer();
}

@SuppressLint("HandlerLeak")
private Handler handler = new Handler() {
@Override
public void handleMessage(android.os.Message msg) {
switch (msg.what) {
case 1:
send.setText(time + "s");
break;
case 2:
destoryTimer();
break;
}
}
};

//定时器
private void setTimer() {
send.setEnabled(false);//不可点击
timer = new Timer();
TimerTask task = new TimerTask() {
@Override
public void run() {
time--;
if (time > 0) {
handler.sendEmptyMessage(1);
} else {
handler.sendEmptyMessage(2);
}
}
};
timer.schedule(task, 0, 1000);//每隔一秒钟执行一次
}

private void destoryTimer() {
time = 60;//重新倒计时
send.setEnabled(true);//重新可点击
send.setText("重新发送");//重设文字
if (timer != null) {
timer.cancel();
timer = null;
}
if (handler!=null) {
handler.removeCallbacksAndMessages(null);
}
}

Timer + 静态 Handler - 更麻烦

相比示例一,是将Handler定义为了静态内部类,以防止内存泄漏

@BindView(R.id.send) Button send;//发送验证码
private int time = 60;//倒计时时间
private Timer timer;

@OnClick({R.id.send, R.id.next})
public void onClickIv(View v) {
switch (v.getId()) {
case R.id.send:
setTimer();//实际是要在点击之后判断如果符合倒计时条件才调用 setTimer 方法,要在失败后调用 destoryTimer
break;
}
}

@Override
protected void onDestroy() {
super.onDestroy();
destoryTimer();
}

private Handler handler = new MyHandler(this);
private static class MyHandler extends Handler {
private SoftReference<ForgetPasswordActivity> mSoftReference; MyHandler(ForgetPasswordActivity activity) {
mSoftReference = new SoftReference<>(activity);
} @Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
ForgetPasswordActivity activity = mSoftReference.get();
if (activity != null) {
switch (msg.what) {
case 1:
activity.send.setText(activity.time + "s");
break;
case 2:
activity.destoryTimer();
break;
}
}
}
}

//定时器
private void setTimer() {
send.setEnabled(false);//不可点击
timer = new Timer();
TimerTask task = new TimerTask() {
@Override
public void run() {
time--;
if (time > 0) {
handler.sendEmptyMessage(1);
} else {
handler.sendEmptyMessage(2);
}
}
};
timer.schedule(task, 0, 1000);//每隔一秒钟执行一次
}

private void destoryTimer() {
time = 60;//重新倒计时
send.setEnabled(true);//重新可点击
send.setText("重新发送");//重设文字
if (timer != null) {
timer.cancel();
timer = null;
}
if (handler!=null) {
handler.removeCallbacksAndMessages(null);
}
}

Timer + runOnUiThread - 也麻烦

可以不用Handler而用其他更精简的API:

@BindView(R.id.send) Button send;//发送验证码
private int time = 60;//倒计时时间
private Timer timer;

@OnClick({R.id.send, R.id.next})
public void onClickIv(View v) {
switch (v.getId()) {
case R.id.send:
setTimer();//实际是要在点击之后判断如果符合倒计时条件才调用 setTimer 方法,要在失败后调用 destoryTimer
break;
}
}

@Override
protected void onDestroy() {
super.onDestroy();
destoryTimer();
}

private void setTimer() {
send.setEnabled(false);
TimerTask task = new TimerTask() {
@Override
public void run() {
runOnUiThread(() -> {
time--;
if (time > 0) {
send.setText(time + "s");
} else {
destoryTimer();
}
});
}
};
timer = new Timer();
timer.schedule(task, 0, 1000);//每隔一秒钟执行一次
}

private void destoryTimer() {
time = 60;
send.setEnabled(true);
send.setText("获取验证码");
if (timer != null) {
timer.cancel();
timer = null;
}
}

使用纯 Handler 实现 - 特麻烦【强烈不建议】

倒计时通过用 Handler 发送 Delayed 消息来实现。核心代码为:

handler.sendMessageDelayed(handler.obtainMessage(1), 1000);

final Handler handler = new Handler() {
public void handleMessage(Message msg) {
switch (msg.what) {
case 1:
time--;
if (time > 0) {
send.setText(time + "S");
handler.sendMessageDelayed(handler.obtainMessage(1), 1000);//循环发送
} else {
send.setEnabled(true);
send.setText("重新发送");
}
}
}
};

倒计时通过用 Handler 发送 Delayed 的 Runnable 来实现,和上面原理是完全一样的。核心代码为:

Handler handler = new Handler();
handler.postDelayed(runnable, 1000);

Runnable runnable = new Runnable() {
@Override
public void run() {
time--;
if (time > 0) {
send.setText(time + "S");
handler.postDelayed(this, 1000);
} else {
send.setEnabled(true);
send.setText("重新发送");
}
}
};

开源框架 CountdownView 简介

GitHub上星星最多的倒计时控件:CountdownView

CountdownView:Android倒计时控件,使用Canvas绘制,支持多种样式

compile 'com.github.iwgang:countdownview:2.1.3'

引用类名

cn.iwgang.countdownview.CountdownView

基本使用

CountdownView mCountdownView = (CountdownView)findViewById(R.id.countdownView);
mCountdownView.start(995550000); // 毫秒 // 或者自己编写倒计时逻辑,然后调用 updateShow 来更新UI
for (int time=0; time<1000; time++) {
mCountdownView.updateShow(time);
}

其他用法

  • 动态设置自定义属性:.dynamicShow(DynamicConfig)
  • 倒计时结束后的回调:.setOnCountdownEndListener(OnCountdownEndListener);
  • 指定间隔时间的回调:.setOnCountdownIntervalListener(long, OnCountdownIntervalListener);

在 RecyclerView 中实现倒计时

更改数据源方式 - 简单但不可靠

这种方案在数据量特别小(即List的size()特别小),且刷新item及计算倒计时耗费的时间特别短时适用,否则,将会产生巨大的时间延迟。

//定时器,用于刷新 GridView 的数据源
private void setQryTimer() {
cancelQryTimer(); //取消之前的定时器
qryTimer = new Timer(); //重新设置定时器
qryTimer.schedule(new TimerTask() {
@Override
public void run() {
runOnUiThread(() -> {
if (fixRpList != null && fixRpList.size() > 0) {
for (FixRpBean item : fixRpList) {
if (item.diff_time >= 0) item.diff_time = item.diff_time - 1000L; //更改数据源
}
if (fixRpDialog != null) fixRpDialog.upDate(fixRpList); //刷新页面
}
});
}
}, 0, 1000); //以固定的周期刷新
} public void upDate(List<FixRpBean> redPacketList) {
list.clear(); //情况旧的数据
list.addAll(redPacketList); //设置新的数据(如果列表的数据和源数据是同一个集合,也可以直接更新)
mRecyclerView.getAdapter().notifyDataSetChanged();//建议使用RecyclerView的局部刷新功能
}

让 System 帮我们倒计时 - 推荐

核心思想为:利用System.currentTimeMillis()帮我们计算倒计时,并且在onViewAttachedToWindow时重新开始倒计时,在onViewDetachedFromWindow时关闭倒计时。

public class RecyclerViewActivity extends Activity {
private List<ItemInfo> mDataList; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_recyclerview); initData(); RecyclerView recyclerView = (RecyclerView) findViewById(R.id.rv);
recyclerView.setAdapter(new MyAdapter(this, mDataList));
recyclerView.setLayoutManager(new LinearLayoutManager(this));
recyclerView.addItemDecoration(new DividerItemDecoration(this, DividerItemDecoration.VERTICAL));
recyclerView.setItemAnimator(new DefaultItemAnimator());
} private void initData() {
mDataList = new ArrayList<>();
for (int i = 1; i < 20; i++) {
mDataList.add(new ItemInfo(i * 20 * 1000));
}
// 校对倒计时
long curTime = System.currentTimeMillis();
for (ItemInfo itemInfo : mDataList) {
itemInfo.endTime = curTime + itemInfo.countdown;
}
} static class MyAdapter extends RecyclerView.Adapter<MyViewHolder> {
private Context mContext;
private List<ItemInfo> mDatas; public MyAdapter(Context context, List<ItemInfo> datas) {
this.mContext = context;
this.mDatas = datas;
} @Override
public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
return new MyViewHolder(LayoutInflater.from(mContext).inflate(R.layout.list_item, parent, false));
} @Override
public void onBindViewHolder(MyViewHolder holder, int position) {
holder.bindData(mDatas.get(holder.getAdapterPosition()));
} @Override
public int getItemCount() {
return mDatas.size();
} //******************************************** 关键代码 ↓↓ **********************************
@Override
public void onViewAttachedToWindow(MyViewHolder holder) {
super.onViewAttachedToWindow(holder);//父类中为空代码
holder.refreshTime(mDatas.get(holder.getAdapterPosition()).endTime - System.currentTimeMillis());
} @Override
public void onViewDetachedFromWindow(MyViewHolder holder) {
super.onViewDetachedFromWindow(holder);
holder.countdownView.stop();
}
//******************************************** 关键代码 ↑↑ **********************************
} static class MyViewHolder extends RecyclerView.ViewHolder {
public CountdownView countdownView; public MyViewHolder(View itemView) {
super(itemView);
countdownView = (CountdownView) itemView.findViewById(R.id.countdownView);
} public void bindData(ItemInfo itemInfo) {
refreshTime(itemInfo.endTime - System.currentTimeMillis());
} public void refreshTime(long leftTime) {
if (leftTime > 0) {
countdownView.start(leftTime);
} else {
countdownView.stop();//停止计时器,mCustomCountDownTimer.stop();
countdownView.allShowZero();//所有计时清零,即mCountdown.setTimes(0, 0, 0, 0, 0);
}
}
} static class ItemInfo {
public long countdown;
/*
根据服务器返回的countdown换算成手机对应的开奖时间 (毫秒)
[正常情况最好由服务器返回countdown字段,然后客户端再校对成该手机对应的时间,不然误差很大]
*/
public long endTime; public ItemInfo(long countdown) {
this.countdown = countdown;
}
}
}

自己维护倒计时 - 既麻烦又低效

自己维护倒计时,再调用 countdownView.updateShow 来刷新显示

并且根据需要在 onResume 时开启倒计时,在 onPause 及 onDestroy 时关闭倒计时。

//自己维护倒计时,再调用 countdownView.updateShow 来刷新显示
public class RecyclerViewActivity2 extends AppCompatActivity {
private MyAdapter mMyAdapter;
private List<ItemInfo> mDataList; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_recyclerview); initData(); RecyclerView recyclerView = (RecyclerView) findViewById(R.id.rv);
mMyAdapter = new RecyclerViewActivity2.MyAdapter(this, mDataList);
recyclerView.setAdapter(mMyAdapter);
recyclerView.setLayoutManager(new LinearLayoutManager(this));
recyclerView.addItemDecoration(new DividerItemDecoration(this, DividerItemDecoration.VERTICAL));
recyclerView.setItemAnimator(new DefaultItemAnimator());
} private void initData() {
mDataList = new ArrayList<>();
for (int i = 1; i < 20; i++) {
mDataList.add(new ItemInfo(1000 + i, "RecyclerView_测试标题_" + i, i * 20 * 1000));
}
// 校对倒计时
long curTime = System.currentTimeMillis();
for (ItemInfo itemInfo : mDataList) {
itemInfo.setEndTime(curTime + itemInfo.getCountdown());
}
} @Override
protected void onResume() {
super.onResume();
if (null != mMyAdapter)
} @Override
protected void onPause() {
super.onPause();
if (null != mMyAdapter)
} @Override
public void onDestroy() {
super.onDestroy();
if (null != mMyAdapter)
} static class MyAdapter extends RecyclerView.Adapter<MyViewHolder> {
private Context mContext;
private List<ItemInfo> mDatas;
private final SparseArray<MyViewHolder> mCountdownVHList;
private Handler mHandler = new Handler();
private Timer mTimer;
private boolean isCancel = true; public MyAdapter(Context context, List<ItemInfo> datas) {
this.mContext = context;
this.mDatas = datas;
mCountdownVHList = new SparseArray<>();
startRefreshTime(); //开启倒计时
} public void startRefreshTime() {
if (!isCancel) return;
if (null != mTimer) mTimer.cancel(); isCancel = false;
mTimer = new Timer();
mTimer.schedule(new TimerTask() {
@Override
public void run() {
mHandler.post(mRefreshTimeRunnable);
}
}, 0, 10);
} public void cancelRefreshTime() {
isCancel = true;
if (null != mTimer) {
mTimer.cancel();
}
mHandler.removeCallbacks(mRefreshTimeRunnable);
} @Override
public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
return new MyViewHolder(LayoutInflater.from(mContext).inflate(R.layout.list_item, parent, false));
} @Override
public void onBindViewHolder(MyViewHolder holder, int position) {
ItemInfo curItemInfo = mDatas.get(position);
holder.bindData(curItemInfo); // 处理倒计时
if (curItemInfo.getCountdown() > 0) {
synchronized (mCountdownVHList) {
mCountdownVHList.put(curItemInfo.getId(), holder); //开启倒计时
}
}
} @Override
public int getItemCount() {
return mDatas.size();
} @Override
public void onViewRecycled(MyViewHolder holder) {
super.onViewRecycled(holder); ItemInfo curAnnounceGoodsInfo = holder.getBean();
if (null != curAnnounceGoodsInfo && curAnnounceGoodsInfo.getCountdown() > 0) {
mCountdownVHList.remove(curAnnounceGoodsInfo.getId()); //移除
}
} private Runnable mRefreshTimeRunnable = new Runnable() {
@Override
public void run() {
if (mCountdownVHList.size() == 0) return;
synchronized (mCountdownVHList) {
long currentTime = System.currentTimeMillis();
int key;
for (int i = 0; i < mCountdownVHList.size(); i++) {
key = mCountdownVHList.keyAt(i);
MyViewHolder curMyViewHolder = mCountdownVHList.get(key);
if (currentTime >= curMyViewHolder.getBean().getEndTime()) {
curMyViewHolder.getBean().setCountdown(0);// 倒计时结束
mCountdownVHList.remove(key);
notifyDataSetChanged();
} else {
curMyViewHolder.refreshTime(currentTime); //刷新时间
}
}
}
}
};
} static class MyViewHolder extends RecyclerView.ViewHolder {
private TextView mTvTitle;
private CountdownView mCvCountdownView;
private ItemInfo mItemInfo; public MyViewHolder(View itemView) {
super(itemView);
mTvTitle = (TextView) itemView.findViewById(R.id.tv_title);
mCvCountdownView = (CountdownView) itemView.findViewById(R.id.cv_countdownView);
} public void bindData(ItemInfo itemInfo) {
mItemInfo = itemInfo;
if (itemInfo.getCountdown() > 0) {
refreshTime(System.currentTimeMillis());
} else {
mCvCountdownView.allShowZero();
}
mTvTitle.setText(itemInfo.getTitle());
} public void refreshTime(long curTimeMillis) {
if (null == mItemInfo || mItemInfo.getCountdown() <= 0) return;
mCvCountdownView.updateShow(mItemInfo.getEndTime() - curTimeMillis);
} public ItemInfo getBean() {
return mItemInfo;
}
} static class ItemInfo {
private int id;
private String title;
private long countdown;
/*
根据服务器返回的countdown换算成手机对应的开奖时间 (毫秒)
[正常情况最好由服务器返回countdown字段,然后客户端再校对成该手机对应的时间,不然误差很大]
*/
private long endTime; public ItemInfo(int id, String title, long countdown) {
this.id = id;
this.title = title;
this.countdown = countdown;
}
//get、set方法...
} }

2017-6-12

倒计时 总结 Timer Handler CountDownTimer RxJava MD的更多相关文章

  1. 倒计时实现方案总结 Timer Handler

    利用Timer实现倒计时 @BindView(R.id.send) Button send;//发送验证码 private int time = 60;//倒计时 private Timer time ...

  2. Android中三种计时器Timer、CountDownTimer、handler.postDelayed的使用

    在android开发中,我们常常需要用到计时器,倒计时多少秒后再执行相应的功能,下面我就分别来讲讲这三种常用的计时的方法. 一.CountDownTimer 该类是个抽象类,如果要使用这个类中的方法, ...

  3. 115、定时器(TimerTask+Timer+Handler)

    public class TimerUtils { public static Activity act; public static List<MaiDianModels> listMa ...

  4. 线程 Timer TimerTask 计时器 定时任务 MD

    Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱 MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina ...

  5. Android基础之——CountDownTimer类,轻松实现倒计时功能

    在发现这个类之前,一直是用的handler,子线程发消息,UI线程进行倒计时的显示工作.前几天在做一个倒计时显示的时候发现了这个类,用起来非常方便 翻看了下源代码.内部已经帮我们实现了handler的 ...

  6. 拓展 Android 原生 CountDownTimer 倒计时

    拓展 Android 原生 CountDownTimer 倒计时 [TOC] CountDownTimer 在系统的CountDownTimer上进行的修改,主要是拓展了功能,当然也保留了系统默认的模 ...

  7. [Android Pro] CountDownTimer倒计时

    定时执行在一段时候后停止的倒计时,在倒计时执行过程中会在固定间隔时间得到通知(译者:触发onTick方法),下面的例子显示在一个文本框中显示一个30s倒计时: new CountdownTimer(3 ...

  8. Android短轮询解决方案——CountDownTimer+Handler

    转载请注明原文地址:http://www.cnblogs.com/ygj0930/p/7657194.html 一:应用场景 在诸如自动售卖机之类的扫码支付场景中,客户端在获得支付二维码或者发出支付请 ...

  9. android中倒计时控件CountDownTimer分析

    android中倒计时控件CountDownTimer分析 1 示例代码 new CountDownTimer(10000, 1000) { public void onTick(long milli ...

随机推荐

  1. C++雾中风景7:闭包

    本来说好要聊一聊命名空间的,因为最近在看C++lambda表达式的内容,所以借这个机会我们来好好聊一聊C++的闭包. 1.什么是闭包? 闭包(closure)是函数式编程的重要的语法结构. 闭包的概念 ...

  2. 试水jdk8 stream

    jdk8出来日子不短了,jdk11都出来了,不过用的最多的不过是1.5罢了. 今年终于鼓起勇气认真对待它,在18年记录下学习stream,画上一个圆. 先看个图 Java8中有两大最为重要的改变.第一 ...

  3. 1024 Palindromic Number (25)(25 point(s))

    problem A number that will be the same when it is written forwards or backwards is known as a Palind ...

  4. Python进阶篇:Python简单爬虫

    目录 前言 要解决的问题 设计方案 代码说明 小结 前言 前一段一直在打基础,已经学习了变量,流程控制,循环,函数这几块的知识点,就想通过写写小程序来实践一下,来加深知识点的记忆和理解.首先考虑的就是 ...

  5. BZOJ 1449: [JSOI2009]球队收益 最小费用最大流 网络流

    https://www.lydsy.com/JudgeOnline/problem.php?id=1449 给每条路加上一个权值,每条路的费用是这条路的流量*权值,求最大流的最小费用. 每次spfa记 ...

  6. Codeforces Round #349 (Div. 1) B. World Tour 暴力最短路

    B. World Tour 题目连接: http://www.codeforces.com/contest/666/problem/B Description A famous sculptor Ci ...

  7. SQL Server Management Studio 教程一:设置sa用户登录

    今天在net项目中添加数据库过程中出现了小问题,就是使用sql server身份验证没登录成功,经过一番调试,终于解决问题. 使用sa账户登录sql server 2008 的方法步骤如下: 1.首先 ...

  8. centos 6.5安装rvm 配置 Ruby开发环境

    我是用ruby写测试脚本用  安装rvm也是费了好大劲  英文不易看懂 ,是个硬伤! rvm是ruby的版本管理工具  还可对ruby进行 安装 卸载 等 1.安装 curl #  sudo yum ...

  9. 自动化运维之 ~cobbler~

    一 .Cobbler简介 Cobbler是一个快速网络安装linux的服务,而且在经过调整也可以支持网络安装windows.该工具使用python开发,小巧轻便(才15k 行python代码),使用简 ...

  10. java基础学习总结——开篇

    java是我学习的第一门编程语言,当初学习java基础的时候下了不少功夫,趁着这段时间找工作之际,好好整理一下以前学习java基础时记录的笔记,当作是对java基础学习的一个总结吧,将每一个java的 ...