前言

市面上绝大部分的APP被打开之后映入眼帘的都是一个美轮美奂的轮播器,所以能做出一个符合需求、高效的轮播器成为了一个程序员的必备技能。所以今天的这篇博客就来谈谈轮播器这个看似简单的控件其中蕴含的道理。

正文

    首先我们来分析一下该如何去实现一个类似下图的轮播器(图片数量、URL由服务器返回):

    

     策略一:UIScrollView->UIImageView->NSTimer轮询 这算是常规的策略,但是如果仔细想想,如果服务器返回给你50图片是不是就需要创建50个UIImageView来做容器。这种性能肯定不是最优的。其次所有的UIImageView是按照次序排列,脑补一下如果到最后一张图片要重新回到第一张图片你的话,UIScrollView会被升拉到第一个位置。效果太差!

    策略二:想想UITableViewCell的重复利用,我们也可以重复利用其中的UIImageView。对于我们来说轮播器三个UIImageView就足够用了,分为当前视野中的CenterImageView、前一张LeftImageView、后一张RightImageView。不同的是他们之间的图片切换,下面我们就尝试去做高效的轮播器。

代码实现

  1. 初始化所需控件:

  

  2.写一个专门控制没次滑动结束去计算左中右三张序号并加载成对应的图片。

 /** 这里我起名叫reloadAllImageView */

 -(void)reloadAllImageView{

     CGPoint offset = _backScrollView.contentOffset;                       /** 获取到scroll的X轴偏移量 */

        if (offset.x ==  * DeviceWidth){                      /** 这里有两种边界情况要处理 (1).由first—>last (2)last->first */

         /** (2) */

             _centerImageIndex        = (_centerImageIndex + )%;          /** 3代表图片的数量,这里要主页类型的强转换。_centerImageIndex(NSUInteger)用于记录当前图片序号*/

             _pageControl.currentPage = (_pageControl.currentPage + )%;

         }

         else if (offset.x == ){

             /** (1) */

           if (_centerImageIndex == ) {                        /** 一种特殊情况 当_centerImageIndex等于0的时候 去计算(_centerImageIndex - 1) % 3并不是我们想要的结果 */

               _centerImageIndex = ;                          /** 尝试了很久,计算类型转换也就只有三张图片会有问题。如果有兴趣的朋友可以进行深入研究 */

           }                                      /** -1 % 3 = 0   如果都是有符号的结果是-1,如果按照无符号处理的话结果是0。难道计算过程应该先向前借位再进行计算? */

           _centerImageIndex        = (_centerImageIndex - ) % ;    

           _pageControl.currentPage = (_pageControl.currentPage - )%;

         }

       _centerImageView.image         = [UIImage imageNamed:[NSString stringWithFormat:@"图%d.jpg",_centerImageIndex + ]];

       NSUInteger leftImageViewIndex  = (_centerImageIndex - )%;

       NSUInteger  rightImageViewIndex = (_centerImageIndex + )%;

       if (leftImageViewIndex ==  && _centerImageIndex == ) {          /** 同上暂时处理计算特殊情况 */ 

           leftImageViewIndex = ;

       }

       _leftImageView.image           = [UIImage imageNamed:[NSString stringWithFormat:@"图%d.jpg",leftImageViewIndex + ]];

       _rightImageView.image          = [UIImage imageNamed:[NSString stringWithFormat:@"图%d.jpg",rightImageViewIndex + ]];

 }

    3.现在轮播器应该可以在边界正常切换了。现在需要再加上一个计时器来自动滑动即可:

 /** 声明一个定时器 */   /** 用weak的原因:self如果强拥有了Timer,之后你要设置计时器的Traget和选择子selector的时候,Timer又会保留目标对象直到失效。产生保留环 */

 @property(nonatomic,weak)NSTimer * timer;

 /** 定时器初始化 */

    -(void)initTimer

    {

        self.timer = [NSTimer scheduledTimerWithTimeInterval:animationTime target:self selector:@selector(updateImageView:) userInfo:nil repeats:YES];

    }

   -(void)updateImageView:(NSTimer *)timer

   {

       [_backScrollView setContentOffset:CGPointMake(DeviceWidth*, ) animated:YES];

       [NSTimer scheduledTimerWithTimeInterval:0.4f target:self selector:@selector(scrollViewDidEndDecelerating:) userInfo:nil repeats:NO];

   }

 /** 计算加载所有图片然后移动的中间视野 */

   -(void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView{

       //重新加载图片

       [self reloadAllImageView];

       //移动到当前视野

       [_backScrollView setContentOffset:CGPointMake(DeviceWidth, )];

       //设置脚标

       _pageControl.currentPage = _centerImageIndex;

   }

  4.自此轮播器大概雏形已经搞定了。剩下的就是需要搞定计时器和用户滑动操作的互斥事件处理。

/**记录一个bool值用于确定滑动操作的愿意你。YES,计时器触发,NO则为用户触发*/

     ().计时器的触发事件中,肯定是由计时器触发的滑动。所以这里bool值为YES

  ().加载替换图片的时候我们要进行判断(如果是用户触发的时候我们要将计时器取消并从空为0)

    if (!bool) {

          [self.timer setFireDate:[NSDate dateWithTimeIntervalSinceNow:animationTime]];

      }

       bool = NO;

结尾

    由此轮播器就实现了,但是其中还是有问题需要解决<1>.限制必须要三张图片不然会crash <2>没有封装成单独的scrllview以供使用。后续可能会对这些问题加以思考并重新优化。也许尝试用UICollection来做也是个很好的想法。

最后经过写这篇博客也有一个目的,其实任何一个看似简单的功能要深入挖掘的话还是有很说知识的,也发现了自身的不足。最后如果各位大神们看到了博客有任何想法意见的欢迎下面留言。大家一起探讨一起进步。谢谢各位!

iOS开发那些事儿(一)轮播器的更多相关文章

  1. iOS开发UI篇—无限轮播(循环利用)

    iOS开发UI篇—无限轮播(循环利用) 一.无限轮播  1.简单说明 在开发中常需要对广告或者是一些图片进行自动的轮播,也就是所谓的无限滚动. 在开发的时候,我们通常的做法是使用一个UIScrollV ...

  2. iOS开发UI篇—无限轮播(新闻数据展示)

    iOS开发UI篇—无限轮播(新闻数据展示) 一.实现效果        二.实现步骤 1.前期准备 (1)导入数据转模型的第三方框架MJExtension (2)向项目中添加保存有“新闻”数据的pli ...

  3. iOS开发UI篇—无限轮播(循环展示)

    iOS开发UI篇—无限轮播(循环展示) 一.简单说明 之前的程序还存在一个问题,那就是不能循环展示,因为plist文件中只有五个数组,因此第一个和最后一个之后就没有了,下面介绍处理这种循环展示问题的小 ...

  4. iOS开发UI篇—无限轮播(功能完善)

    iOS开发UI篇—无限轮播(功能完善) 一.自动滚动 添加并设置一个定时器,每个2.0秒,就跳转到下一条. 获取当前正在展示的位置. [self addNSTimer]; } -(void)addNS ...

  5. IOS第六天(3:scrollView 图片轮播器)

    IOS第六天(3:scrollView 图片轮播器) #import "HMViewController.h" #define kImageCount 5 @interface H ...

  6. ios之无限 自动 图片轮播器的实现

    比较之前发布的手动无限图片轮播器进行了改进.实现了自动无限轮播的功能.比较适合团购标题分类下面的轮播器功能. 实现思路: * 开启一个定时器,把操作放入消息循环池.每隔一定时间,操作执行一次. * 注 ...

  7. 【iOS开发-55】图片轮播案例:scrollView的分页、滚动栏、利用代理控制定时器和Page Control以及多线程问题

    案例: (1)用storyboard布局,这里用了三样东西. --UIScrollView就是我们准备存放滚动图片的容器. --Page Control就是控制页数的那几个小点点.能够设置有多少个点. ...

  8. ios开发图片轮播器以及定时器小问题

    一:图片轮播器效果如图:能实现自动轮播,到最后一页时,轮播回来,可以实现拖拽滚动 二:代码: #import "ViewController.h" ; @interface Vie ...

  9. ios 学习 广告图片轮播器

    // // ViewController.m // 图片轮播器 // // Created by zjj on 15/5/23. // Copyright (c) 2015年 zjj. All rig ...

随机推荐

  1. 50中制作图表的JS库

    参看以下链接:http://www.tuicool.com/articles/FZNjMz

  2. JOSN的stringify()和parse()方法

    JOSN的stringify()和parse()方法使用,stringify()把对象序列化,parse()把JSON字符串解析为js对象 <script> var book = { ti ...

  3. gulp压缩js

    1.安装nodejs -> 全局安装gulp -> 项目安装gulp以及gulp插件 -> 配置gulpfile.js -> 运行任务 2.查看nodejs的版本号 3.npm ...

  4. C语言函数入门

    由于采用了函数模块式的结构,C语言易于实现结构化程序设计.使程序的层次结构清晰,便于程序的编写.阅读.调试. main 函数是主函数,它可以调用其它函数,而不允许被其它函数调用.因此,C程序的执行总是 ...

  5. display:table-cell的惊天作用,直接惊呆你!

    一 display:table-cell介绍 ... 二 用法 (1)高度不固定元素,垂直居中 ... (2)高度不固定列表元素,登高排列 ... (3)宽度不固定元素,平均分配 ...

  6. 感觉tbceditor很不错,如果作者能坚持下来,非常非常看好啊

    感觉tbceditor很不错,如果作者能坚持下来,非常非常看好啊 你技术好,可以做个自用的IDE慢慢加功能 ,很方便的用这个控件,写个支持Delphi和html混编的编辑器,要不到2个小时

  7. 认识元数据和IL(上) <第三篇>

    说在,开篇之前 很早就有说说Metadata(元数据)和IL(中间语言)的想法了,一直在这篇开始才算脚踏实地的对这两个阶级兄弟投去些细关怀,虽然来得没有<第一回:恩怨情仇:is和as>那么 ...

  8. X windows的底层实现机制

    Qt在Linux上运行崩溃了,很可能的原因是对于X11机制的不了解.很可能是UI代码里面对窗口的操作不规范而导致Qt内部的BUG暴露出来.具体UI实现代码我也没有看.是别人维护的.打算今天去看下代码, ...

  9. unity3d 导出 Excel

    我在unity里需要导出成Excel格式,试了一些方法,其中用c#的com组件的我还没成功不知道该怎么在unity里调用,(如果哪位大哥用别的方法在unity里成功了,可以交流下,最好给我一个小dem ...

  10. ysql+heartbeat+DRBD+LVS实现mysql高可用

    在企业应用中,mysql+heartbeat+DRBD+LVS是一套成熟的集群解决方案,通过heart+DRBD实现mysql的主 节点写操作的高可用性,而通过mysql+LVS实现数据库的主从复制和 ...