logo头像

学如逆水行舟

iOS音频播放器锁屏歌词显示与性能优化

iOS音频播放器锁屏歌词显示与性能优化

一、引言

前边有博客探讨了有关iOS开发中音频播放的技术与进行后台音频播放并在后台与用户进行交互的方法,本篇将探讨一种在锁屏界面同步显示歌词歌词的方法,并在应用性能上进行一些处理。前边博客地址如下:

iOS音频开发AVAudioPlayer的应用:http://my.oschina.net/u/2340880/blog/420129

iOS后台音频开发与交互技术:http://my.oschina.net/u/2340880/blog/420183

二、在锁屏界面同步显示歌词

我们知道,在音频后台播放时,锁屏界面的信息是由MPNowPlayingInfoCenter来设置的,其中的歌曲信息字典可以设置类似歌曲封面,艺术家,歌曲名,歌曲时间等。然而,对于MPNowPlayingInfoCenter中可以由开发者掌握的接口十分有限,若要在锁屏界面同步的显示歌曲歌词,一个比较简单的方法是不停的将当前歌词与封面进行图片合成,之后刷新锁屏界面的图片。下面代码是根据解析好的LRC歌词数据进行图片合成的示例代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
//这个方法通过传入的LRC歌词数据进行图片的合成 array参数为每行lrc歌词数据集合 index为当前需要播放的lrc歌词行数
-(void)setCurretLRCArray:(NSArray *)array index:(int)index{
    //LRCItem是歌词模型 里面的lrc方法获取歌词文字字符串
    NSString * lineLRC = [(LRCItem *)array[index] lrc];
    //提高性能 进行判断 当前显示的歌词有无变化 如果没变化 不进行后续操作
    if ([_lrcLabel.text isEqualToString:lineLRC]) {
        return;
    }
    _lrcLabel.text = lineLRC;
    //将歌词整理成整齐数据
    //进行行数设置
        NSMutableString * lrcStr = [[NSMutableString alloc]init];
        if (index<_lines/2) {
            //前面用\n补齐
            int offset = (int)_lines/2-index;
            for (int j=0; j<offset; j++) {
                [lrcStr appendFormat:@"\n"];
            }
            for (int j=0; j<_lines-offset; j++) {
                [lrcStr appendFormat:@"%@\n",[(LRCItem *)array[j] lrc]];
            }
        } else if (array.count-1-index<_lines/2) {
            //后面用\n补齐
            int offset = (int)_lines/2-(int)(array.count-index-1);
            for (int j=index-(_lines/2); j<array.count; j++) {
                [lrcStr appendFormat:@"%@\n",[(LRCItem *)array[j] lrc]];
            }
            for (int j=0; j<offset; j++) {
                [lrcStr appendFormat:@"\n"];
            }
        }else {
            for (int j=0; j<_lines; j++) {
                [lrcStr appendString:[(LRCItem *)array[index-_lines/2+j] lrc]];
                [lrcStr appendString:@"\n"];
            }
        }
    //将当前显示的歌词部分高亮
    NSMutableAttributedString * attriStr = [[NSMutableAttributedString alloc]initWithString:lrcStr];
    NSRange range = [lrcStr rangeOfString:[array[index] lrc]];
    [attriStr setAttributes:@{NSForegroundColorAttributeName:[UIColor greenColor]} range:range];
    _lrcView.attributedText = attriStr;
    _lrcIMGLabel.attributedText = attriStr;
    //进行截屏
    if (!_lrcIMGbg) {
        _lrcIMGbg = [[UIImageView alloc]initWithFrame:CGRectMake(0, 0, self.frame.size.width, self.frame.size.height)];
        _lrcIMGbg.image = [UIImage imageNamed:@"BG.jpeg"];
        [_lrcIMGbg addSubview:_lrcIMGLabel];

    }
    //获取添加了歌词数据的背景图
    UIGraphicsBeginImageContext(_lrcIMGbg.frame.size);
    CGContextRef context = UIGraphicsGetCurrentContext();
    [_lrcIMGbg.layer renderInContext:context];
    UIImage *img = UIGraphicsGetImageFromCurrentImageContext();
    UIGraphicsEndImageContext();
    _lrcImage = [img copy];
}

锁屏歌词的显示效果如下:

三、应用性能的优化

从功能上来说,通过一直合成刷新锁屏界面的封面图是可行的,然而在使用中我发现,不停和合成图片和刷新操作将使设备非常耗电,如果可以完成下面的优化,则可以大大提高应用的性能:

1.应用在前台时不进行后台锁屏界面的刷新操作。

2.应用在后台时不进行前台歌词信息、歌曲信息的相应更新。

3.当屏幕变黑关闭时,停止所有刷新操作。

4.当屏幕从新被点亮时,进行后台信息刷新。

上面4点需求,前两点十分容易搞定,在AppDelegate中有如下的方法可以监听应用程序前台与后台的切换:
1
2
3
4
5
6
7
8
9
//应用进入后台时调用
- (void)applicationWillResignActive:(UIApplication *)application {
   //发送通知
    [[NSNotificationCenter defaultCenter]postNotificationName:@"goBack" object:nil];
}
//应用进入前台时调用
- (void)applicationDidBecomeActive:(UIApplication *)application { 
    [[NSNotificationCenter defaultCenter]postNotificationName:@"goForward" object:nil];
}

在相应的类中监听这两个通知即可以做到刷新模式的切换。

对于上面需求的第3点与第4点,Foundation框架中没有提供公开的方法来监听屏幕设备的点亮状态,但是可以通过notify\_register\_dispath()方法来监听。使用示例代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
#import <notify.h>

    //在定时器中使用如下代码实时监听屏幕状态
    uint64_t locked;
    __block int token = 0;    
    notify_register_dispatch("com.apple.springboard.hasBlankedScreen",&token,dispatch_get_main_queue(),^(int t){
    });
    notify_get_state(token, &locked);
    //如果屏幕变暗 直接不走更新方法 lock为1则为变暗关闭状态 0则为点亮状态 
    if (locked) {
        return;
    }

通过上面优化的代码,性能将会强很多。

由于前面博客已经详细的介绍了音频开发与后台交互的内容,这篇博客主要探讨了锁屏歌词相关的开发思路与优化方向,提供的示例代码都是片段,并不完整,经过优化的音频播放器源码在如下git地址上,需要的朋友可以作为参考:

同步显示锁屏歌词的音乐播放器:https://github.com/ZYHshao/MyPlayer

专注技术,热爱生活,交流技术,也做朋友。

——珲少 QQ群:203317592