多媒体-图片、音频、视频的基本实现

前言

iOS开发中关于多媒体是经常使用的,下面就简单总结下基本的实现方式。当然首先需要获得系统的私有设置访问权限。(info.plist中添加)


选取系统图片

  • 选择相册中图片

    UIImagePickerController *imagePickerController = [[UIImagePickerController alloc] init];
      imagePickerController.delegate = self;
      imagePickerController.allowsEditing = YES;
      imagePickerController.sourceType = UIImagePickerControllerSourceTypePhotoLibrary;
      [self presentViewController:imagePickerController animated:YES completion:nil];
    
  • 拍照

     if (![UIImagePickerController isSourceTypeAvailable:UIImagePickerControllerSourceTypeCamera]) {
          UIAlertAction *suerAction = [UIAlertAction actionWithTitle:@"确定" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
          }];
          UIAlertController *alertVC = [UIAlertController alertControllerWithTitle:@"温馨提示" message:@"相机不可用" preferredStyle:UIAlertControllerStyleAlert];
          [alertVC addAction:suerAction];
          [self presentViewController:alertVC animated:YES completion:nil];
          
          return;
      }
      UIImagePickerController *picker = [[UIImagePickerController alloc] init];
      picker.delegate = self;
      picker.allowsEditing = YES;
      picker.sourceType = UIImagePickerControllerSourceTypeCamera;
      //设置 前后摄像头  UIImagePickerControllerCameraDeviceFront 前置 UIImagePickerControllerCameraDeviceRear 后置摄像头
      picker.cameraDevice = UIImagePickerControllerCameraDeviceRear;
      [self presentViewController:picker animated:YES completion:nil];
      [self.imagePickerController takePicture];
    

音频

Core Audio 是iOS和 MAC 的关于数字音频处理的基础,它提供应用程序用来处理音频的一组软件框架,所有关于IOS音频开发的接口都是由Core Audio来提供或者经过它提供的接口来进行封装的,按照官方的说法是集播放,音频处理录制为一体的专业技术,通过它我们的程序可以同时录制,播放一个或者多个音频流,自动适应耳机,蓝牙耳机等硬件,响应各种电话中断,静音,震动等,甚至提供3D效果的音乐播放。
Core Audio有5个框架:1.Core Audio.framework,2.AudioToolbox.framework,3.AudioUnit.framework ,4.AVFoundation.framework,5.OpenAL.framework。
Core Audio.framework并不提供服务,仅提供其他框架可以使用的头文件和数据类型。这其中AVFoundation 框架 (AVFoundation.framework)提供一组播放、记录和管理声音和视频内容的Objective-C类,因此下面我就简单介绍一下他就可以了。

AVFoundation的录音和播放
音频的录制与播放主要和三个类有关AVAudioSession,AVAudioRecorder,AVAudioPlayer。
AVAudioSession
AVAudioSession类由AVFoundation框架引入,每个iOS应用都有一个音频会话,这个会话可以被AVAudioSession类的sharedInstance类方法访问,如下:

AVAudioSession *audioSession = [AVAudioSession sharedInstance];
在获得一个AVAudioSession类的实例后,你就能通过调用音频会话对象的setCategory:error:实例方法,来从IOS应用可用的不同类别中作出选择。

AVAudioRecorder
在使用AVAudioRecorder进行音频录制的时候,需要设置一些参数,下面就是参数的说明,并且写下了音频录制的代码:

//音频开始录制

- (void)startRecordWithFilePath:(NSString *)path{

[[AVAudioSession sharedInstance] setCategory: AVAudioSessionCategoryPlayAndRecord error:nil];
[[AVAudioSession sharedInstance] setActive:YES error:nil];
/***
AVFormatIDKey  音乐格式,这里采用PCM格式
AVSampleRateKey 采样率   
AVNumberOfChannelsKey 音乐通道数
AVLinearPCMBitDepthKey,采样位数 默认 16
AVLinearPCMIsFloatKey,采样信号是整数还是浮点数
AVLinearPCMIsBigEndianKey,大端还是小端 是内存的组织方式
AVEncoderAudioQualityKey,音频编码质量   
*/

 NSMutableDictionary *recordSettings = [[NSMutableDictionary alloc] init];
[recordSettings setValue :[NSNumber numberWithInt:kAudioFormatLinearPCM] forKey: AVFormatIDKey];
[recordSettings setValue :[NSNumber numberWithFloat:11025.0] forKey: AVSampleRateKey];//44100.0
[recordSettings setValue :[NSNumber numberWithInt:2] forKey: AVNumberOfChannelsKey];
//[recordSettings setValue :[NSNumber numberWithInt:16] forKey: AVLinearPCMBitDepthKey];
[recordSettings setValue:[NSNumber numberWithInt:AVAudioQualityMin] forKey:AVEncoderAudioQualityKey];  

//初始化录音
self.recorder = [[AVAudioRecorder alloc]initWithURL:[NSURL URLWithString:path]
settings:recordSettings  error:nil];
_recorder.delegate = self;
_recorder.meteringEnabled = YES;
[_recorder prepareToRecord];
[_recorder record];
}

//音频停止录制
 - (void)stopRecord
{
[self.recorder stop];
self.recorder = nil;
}

AVAudioPlayer
AVAudioPlayer类是音频播放的类,一个AVAudioPlayer只能播放一个音频,如果你想混音你可以创建多个AVAudioPlayer实例,每个相当于混音板上的一个轨道,下面就是音频播放的方法。

 //音频开始播放
- (void)startPlayAudioFile:(NSString *)fileName{
//初始化播放器
 player = [[AVAudioPlayer alloc]init];
player = [player initWithContentsOfURL:[NSURL URLWithString:fileName] error:nil];
self.player.delegate = self;   
[player play];
}

//音频停止播放
- (void)stopPlay{
if (self.player) {
     [self.player stop];     
    self.player.delegate = nil;
    self.player = nil;  
}
}

关于音频的播放

   #播放一个视屏
  self.playerLayer = [[AVPlayerLayer alloc]init];
  self.playerLayer.frame = CGRectMake(Scale_X(10), Scale_Y(10), _resultView.width-Scale_X(20), _resultView.height-Scale_X(20));
  [_resultView.layer addSublayer:self.playerLayer];

  AVPlayerLayer *playerLayer;
  //播放设置
  AVPlayer *myPlayer = [AVPlayer playerWithURL:data];
  _playerLayer.player = myPlayer;
  [myPlayer play];

首先了解一下音频播放的实现级别:

  • 离线播放:这里并不是指应用不联网,而是指播放本地音频文件,包括先下完完成音频文件再进行播放的情况,这种使用AVFoundation里的AVAudioPlayer可以满足
  • 在线播放:使用AVFoundation的AVPlayer可以满足
  • 在线播放同时存储文件:使用AudioFileStreamer + AudioQueue 可以满足
  • 在线播放且带有音效处理:使用AudioFileStreamer + AudioQueue + 音效模块(系统自带或者自行开发)来满足

AVAudioPlayer 播放在线音频,会把在线音频完全下载完之后才会播放。如果音频很大,要等待很长时间 ,所以说嘛,AVAudioPlayer压根是不能播放流媒体的。完全下载后才播放就不能算在线播放了
所有苹果公司提供了功能强大的AVPlayer,AVPlayer存在于AVFoundation中,其实它是一个视频播放器,但是用它来播放音乐是没问题的,当然播放音乐不需要呈现界面,因此我们不需要实现它的界面。支持本地和网链,更加接近底层,定制也更加灵活。

  AVPlayer简单一句初始化:
  AVPlayer *newPlayer = [[AVPlayer alloc] initWithURL:[NSURL URLWithString:urlStr]]; //在线
  AVPlayer *newPlayer = [[AVPlayer alloc] initWithURL:    [NSURL fileWithString:urlStr]];   //本地
  #或者通过 playItem 进行初始化


#获取在线音频文件的时长,是通过 playItem 的 playItem.asset.duration
#而不是  playItem.duration ,使用后者根本无法获得时长而是得到一个@“nan”的字符。
#每个cell都需要获取时长的话,在 Tb滑动的时候会有明显的卡顿现象。使用GCD多线程可以解决这个问题

   //使用多线程解决每个cell获取时长造成的卡顿现象
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        self.playItem = [[AVPlayerItem alloc]initWithURL:[NSURL URLWithString:interNetUrl]];
        CMTime duration = self.playItem.asset.duration;
        float seconds = CMTimeGetSeconds(duration);
        
        if (seconds) {
            dispatch_async(dispatch_get_main_queue(), ^{
                self.timeLabel.text = [NSString stringWithFormat:@"%.2f\"",CMTimeGetSeconds(self.playItem.asset.duration)];
            });
        }
        
    });
}

视屏

  • 录制视屏

    -(UIImagePickerController *)VideoPickerController
    {
    if (!_VideoPickerController)
    {
      _VideoPickerController = [[UIImagePickerController alloc] init];
      _VideoPickerController.sourceType = UIImagePickerControllerSourceTypeCamera;
      _VideoPickerController.cameraDevice = UIImagePickerControllerCameraDeviceRear;
      _VideoPickerController.mediaTypes = @[(NSString *)kUTTypeMovie];
      _VideoPickerController.videoQuality = UIImagePickerControllerQualityTypeIFrame1280x720;
      _VideoPickerController.cameraCaptureMode = UIImagePickerControllerCameraCaptureModeVideo;
      _VideoPickerController.allowsEditing = YES;
      _VideoPickerController.delegate = self;
    }
     return _VideoPickerController;
    }
    [self presentViewController:self.VideoPickerController animated:YES completion:nil];
    
  • 选择系统的视屏

    - (UIImagePickerController *)ZYQPick{
      if (!_ZYQPick)
    {
      _ZYQPick = [[UIImagePickerController alloc] init];
      //sourcetype有三种分别是camera,photoLibrary和photoAlbum
      _ZYQPick.sourceType = UIImagePickerControllerSourceTypePhotoLibrary;
      //Camera所支持的Media格式都有哪些,共有两个分别是@"public.image",@"public.movie"
      NSArray *availableMedia = [UIImagePickerController availableMediaTypesForSourceType:UIImagePickerControllerSourceTypeCamera];
      //设置媒体类型为public.movie
      _ZYQPick.mediaTypes = [NSArray arrayWithObject:availableMedia[1]];
      _ZYQPick.delegate = self;//设置委托
    }
        return _ZYQPick;
    }
    [self presentViewController:self.ZYQPick animated:YES completion:NULL];
    
  • 处理选择后的视频或者录制后的视频

    #pragma mark - UIImagePickerControllerDelegate
    - (void)imagePickerController:(UIImagePickerController *)picker didFinishPickingMediaWithInfo:(NSDictionary *)info {
    NSString *mediaType = [info objectForKey:UIImagePickerControllerMediaType];
     #视频
    if ([mediaType isEqualToString:(NSString *)kUTTypeMovie])
    {
    //        NSLog(@"VIDEO....");
    //        NSURL *url = [info objectForKey:UIImagePickerControllerMediaURL];
    //        NSString *urlStr = [url path];
    //        if (UIVideoAtPathIsCompatibleWithSavedPhotosAlbum(urlStr))
    //        {
                  #保存到相册
    //            UISaveVideoAtPathToSavedPhotosAlbum(urlStr, self, @selector(video:didFinishSavingWithError:contextInfo:), nil);
    //        }
      
      NSURL *sourceURL = [info objectForKey:UIImagePickerControllerMediaURL];
      NSURL *newVideoUrl ; //一般.mp4
      NSDateFormatter *formater = [[NSDateFormatter alloc] init];//用时间给文件全名,以免重复,在测试的时候其实可以判断文件是否存在若存在,则删除,重新生成文件即可
      [formater setDateFormat:@"yyyy-MM-dd-HH:mm:ss"];
      //这个是保存在app自己的沙盒路径里,后面可以选择是否在上传后删除掉。我建议删除掉,免得占空间。
      newVideoUrl = [NSURL fileURLWithPath:[NSHomeDirectory() stringByAppendingFormat:@"/Documents/output-%@.mp4", [formater stringFromDate:[NSDate date]]]] ;
      [picker dismissViewControllerAnimated:YES completion:nil];
      # 对视频进行压缩
      [self convertVideoQuailtyWithInputURL:sourceURL outputURL:newVideoUrl completeHandler:nil];        
    }
    #图片
    else
    {
        #获取到编辑后的图片
       chosenImage = info[UIImagePickerControllerEditedImage];     
    }
    [self dismissViewControllerAnimated:YES completion:nil]; 
    }
    - (void)imagePickerControllerDidCancel:(UIImagePickerController *)picker {
      [self dismissViewControllerAnimated:YES completion:NULL];
    }
    
关于视频的播放
#import <AVFoundation/AVFoundation.h>
@property (nonatomic,strong)AVPlayerLayer *playerLayer;//播放器layer,用于录制完视频后播放视频

  //视频 
 self.playerLayer = [[AVPlayerLayer alloc]init];
 self.playerLayer.frame = CGRectMake(Scale_X(10), Scale_Y(10), _resultView.width-Scale_X(20), _resultView.height-Scale_X(20));
 [_resultView.layer addSublayer:self.playerLayer];

//播放设置
 AVPlayer *myPlayer = [AVPlayer playerWithURL:data];
 _playerLayer.player = myPlayer;
[myPlayer play];

PS:AVPlayer视频播放完成的通知监听

[[NSNotificationCenter defaultCenter] 
  addObserver:self
  selector:@selector(videoPlayEnd)
  name:AVPlayerItemDidPlayToEndTimeNotification 
  object:nil];

持续更新中

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351

推荐阅读更多精彩内容