开源的.NET媒体文件操作组件TagLib#解析

  人生得意须尽欢 莫使金樽空对月。写文章都会在吃饭后,每次吃饭都要喝上二两小酒,写文章前都要闲扯,这些都是个人爱好,改不掉了,看不惯的人,还望多多包含一下,有相同爱好的同学,咱们可以一起喝着小酒一边吹牛逼。总有人跟我说要做正事,但是这个世界什么叫做正事什么叫做闲事呢?见解各不相同吧,我自己喜欢的在我的世界就是正事。

   万事急不得,需要等待时机和足够的积累方可一举成事。

   接着前面的组件进行一个简单的介绍,在写文章之前,自己都会做一个简单的了解和应用,毕竟利人必须先利己嘛。有时候没有必要去写一些简单的demo,拿出来会占用篇幅,在实际的项目中使用起来也不会是那么顺畅,只需要了解其原理和使用场景,在项目中都应该可以很好的应用,真正好的代码是需要经过实际的业务需求打磨和从实际的业务去修改和提炼。所以在这些博文中,会简单的介绍背景、应用方法,以及组件的核心对象。真正深入的了解,需要自己去学习。

   知识就是如此,站在前人的肩膀上去展望。有人发现了0和1,所以我们如今不需要再去设计阴和阳做替代。那些所谓的项目中的干货,我只能说,项目千千万,每一个都有不同的需求和场景,如果要想都学完,那只是一些人拿来装逼的笑话,只要把握核心原理,结合真实业务,以自己的思维去整合,得到的就是最合适的技术和框架。 

   前面扯淡半天,下面就直接进入正题。(如果有人有意见,我觉得没有什么不是一瓶酒解决不掉了,对我有意见,可以约出来干一架,也可以喝一顿,我请...哈哈哈...)

一.TagLib#组件概述

   TagLib#用于处理媒体文件,例如视频,音频和照片等等,TagLib#采用LGPL和MPL两种开源协议。TagLib#是用于读取和编辑几种流行音频格式的元数据的库。目前,它支持 MP3文件的ID3v1和ID3v2,FLAC,MPC,Speex,WavPack,TrueAudio,WAV,AIFF,MP4和ASF文件中的Ogg Vorbis注释和ID3标签和Vorbis注释。

   该组件属于比较老的一种了,在GitHub上一直都在更新修改。该库由2001年开始创建,但是该库一直有人在维护,需要使用到相关功能的同学,可以看看该组件。该组件的当前版本为2.1 。该库的地址:https://github.com/mono/taglib-sharp。

   TagLib#(又名taglib-sharp)是一个用于阅读和写作的库媒体文件中的元数据,包括视频,音频和照片格式。

   这个玩意的文档真是少,国内国外翻遍了,也没找到多少,写一篇不容易啊。

   在这里提供一个该库的扩展:https://github.com/timheuer/taglib-sharp-portable,该扩展库支持.NET Framework 4.5+,Windows 8+,Windows Phone 8.1,Windows Phone Silverlight 8,Xamarin.Android,Xamarin.iOS。

二.TagLib#组件应用

   上面介绍了组件的背景和简单的叙述,下面就该介绍一下简单的应用,毕竟无法应用的组件,没有介绍的意义。这里如果需要编写较为通用的组件代码,可以自己根据项目需求进行总结,欢迎大家将自己的知识进行分享,分享自己的知识积累。

   1.解析照片

  public static TagLib.File ParsePhoto(string path)
        {
           if (string.IsNullOrEmpty(path))
           {
                throw new ArgumentNullException(path);
           }
           TagLib.File file;
            try
            {
                file = TagLib.File.Create(path);
            }
            catch (UnsupportedFormatException uex)
            {
                throw uex;
            }
            var image = file as TagLib.Image.File;
            if (file == null)
            {
                throw new ArgumentNullException("file");
            }
           return image;
        }

2.从文件中加载图片

public static void Main(string [] args)
    {
        if(args.Length < 2) {
            Console.Error.WriteLine("USAGE: mono SetPictures.exe AUDIO_PATH IMAGE_PATH_1[...IMAGE_PATH_N]");
            return;
        }    
        TagLib.File file = TagLib.File.Create(args[0]);
        Console.WriteLine("Current picture count: " + file.Tag.Pictures.Length);        
        Picture [] pictures = new Picture[args.Length - 1];   
        for(int i = 1; i < args.Length; i++) {
            Picture picture = new Picture(args[i]);
            pictures[i - 1] = picture;
        }        
        file.Tag.Pictures = pictures;
        file.Save();      
        Console.WriteLine("New picture count: " + file.Tag.Pictures.Length);
    }

 3.加载音频文件

var tagFile = TagLib.File.Create("ironlionzion.mp3");
var tags = tagFile.GetTag(TagTypes.Id3v2);
string album = tags.Album;
string genre = "Hip-Hop, Rock"; 
var matchingFiles = Directory.GetFiles(@"Folder\SubFolder", "*.mp3",   SearchOption.AllDirectories).Where(x => { var f = TagLib.File.Create(x);   return ((TagLib.Id3v2.Tag) f.GetTag(TagTypes.Id3v2)).JoinedGenres == genre; });
foreach (string f in matchingFiles)
{
     System.IO.File.Move(f, Path.Combine(@"D:\NewFolder", new FileInfo(f).Name));
}

4.执行Tag

 public byte[] ExecuteTagging(byte[] inputFile, string title, string artist, 
            string album, string comment, uint year, string copyright, byte[] image)
        {
            var stream = new MemoryStream();
            var writer = new BinaryWriter(stream);
            writer.Write(inputFile);
            using (var audioFile = TagLib.File.Create(new SimpleFileAbstraction(stream)))
            {
                audioFile.Tag.Title = title;
                audioFile.Tag.Performers = new[] { artist };
                audioFile.Tag.Album = album;
                audioFile.Tag.Comment = comment;
                audioFile.Tag.Genres = new[] { "Podcast" };
                audioFile.Tag.Year = year;
                audioFile.Tag.Copyright = copyright;
                audioFile.Tag.Pictures = new[] { new Picture(image) };
                audioFile.Save();
            }

            stream.Position = 0;
            using (var reader = new BinaryReader(stream))
            {
                return reader.ReadBytes((int)stream.Length);
            }
        }

三.TagLib#组件核心对象解析

    介绍完基本的用法,如果需要了解更多,可以取GitHub自行下载查看源码,有比较详细的介绍,这里就不再做赘述,下面介绍一个该组件的一些核心对象。

1.File.Create()

 public static File Create (IFileAbstraction abstraction,string mimetype,ReadStyle propertiesStyle)
        {
            if(mimetype == null) {
                string ext = String.Empty;                
                int index = abstraction.Name.LastIndexOf (".") + 1;            
                if(index >= 1 && index < abstraction.Name.Length)
                    ext = abstraction.Name.Substring (index,abstraction.Name.Length - index);              
                mimetype = "taglib/" + ext.ToLower(CultureInfo.InvariantCulture);
            }            
            foreach (FileTypeResolver resolver in file_type_resolvers) {
                File file = resolver(abstraction, mimetype,propertiesStyle);            
                if(file != null)return file;
            }        
            if (!FileTypes.AvailableTypes.ContainsKey(mimetype))
                throw new UnsupportedFormatException (
                    String.Format (CultureInfo.InvariantCulture,"{0} ({1})",abstraction.Name,mimetype));        
            Type file_type = FileTypes.AvailableTypes[mimetype];        
            try {
                File file = (File) Activator.CreateInstance(file_type,new object [] {abstraction, propertiesStyle});                
                file.MimeType = mimetype;
                return file;
            } catch (System.Reflection.TargetInvocationException e) {
                PrepareExceptionForRethrow(e.InnerException);
                throw e.InnerException;
            }
        }

    该方法用于创建一个File子类的新实例对于指定的文件抽象,mime类型和读取风格。该方法存在6个重载,接受两个参数,abstraction一个IFileAbstraction对象要使用的时候从当前实例读取和写入。mimetype包含mime类型的string对象在选择要使用的适当类时使用,或langword =null如果扩展名name abstraction被使用。propertiesStyle一个ReadStyle值指定的级别从...阅读媒体信息时使用的细节新实例。该方法返回一个File对象,一个新的实例File从中读取指定抽象。Activator.CreateInstance()指定参数匹配程度最高的构造函数创建指定类型的实例。

2.File.WriteBlock()

 public void WriteBlock (ByteVector data)
        {
            if (data == null)
                throw new ArgumentNullException ("data");            
            Mode = AccessMode.Write;            
            file_stream.Write (data.Data, 0, data.Count);
        }

    该方法将一个数据块写入该代表的文件当前实例在当前查找位置,参数data一个ByteVector包含数据的对象写入当前实例。AccessMode是一个枚举类型,指定当前文件访问操作的类型允许在File的实例上。file_stream.Write (data.Data, 0, data.Count)向当前流中写入字节序列,并将此流中的当前位置提升写入的字节数。

四.总结

  技术没有最好,只有最合适。不是你项目应用了当前最先进的技术,就说你的项目就是最先进的,只有你项目使用了最合适的技术,才可以展现最富有生命力的一面。你在团队写出了大家都不熟悉的方式,即使再新潮,也未见得是最富有生命力的。

  写代码,且写且珍惜。代码是你思想的见证,也是你对生活的主张。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏FSociety

Python使用itchat获取微信好友

最近发现了一个好玩的包itchat,通过调用微信网页版的接口实现收发消息,获取好友信息等一些功能,各位可以移步itchat项目介绍查看详细信息。

4332
来自专栏GuZhenYin

用SignalR 2.0开发客服系统[系列2:实现聊天室]

前言 交流群:195866844 上周发表了 用SignalR 2.0开发客服系统[系列1:实现群发通讯] 这篇文章,得到了很多帮助和鼓励,小弟在此真心的感谢大...

4768
来自专栏区块链

逆向基础(二)

写在前面:最近在学逆向,然后很多人说不知道怎么入门,我也在摸着石头过河,然后想的发一个系列。一来分享学习的过程,大家一起进步,二来就当笔记来记录一下,加深印象。...

1966
来自专栏王清培的专栏

.NET重构—单元测试的代码重构

阅读目录: 1.开篇介绍 2.单元测试、测试用例代码重复问题(大量使用重复的Mock对象及测试数据) 2.1.单元测试的继承体系(利用超类来减少Mock对象的...

2136
来自专栏QQ会员技术团队的专栏

gulp 源码解析(一):Stream 详解

作为前端,我们常常会和 Stream 有着频繁的接触。比如使用 gulp 对项目进行构建的时候,我们会使用 gulp.src 接口将匹配到的文件转为 strea...

2450
来自专栏韩东吉的Unity杂货铺

零基础入门 30:获取移动端手机的电量/时间/网络

大家在玩手游的时候经常会看到游戏里有一些提醒状态,图示如下,右上角有当前手机端的时间,wifi网络状态展示,以及电量的展示效果

1423
来自专栏猿人谷

输入某年某月某日,判断这一天是这一年的第几天

题目:输入某年某月某日,判断这一天是这一年的第几天? 1.程序分析:以3月5日为例,应该先把前两个月的加起来,然后再加上5天即本年的第几天,特殊情况,闰年且输入...

3868
来自专栏王大锤

iOS中的预编译指令的初步探究

4088
来自专栏Gcaufy的专栏

async/await 带你逃离回调地狱

不曾见过天堂就不会畏惧地狱,async/await是目前尝试过的最好的异步回调解决方案,本文通过详细步骤给大家讲解。

5200
来自专栏web前端教室

JavaScript ES6的一些新鲜玩艺儿。。

ES6的那些新东西,现在我了解的也不多。 先从简单的来吧,一些个语法看看,, 以前我们声明个对象得这样: var Obj = { data:function(...

20710

扫码关注云+社区