在开发过程中,需要在Mac应用程序中获取到音频的信息,已经导入了FFmpeg三方库,但是发现在查询音频的基本信息时并不好用,所以选择用ffprobe实现获取音频信息的功能。由于在代码中需要调用终端命令执行,因此需要使用NSTask调用,具体过程如下:
- 导入ffprobe库文件到项目中,如果电脑终端已经安装了FFmpeg库,则可以在终端通过
which ffprobe
获取到ffprobe的快捷方式,通过查看原身将其复制到项目中即可。 - 定义工具类ReadFilesTool,专门用于获取音频信息
ReadFilesTool.h文件内容如下:
#import <Foundation/Foundation.h>
#import "SongModel.h"
@interface ReadFilesTool : NSObject
+ (NSArray *)getFileMessage;
+ (SongModel *)getInfoWithPath:(NSString *)path;
@end
ReadFilesTool.m文件内容如下:
//
// ReadFilesTool.m
// musicShow
//
// Created by mac123 on 2018/6/26.
// Copyright © 2018年 mac123sdfsdqweqw. All rights reserved.
//
#import "ReadFilesTool.h"
#import <AVFoundation/AVFoundation.h>
@implementation ReadFilesTool
//获取指定路径下的所有mp3文件信息
+ (NSArray *)getFileMessage {
NSMutableArray *songsArr = [NSMutableArray array];
NSFileManager *fm=[NSFileManager defaultManager];
NSString *musicFile=@"/Users/mac123/Desktop/music";
NSArray *nameArr=[fm contentsOfDirectoryAtPath:musicFile error:nil];
for (NSString *obj in nameArr) {
NSString *allPathStr = [musicFile stringByAppendingPathComponent:obj];
SongModel *songModel = [self getInfoWithPath:allPathStr];
[songsArr addObject:songModel];
}
return songsArr;
}
//通过单个音频的路径获取详细信息
+ (SongModel *)getInfoWithPath:(NSString *)path {
NSArray *arguArr = @[@"-v", @"quiet", @"-print_format", @"json", @"-show_format", @"-show_streams", path];
NSData * getJsonData = [[self getSongMessByTerminalWithArguments:arguArr] dataUsingEncoding:NSUTF8StringEncoding];
NSDictionary * songMessDict = [NSJSONSerialization JSONObjectWithData:getJsonData options:NSJSONReadingMutableContainers error:nil];
NSMutableDictionary *songDic = [NSMutableDictionary dictionary];
//取出歌曲需要使用的信息
NSDictionary *formatDict = [songMessDict objectForKey:@"format"];
if (path.length > 0) {
[songDic setObject:path forKey:@"path"];
}
NSString * duration = [formatDict objectForKey:@"duration"];
if (duration != nil) {
[songDic setObject:duration forKey:@"duration"];
}
NSString *size = [formatDict objectForKey:@"size"];
if (size != nil) {
[songDic setObject:size forKey:@"size"];
}
NSDictionary *tags = [formatDict objectForKey:@"tags"];
NSString *title = [tags objectForKey:@"title"];
if (title != nil) {
[songDic setObject:title forKey:@"title"];
}
NSString *artist = [tags objectForKey:@"artist"];
if (artist != nil) {
[songDic setObject:artist forKey:@"artist"];
}
NSString *album = [tags objectForKey:@"album"];
if (album != nil) {
[songDic setObject:artist forKey:@"album"];
}
SongModel *model = [[SongModel alloc] initWithDict:songDic];
return model;
}
//通过ffprobe获取歌曲信息
+ (NSString *)getSongMessByTerminalWithArguments:(NSArray *)arguments {
NSString *string = nil;
@try {
NSString *ffmpegPath = [[NSBundle mainBundle] pathForResource:@"ffprobe" ofType:@""];
NSTask *task = [[NSTask alloc]init];
[task setLaunchPath:ffmpegPath];
[task setArguments: arguments];
// 新建输出管道作为Task的输出
NSPipe *pipe = [NSPipe pipe];
[task setStandardOutput: pipe];
// 开始task
NSFileHandle *file = [pipe fileHandleForReading];
[task launch];
// 获取运行结果
NSData *data = [file readDataToEndOfFile];
string = [[NSString alloc] initWithData: data encoding: NSUTF8StringEncoding];
}
@catch (NSException *exception) {
NSLog(@"调用命令行失败:%@", exception.reason);
}
@finally {
}
return string;
}
@end
- 对得到的数据进行解析处理,以及进行后续的操作。
难点在于通过ffprobe获取歌曲信息部分,该部分是通过NSTask调用终端将命令结果以字符串形式返回到原程序中,另外,调用ffprobe的完整过程也是需要重点关注的部分,最后在得到结果后进行后续处理即可。