使用PullToRefresh实现下拉刷新和上拉加载
来源:互联网 发布:mac风扇一直转 编辑:程序博客网 时间:2024/05/06 16:16
http://blog.csdn.net/ueryueryuery/article/details/17440465
http://blog.csdn.net/ueryueryuery/article/details/17440465
http://blog.csdn.net/ueryueryuery/article/details/17440465
http://blog.csdn.net/ueryueryuery/article/details/17440465
使用PullToRefresh实现下拉刷新和上拉加载
PullToRefresh是一套实现非常好的下拉刷新库,它支持:
1.ListView
2.ExpandableListView
3.GridView
4.WebView
等多种常用的需要刷新的View类型,而且使用起来也十分方便。
(下载地址:https://github.com/chrisbanes/Android-PullToRefresh)
下载完成,将它导入到eclipse中,作为一个library导入到你的工程中就好了。
一、废话少说,下拉刷新go。
1.在你的布局文件中加上你想用的View就好了,比如这儿我想用一个支持下拉 刷新的ExpandableListView
- <com.handmark.pulltorefresh.library.PullToRefreshExpandableListView
- android:id="@+id/expand_list"
- android:layout_width="match_parent"
- android:layout_height="match_parent" />
2. 在你的Activity代码中进行简单的设置:
- mExpandList = (PullToRefreshExpandableListView) rootView.findViewById(R.id.expand_list);
- mExpandList.getRefreshableView().setGroupIndicator(null);
- mExpandList.getRefreshableView().setDivider(null);
- mExpandList.getRefreshableView().setSelector(android.R.color.transparent);
- mExpandList.getRefreshableView().setOnGroupClickListener(this);
- mExpandList.setOnRefreshListener(this);
第一行是找到这个View,最后一行是为它加上刷新的监听器,中间的几行是我对ExpandableListView进行一些设置。
这样其实就已经可以下拉刷新了,但刷新时需要运行的代码写在哪呢,还有为什么下拉不会收起来呢,且往下看。
3.下拉刷新时执行的方法onRefresh()
- @Override
- public void onRefresh(PullToRefreshBase<ExpandableListView> refreshView) {
- if (!isRefreshing) {
- isRefreshing = true;
- updateList(true);
- } else {
- mExpandList.onRefreshComplete();
- }
- }
一般来说我们会开另一个线程去获取数据,所以这儿会加上一个判断,如果已经在获取数据了,就onRefreshComplete(),就是将下拉收起;否则就去开新线程取数据,取完记得也要onRefreshComplete()哦!
二、上拉加载
如果你不想再费时间去自己写一个上拉加载,不妨试一下PullToRefresh自带的上拉效果哦!
PullToRefresh本身支持下拉刷新和上拉刷新,所以我们只需要将上拉刷新改成上拉加载就行了。
1.设置Mode
- // set mode to BOTH
- mExpandList.setMode(Mode.BOTH);
- mExpandList.getLoadingLayoutProxy(false, true).setPullLabel(getString(R.string.pull_to_load));
- mExpandList.getLoadingLayoutProxy(false, true).setRefreshingLabel(getString(R.string.loading));
- mExpandList.getLoadingLayoutProxy(false, true).setReleaseLabel(getString(R.string.release_to_load));
Mode设置为Mode.BOTH后,下拉和上拉都会执行onRefresh()中的方法了。
因为界面上边,我们要显示“下拉刷新”,下边我们要显示“上拉加载”,所以后三行就是改变下边部分的文字,getLoadingLayoutProxy(false, true)方法大家可以自己感受一下。
2.怎么区分下拉/上拉
网上有的同学是用onScrollListener来判断,这样并不严谨,我依靠是header还是footer处于可见状态来区分下拉和上拉,如果是下拉,那header一定是可见的;反之,footer一定是可见的。
但是PullToRefreshExpandableListView并没有提供这样的接口,那我们就来小改一下我们引入的工程吧,很简单:
找到包“com.handmark.pulltorefresh.library”下的PullToRefreshAdapterViewBase.java这个类,加入两个新接口:
- public boolean isHeaderShown() {
- return getHeaderLayout().isShown();
- }
- public boolean isFooterShown() {
- return getFooterLayout().isShown();
- }
这样就行了哦,重新编译一下这个工程,和你自己的工程。
在onRefresh()中这样来用:
- @Override
- public void onRefresh(PullToRefreshBase<ExpandableListView> refreshView) {
- if (!isRefreshing) {
- isRefreshing = true;
- if (mExpandList.isHeaderShown()) {
- Utils.LOGD("pull-to-refresh");
- refreshOnlineStatus(true);
- } else if (mExpandList.isFooterShown()) {
- Utils.LOGD("pull-to-load-more");
- loadNextPage();
- }
- } else {
- mExpandList.onRefreshComplete();
- }
- }
很简单吧,这样我们就YD地使用PullToRefresh实现了下拉刷新和上拉加载,LOL,希望多多少少能帮到大家。
=================================================================
更新于2014-07-01
近来发现:
1.实现上拉监听,只需要实现OnRefreshListener2就可以了,同时别忘记setMode(Mode.BOTH) 哦!
2.PullToRefreshListView在使用上有一个BUG,在你的xml layout中,不能一开始将它的visiablity设置为GONE,否则,在代码中设置visiablity为VISIABLE也没有作用。
最后放上一张效果图
转载于http://blog.csdn.net/mmjiajia132/article/details/40397813
PullToRefreshListView 用法和ListView 没有什么区别 listview能用的属性 pulltorefresh也能用
我一直认为动手是最好的学习方法...
一:首先看布局文件
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:layout_width="match_parent"
- android:layout_height="match_parent"
- android:orientation="vertical" >
- <!-- ptr:ptrAnimationStyle="flip" flip:翻转 rotate:旋转-->
- <!-- ptr:ptrShowIndicator="true" 右上角 右下角出现箭头-->
- <com.handmark.pulltorefresh.library.PullToRefreshListView
- xmlns:ptr="http://schemas.android.com/apk/res-auto"
- android:id="@+id/pullToRefresh"
- android:layout_width="match_parent"
- android:layout_height="wrap_content"
- ptr:ptrDrawable="@drawable/default_ptr_flip"
- ptr:ptrAnimationStyle="flip"
- ptr:ptrHeaderBackground="#383838"
- ptr:ptrHeaderTextColor="#FFFFFF"
- />
- </LinearLayout>
ptr是pullToRefresh的配置属性 使用是需要添加 xmlns:ptr="http://schemas.android.com/apk/res-auto"
ptr:ptrDrawable=“” 上拉下拉图标
ptr:ptrAnimationStyle="" 图标动画 取值: flip:翻转 rotate旋转
ptr:ptrHeaderBackground="" 上拉下拉时 头部的背景色
ptr:ptrHeaderTextColor="" 上拉下拉时 文字颜色
还有一些常用属性
ptrRefreshableViewBackground 设置整个mPullRefreshListView的背景色
ptrScrollingWhileRefreshingEnabled刷新的时候,是否允许ListView或GridView滚动。觉得为true比较好。
ptrListViewExtrasEnabled 决定了Header,Footer以何种方式加入mPullRefreshListView,true为headView方式加入,就是滚动时刷新头部会一起滚动。
注:上述属性都可以代码添加,请用pullToRefresh.set查看
二:MainActivity代码
- public class MainActivity extends ActionBarActivity {
- private PullToRefreshListView pullToRefresh;
- private List<PullBean> data = new ArrayList<PullBean>();
- MyAdapter adapter;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
- pullToRefresh = (PullToRefreshListView) findViewById(R.id.pullToRefresh);
- data = getData();
- adapter = new MyAdapter(this);
- pullToRefresh.setAdapter(adapter);
- /*
- * Mode.BOTH:同时支持上拉下拉
- * Mode.PULL_FROM_START:只支持下拉Pulling Down
- * Mode.PULL_FROM_END:只支持上拉Pulling Up
- */
- /*
- * 如果Mode设置成Mode.BOTH,需要设置刷新Listener为OnRefreshListener2,并实现onPullDownToRefresh()、onPullUpToRefresh()两个方法。
- * 如果Mode设置成Mode.PULL_FROM_START或Mode.PULL_FROM_END,需要设置刷新Listener为OnRefreshListener,同时实现onRefresh()方法。
- * 当然也可以设置为OnRefreshListener2,但是Mode.PULL_FROM_START的时候只调用onPullDownToRefresh()方法,
- * Mode.PULL_FROM的时候只调用onPullUpToRefresh()方法.
- */
- pullToRefresh.setMode(Mode.BOTH);
- init();
- /*
- * setOnRefreshListener(OnRefreshListener listener):设置刷新监听器;
- * setOnLastItemVisibleListener(OnLastItemVisibleListener listener):设置是否到底部监听器;
- * setOnPullEventListener(OnPullEventListener listener);设置事件监听器;
- * onRefreshComplete():设置刷新完成
- */
- /*
- * pulltorefresh.setOnScrollListener()
- */
- // SCROLL_STATE_TOUCH_SCROLL(1) 正在滚动
- // SCROLL_STATE_FLING(2) 手指做了抛的动作(手指离开屏幕前,用力滑了一下)
- // SCROLL_STATE_IDLE(0) 停止滚动
- /*
- * setOnLastItemVisibleListener
- * 当用户拉到底时调用
- */
- /*
- * setOnTouchListener是监控从点下鼠标 (可能拖动鼠标)到放开鼠标(鼠标可以换成手指)的整个过程 ,他的回调函数是onTouchEvent(MotionEvent event),
- * 然后通过判断event.getAction()是MotionEvent.ACTION_UP还是ACTION_DOWN还是ACTION_MOVE分别作不同行为。
- * setOnClickListener的监控时间只监控到手指ACTION_DOWN时发生的行为
- */
- pullToRefresh.setOnRefreshListener(new OnRefreshListener2<ListView>(){
- @Override
- public void onPullDownToRefresh(
- PullToRefreshBase<ListView> refreshView) {
- // TODO Auto-generated method stub
- PullBean bean = new PullBean();
- bean.setTitle("下拉刷新");
- bean.setContent("我的神");
- adapter.addFirst(bean);
- new FinishRefresh().execute();
- adapter.notifyDataSetChanged();
- }
- @Override
- public void onPullUpToRefresh(
- PullToRefreshBase<ListView> refreshView) {
- // TODO Auto-generated method stub
- PullBean bean = new PullBean();
- bean.setTitle("上拉刷新");
- bean.setContent("我的神");
- adapter.addLast(bean);
- new FinishRefresh().execute();
- adapter.notifyDataSetChanged();
- }
- });
- // pullToRefresh.setOnRefreshListener(new OnRefreshListener<ListView>() {
- //
- // @Override
- // public void onRefresh(PullToRefreshBase<ListView> refreshView) {
- // // TODO Auto-generated method stub
- // String label = DateUtils.formatDateTime(getApplicationContext(), System.currentTimeMillis(),
- // DateUtils.FORMAT_SHOW_TIME | DateUtils.FORMAT_SHOW_DATE | DateUtils.FORMAT_ABBREV_ALL);
- //
- // // Update the LastUpdatedLabel
- // refreshView.getLoadingLayoutProxy().setLastUpdatedLabel(label);
- // PullBean bean = new PullBean();
- // bean.setTitle("我的神");
- // bean.setContent("我的神");
- // adapter.addFirst(bean);
- // new FinishRefresh().execute();
- // }
- //
- // });
- }
- private void init()
- {
- ILoadingLayout startLabels = pullToRefresh
- .getLoadingLayoutProxy(true, false);
- startLabels.setPullLabel("下拉刷新...");// 刚下拉时,显示的提示
- startLabels.setRefreshingLabel("正在载入...");// 刷新时
- startLabels.setReleaseLabel("放开刷新...");// 下来达到一定距离时,显示的提示
- ILoadingLayout endLabels = pullToRefresh.getLoadingLayoutProxy(
- false, true);
- endLabels.setPullLabel("上拉刷新...");// 刚下拉时,显示的提示
- endLabels.setRefreshingLabel("正在载入...");// 刷新时
- endLabels.setReleaseLabel("放开刷新...");// 下来达到一定距离时,显示的提示
- // // 设置下拉刷新文本
- // pullToRefresh.getLoadingLayoutProxy(false, true)
- // .setPullLabel("上拉刷新...");
- // pullToRefresh.getLoadingLayoutProxy(false, true).setReleaseLabel(
- // "放开刷新...");
- // pullToRefresh.getLoadingLayoutProxy(false, true).setRefreshingLabel(
- // "正在加载...");
- // // 设置上拉刷新文本
- // pullToRefresh.getLoadingLayoutProxy(true, false)
- // .setPullLabel("下拉刷新...");
- // pullToRefresh.getLoadingLayoutProxy(true, false).setReleaseLabel(
- // "放开刷新...");
- // pullToRefresh.getLoadingLayoutProxy(true, false).setRefreshingLabel(
- // "正在加载...");
- }
- private List<PullBean> getData(){
- List<PullBean> list = new ArrayList<PullBean>();
- for(int i = 0;i < 10;i ++){
- PullBean bean = new PullBean();
- bean.setTitle("item " + i + " 搜索业务增速下滑 Google廉颇老矣?");
- bean.setContent("Google于10月17日发布了2014年第三季度财报");
- list.add(bean);
- }
- return list;
- }
- private class FinishRefresh extends AsyncTask<Void, Void, Void>{
- @Override
- protected Void doInBackground(Void... params) {
- try {
- Thread.sleep(1000);
- } catch (InterruptedException e) {
- }
- return null;
- }
- @Override
- protected void onPostExecute(Void result){
- // adapter.notifyDataSetChanged();
- pullToRefresh.onRefreshComplete();
- }
- }
- private class MyAdapter extends BaseAdapter{
- private LayoutInflater mInflater;
- public MyAdapter(Context context) {
- // TODO Auto-generated constructor stub
- mInflater = LayoutInflater.from(context);
- }
- public void addFirst(PullBean bean){
- data.add(0, bean);
- }
- public void addLast(PullBean bean){
- data.add(bean);
- }
- @Override
- public int getCount() {
- // TODO Auto-generated method stub
- return data.size();
- }
- @Override
- public Object getItem(int position) {
- // TODO Auto-generated method stub
- return data.get(position);
- }
- @Override
- public long getItemId(int position) {
- // TODO Auto-generated method stub
- return 0;
- }
- @Override
- public View getView(int position, View convertView, ViewGroup parent) {
- // TODO Auto-generated method stub
- ViewHolder viewHolder = null;
- if(convertView == null){
- viewHolder = new ViewHolder();
- convertView = mInflater.inflate(R.layout.item, null);
- viewHolder.title = (TextView) convertView.findViewById(R.id.title);
- viewHolder.content = (TextView) convertView.findViewById(R.id.content);
- convertView.setTag(viewHolder);
- }else{
- viewHolder = (ViewHolder) convertView.getTag();
- }
- viewHolder.title.setText(data.get(position).getTitle());
- viewHolder.content.setText(data.get(position).getContent());
- return convertView;
- }
- class ViewHolder{
- TextView title;
- TextView content;
- }
- }
- }
pullToRefresh适配器Adapter和listview也是继承于BaseAdapter 看一下item的布局
item.xml
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:layout_width="match_parent"
- android:layout_height="match_parent"
- android:padding="5dp"
- android:orientation="vertical" >
- <TextView
- android:id="@+id/title"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:textSize="18sp"
- android:textColor="#BA55D3"
- android:text="我的神"/>
- <TextView
- android:id="@+id/content"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:textSize="14.0sp"
- android:layout_marginTop="5dp"
- android:textColor="#7CFC00"
- android:text="我的神"/>
- </LinearLayout>
pullToRefresh 通过setMode来设置是否可以上拉下拉
Mode.BOTH:同时支持上拉下拉
Mode.PULL_FROM_START:只支持下拉Pulling Down
Mode.PULL_FROM_END:只支持上拉Pulling Up
也可以用 ptr:ptrMode="both"
可选值为:disabled(禁用下拉刷新),pullFromStart(仅支持下拉刷新),pullFromEnd(仅支持上拉刷新),both(二者都支持),manualOnly(只允许手动触发)
如果Mode设置成Mode.BOTH,需要设置刷新Listener为OnRefreshListener2,并实现onPullDownToRefresh()、onPullUpToRefresh()两个方法。
如果Mode设置成Mode.PULL_FROM_START或Mode.PULL_FROM_END,需要设置刷新Listener为OnRefreshListener,同时实现onRefresh()方法。
当然也可以设置为OnRefreshListener2,但是Mode.PULL_FROM_START的时候只调用onPullDownToRefresh()方法,Mode.PULL_FROM的时候只调用onPullUpToRefresh()方法.
如果想上拉、下拉刷新的时候 做一样的操作,那就用OnRefreshListener,上拉下拉的时候都调用
如果想上拉、下拉做不一样的的操作,那就在setOnRefreshListener时 用new OnRefreshListener2<ListView>
当然如果想自己设置上拉下拉中的文字 可以这样
- ILoadingLayout startLabels = pullToRefresh
- .getLoadingLayoutProxy(true, false);
- startLabels.setPullLabel("下拉刷新...");// 刚下拉时,显示的提示
- startLabels.setRefreshingLabel("正在载入...");// 刷新时
- startLabels.setReleaseLabel("放开刷新...");// 下来达到一定距离时,显示的提示
- ILoadingLayout endLabels = pullToRefresh.getLoadingLayoutProxy(
- false, true);
- endLabels.setPullLabel("上拉刷新...");// 刚下拉时,显示的提示
- endLabels.setRefreshingLabel("正在载入...");// 刷新时
- endLabels.setReleaseLabel("放开刷新...");// 下来达到一定距离时,显示的提示
当然也可以这样
- pullToRefresh.getLoadingLayoutProxy(false, true)
- .setPullLabel("上拉刷新...");
- pullToRefresh.getLoadingLayoutProxy(false, true).setReleaseLabel(
- "放开刷新...");
- pullToRefresh.getLoadingLayoutProxy(false, true).setRefreshingLabel(
- "正在加载...");
- // 设置上拉刷新文本
- pullToRefresh.getLoadingLayoutProxy(true, false)
- .setPullLabel("下拉刷新...");
- pullToRefresh.getLoadingLayoutProxy(true, false).setReleaseLabel(
- "放开刷新...");
- pullToRefresh.getLoadingLayoutProxy(true, false).setRefreshingLabel(
- "正在加载...");
显然在实际操作的时候也会用到其他监听
setOnScrollListener()
SCROLL_STATE_TOUCH_SCROLL 正在滚动
SCROLL_STATE_FLING 手指做了抛的动作(手指离开屏幕前,用力滑了一下)
SCROLL_STATE_IDLE 停止滚动
setOnLastItemVisibleListener
当用户拉到底时调用
setOnItemClickListener()
为pullToRefresh中每一个item设置事件
代码下载:点击下载代码
下拉上拉 图标和文字 位置改动是在PullToRefresh源代码中改的即:PullToRefreshListView.handleStyledAttributes 中lp的Gravity改为CENTER_VERTICAL
如果想要改动图标和文字的距离和布局 在这library项目下这两个文件改
pull_to_refresh_header_horizontal.xml
pull_to_refresh_header_vertical.xml
参考博客:
http://blog.csdn.net/lmj623565791/article/details/38238749
http://blog.csdn.net/harvic880925/article/details/17680305
谢谢原作者
持续更新ing MMjiajia132
、、、、、、、、、、、、、、、、、
、、、、、、、、、、、、、、、、、
、、、、、、、、、、、、、、、、、
国外源码精品-Android-PullToRefresh 简介与DEMO导入
目录(?)[+]
转载地址:http://my.oschina.net/cuitongliang/blog/170708 (一)&&http://my.oschina.net/cuitongliang/blog/170737 (二)
PLEASE NOTE, THIS PROJECT IS NO LONGER BEING MAINTAINED
一、介绍
Pull To Refresh Views for Android
This project aims to provide a reusable Pull to Refresh widget for Android. It was originally based on Johan Nilsson's library (mainly for graphics, strings and animations), but these have been replaced since.
Features
- Supports both Pulling Down from the top, and Pulling Up from the bottom (or even both).
- Animated Scrolling for all devices.
- Over Scroll supports for devices on Android v2.3+.
- Currently works with:
- ListView
- ExpandableListView
- GridView
- WebView
- ScrollView
- HorizontalScrollView
- ViewPager
- Integrated End of List Listener for use of detecting when the user has scrolled to the bottom.
- Maven Support.
- Indicators to show the user when a Pull-to-Refresh is available.
- Support for ListFragment!
- Lots of Customisation options!
Repository at https://github.com/chrisbanes/Android-PullToRefresh.
附源码截图:
ListView ExpandableListView
GridView WebView
ScrollView Horizontal ScrollView
ViewPager ListView Fragment
WebView Advanced ListView in ViewPager
二、DEMO导入
很多人看到有好源码,但是在使用时碰到了问题。在此简单介绍一下,希望能够对那些不会的童鞋们有所帮助。
首先下载源码,源码地址:https://github.com/chrisbanes/Android-PullToRefresh。
下载完源码之后,解压。在Android-PullToRefresh-master文件夹下,我们会看到还有三个文件夹:extras,
library,sample。其中sample就是作者为我们提供的Demo,library是我们在使用Sample必须用到的jar。extras中是使用ListFragment和ViewPage用到的jar。讲解完目录后现在开始导入了。
1. 导入library:导入很简单,如图,点击finish即可。
导入之后没有问题。OK!
2. 同样导入extras中的PullToRefreshListFragment和PullToRefreshViewPager工程。切记:分别导入。
如图:
导入之后,你会发现工程报错了,很简单,没有关联libraray。右键工程--选择Properties,打开之后会发现有个红色的X,这就是报错的原因。改了就可以。
选中红色的报错的library,点击Remove,再点击Add,将Libarary添加上即可!如图:
同理将PullToRefreshViewPager工程的library也修改好。
3. 导入Sample工程,同时修改library。如图:
至此,工程全部导入,错误全部解决。
运行一下,看是否能够正常运行到手机上。接下来就可以慢慢研究着精品源码吧!
感谢原作者详细的讲解,解决了我的问题!
三、DEMO导入出现问题及解决
一、运行报错:Unable to execute dex: Multiple dex files define Lcom/my/cop/miles/R$anim
PullToRefresh基本用法:
1、在布局文件中添加PullToRefresh控件,比如PullToRefreshListView;
2、在Activity中,设置监听器OnRefreshListener以响应用户下拉操作;
3、在监听器的onRefresh()方法中执行数据刷新操作,可以通过AsyncTask来实现;
4、在AsyncTask中获取到数据后,记得调用onRefreshComplete()方法通知PullToRefresh控件数据已获取完毕,可以结束刷新操作。
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- 使用PullToRefresh实现下拉刷新和上拉加载
- PullToRefresh实现上拉加载下拉刷新
- 云存储FSS使用规范
- 二叉树题目
- 机器学习入门-了解相关概念
- SDM For Face Alignment流程介绍及Matlab代码实现之测试篇
- 面向对象笔记
- 使用PullToRefresh实现下拉刷新和上拉加载
- Dijkstra最短路径算法详解
- 双链表:实现基本的增删查改,正反向现实双链表的节点
- 垃圾回收机制与内存分配策略
- 黑马程序员---指向函数的指针与返回指针的函数
- 黑马程序员---c语言基础---数组、指针
- java学习笔记——static关键字
- Ruby On Rails 快速创建项目
- HDU 5344 多个数的和异或-思维-(位运算)