在C#中有许多音频播放的方案,例如WinForm里调用系统自带MediaPlayer的COM组件和WPF的MediaPlayer(实质上还是WindowsMediaPlayer)

以及一堆API播放和DirectX (SDK一大堆)

于是我找到了适用于全平台、高效、小巧的音频解码器--Bass (主程序基于C++ C#可通过官方库Bass.Net调用)

一、开始

首先需要到官网下载bass.dll 主程序文件(大约 257kb): http://www.un4seen.com/

以及类库(.Net平台调用) :你可以在http://www.bass.radio42.com/bass_register.html  中使用你的邮箱即可注册到一个key 和下载Bass.Net.dll(大约520kb)

官方文档:http://www.bass.radio42.com/help/

P.S:bass.dll需要放在程序主目录下 Bass.Net.dll随意(添加到程序集引用)

二、编码

在一切开始之前,你需要先注册程序和初始化Bass解码器:

using Un4seen.Bass;//添加引用
...
BassNet.Registration("你的邮箱", "你注册到的Key");
Bass.BASS_Init(-, , BASSInit.BASS_DEVICE_CPSPEAKERS, 窗口句柄 没有的话就IntPtr.Zero);

1.实现简单的mp3播放

播放mp3有2种方式 :从文件中加载、从URL中加载

例1:从文件中加载:

//---------调用到的方法------------
public static int BASS_StreamCreateFile(
string file,//文件路径
long offset,//偏移量,一般不怎么使用
long length,//如果你使用了偏移量,定义一个偏移量之后的读取长度
BASSFlag flags//以什么方式创建流
)
//帮助链接和其他信息:http://www.bass.radio42.com/help/html/e49e005c-52c0-fc33-e5f9-f27f2e0b1c1f.htm
//---------------------------------- //创建流的id,建议作为全局变量加入(如果是播放单文件)
private int stream = -;//可以自己定义一个初始值
...
//从文件中创建一个简单的FLOAT音频流,返回流的id 便于控制和查询
stream = Bass.BASS_StreamCreateFile(你的文件完整路径, 0L, 0L, BASSFlag.BASS_SAMPLE_FLOAT);
... 开始播放
//参数:int 要播放的流的id,bool 是否在播放完成之后再重新播放
Bass.BASS_ChannelPlay(stream, false);
... 暂停播放
Bass.BASS_ChannelPause(stream);

例2:从URL中加载:

stream = Bass.BASS_StreamCreateURL(url, , BASSFlag.BASS_SAMPLE_FLOAT, null, IntPrt.Zero);

只是加载音频流的方式改变了,其他的一致

如果你需要一些其他的功能(请求Url时的标头和下载回调),请参阅以下:

1.添加URL请求标头:

很简单:在url参数里添加即可,url与每一条标头之间用"\r\n"换行,例如:

stream = Bass.BASS_StreamCreateURL(url+"\r\n"+"一条标头,Header:Content"+"\r\n"+"再一条标头",...);

2.下载回调: 多用于缓存

//必须是全局变量,否则会被GC回收!
private DOWNLOADPROC _myDownloadProc;
private FileStream _fs = null;//写入文件的流
private byte[] _data; // 本地缓存 ...
//添加调用
_myDownloadProc = new DOWNLOADPROC(DownloadCallBack);
... //下载回调,由Bass调用
private void DownloadCallBack(IntPtr buffer, int length, IntPtr user)
{
// file length
long len = Bass.BASS_StreamGetFilePosition(stream, BASSStreamFilePosition.BASS_FILEPOS_END);
// download progress
long down = Bass.BASS_StreamGetFilePosition(stream, BASSStreamFilePosition.BASS_FILEPOS_DOWNLOAD);
//可在此处添加下载进度的Callback if (_fs == null)
{
// 开始下载,打开文件流
//坑:当你切歌的时候,Bass并不会继续下载而且会向你发送已经下载完成的标识,此时你得到的文件是不完整的!所以此处先作为.cache下载
_fs = File.OpenWrite(保存的路径 + ".cache");
}
if (buffer == IntPtr.Zero)
{
// 下载完成
_fs.Flush();
_fs.Close();
_fs = null;
FileInfo fi = new FileInfo(DLPath + ".cache");
//如果下载不完整的话就删除.cache
if (fi.Length != len)
{
fi.Delete();
}
else
{
//如果下载完整的话就移动到缓存(下载)目录
fi.MoveTo(你的路径, true);
//这里可以做下载完成的回调
}
}
else
{
//接受到下载数据,实质上是Bass传来数据的指针,C#根据指针从内存中复制数据
// increase the data buffer as needed
if (_data == null || _data.Length < length)
_data = new byte[length];
// copy from managed to unmanaged memory
Marshal.Copy(buffer, _data, , length);
// write to file
_fs.Write(_data, , length);
}
}

2.获得和设置一些常规数据(播放进度、声音):

1.设置、获取音量:

坑:音量的设置是暂时性的,仅对于你输入的stream,当你再次新建音频流时(例如切歌),音量会恢复默认!你可能需要记录下设置的音量并在下一次加载流时将音量设置Set到stream中!

//设置音量 值在0~1之间 默认值为1
Bass.BASS_ChannelSetAttribute(stream, BASSAttribute.BASS_ATTRIB_VOL, value); ... //获取当前音量 ref value
float value=;
Bass.BASS_ChannelGetAttribute(stream, BASSAttribute.BASS_ATTRIB_VOL,ref value);

2.播放进度和总长(时间):

 //长度
public TimeSpan GetLength
{
get
{
double seconds = Bass.BASS_ChannelBytes2Seconds(stream, Bass.BASS_ChannelGetLength(stream));
return TimeSpan.FromSeconds(seconds);
}
}
//当前播放位置
public TimeSpan Position
{
get
{
double seconds = Bass.BASS_ChannelBytes2Seconds(stream, Bass.BASS_ChannelGetPosition(stream));
return TimeSpan.FromSeconds(seconds);
}
set => Bass.BASS_ChannelSetPosition(stream, value.TotalSeconds);
}

获取FFT数据,你可以用这个数据来做频谱:

//获取256位的FFT数据,当然你可以选择更大的,但是也足够了
//坑:暂停播放时获得的FFT数据仍是没有暂停前的数据(停留在此位置的FFT)如果你需要做频谱图,在暂停时应该手动设置为0,因为Bass并不会在暂停时返回0
float[] fft = new float[];
Bass.BASS_ChannelGetData(stream, fft, (int)BASSData.BASS_DATA_FFT256);

可惜的是,Bass并没有提供播放完成的回调,你需要设置一个Timer来判断是否播放完成,理论上当当前位置=播放总长时算播放完成...

3.更新设备和结束时

1.系统会将当前的默认设备放在集合的[0]位,但是Bass并不会自动更新设备也没有更新设备的事件回调(或是我没有找到?),所以你需要自己检查下有没有新的设备插入,你需要手动更新设备(如果需要)

public void UpdataDevice()
{
var data = Bass.BASS_GetDeviceInfos();
int index = -;
for (int i = ; i < data.Length; i++)
if (data[i].IsDefault)
{
index = i;
break;
}
if (!data[index].IsInitialized)
Bass.BASS_Init(index, , BASSInit.BASS_DEVICE_CPSPEAKERS, wind);
var a = Bass.BASS_ChannelGetDevice(stream);
if (a != index)
{
Bass.BASS_ChannelSetDevice(stream, index);
Bass.BASS_SetDevice(index);
}
}

2.结束时,需要释放流和Bass解码器:

Bass.BASS_ChannelStop(stream);
Bass.BASS_StreamFree(stream);
Bass.BASS_Stop();
Bass.BASS_Free();

4.倍速播放和其他FX效果

为什么这里需要重新起个标题呢?

这个方法推翻了之前用到的stream创建形式,但是控制播放暂停等方法不变。

首先你需要引入几个Bass的概念:

音频流:stream,用于播放音频,就是你控制播放暂停时传入的stream

解码流 decode,用于解码音频,然后传给Fx效果器,Fx效果器会返回给你一个音频流

实现此方法,你需要下载FX扩展(就在bass.dll的地方下载 选择add-on 即可),大约85kb

添加using引用:

using Un4seen.Bass.AddOn.Fx;

创建流:

//全局变量,解码流
private int decode; decode = Bass.BASS_StreamCreateFile(你的文件, 0L, 0L, BASSFlag.BASS_STREAM_DECODE);
//相对于前面的你只需要把标签换成BASS_STREAM_DECODE即可,CreateURL亦是如此 //将解码流传入Fx效果器中,你将得到一个音频流
stream = BassFx.BASS_FX_TempoCreate(decode, BASSFlag.BASS_FX_FREESOURCE );

设置Fx效果:

value值在-90~0~5000之间  以百分制计算     例如 加速/减速 播放10%那么value=10,不用倍速播放value=0 减速 value=-10

Bass.BASS_ChannelSetAttribute(stream, BASSAttribute.BASS_ATTRIB_TEMPO, value);

获取Fx效果的方法和音量一致,更换标签为BASSAttribute.BASS_ATTRIB_TEMPO即可.

更多的FX效果可以参阅:http://www.bass.radio42.com/help/html/90d034c4-b426-7f7c-4f32-28210a5e6bfb.htm 原理一样 修改标签和value值即可

完结..Thanks!

有啥问题,QQ俺:

QQ:2728578956 (Twilight./Lemon)

C# 简单地使用下 音频解码器Bass.Net的更多相关文章

  1. 最简单的基于FFmpeg的解码器-纯净版(不包含libavformat)

    ===================================================== 最简单的基于FFmpeg的视频播放器系列文章列表: 100行代码实现最简单的基于FFMPEG ...

  2. 基于modelsim-SE的简单仿真流程—下

    基于modelsim-SE的简单仿真流程—下 编译 在 WorkSpace 窗口的 counter_tst.v上点击右键,如果选择Compile selected 则编译选中的文件,Compile A ...

  3. SQL Server SQL性能优化之--数据库在“简单”参数化模式下,自动参数化SQL带来的问题

    数据库参数化的模式 数据库的参数化有两种方式,简单(simple)和强制(forced),默认的参数化默认是“简单”,简单模式下,如果每次发过来的SQL,除非完全一样,否则就重编译它(特殊情况会自动参 ...

  4. 浅析SqlServer简单参数化模式下对sql语句自动参数化处理以及执行计划重用

    我们知道,SqlServer执行sql语句的时候,有一步是对sql进行编译以生成执行计划, 在生成执行计划之前会去缓存中查找执行计划 如果执行计划缓存中有对应的执行计划缓存,那么SqlServer就会 ...

  5. 简单实用的下拉菜单(CSS+jquery)

    原文 简单实用的下拉菜单(CSS+jquery) 没什么可以说的,直接上例子 html+jquery代码 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTM ...

  6. 简单实现TCP下的大文件高效传输

    简单实现TCP下的大文件高效传输 在TCP下进行大文件传输不象小文件那样直接打包个BUFFER发送出去,因为文件比较大所以不可能把文件读到一个BUFFER发送出去.主要有些文件的大小可能是1G,2G或 ...

  7. javaCV开发详解之7:让音频转换更加简单,实现通用音频编码格式转换、重采样等音频参数的转换功能(以pcm16le编码的wav转mp3为例)

    javaCV系列文章: javacv开发详解之1:调用本机摄像头视频 javaCV开发详解之2:推流器实现,推本地摄像头视频到流媒体服务器以及摄像头录制视频功能实现(基于javaCV-FFMPEG.j ...

  8. sql server 备份与恢复系列三 简单恢复模式下的备份与还原

    一.概述 前面讲了备份的一些理论知识,这篇开始讲在简单恢复模式下的备份与还原.在简单模式下是不能做日志备份的,发生灾难后,数据库最后一次备份之后做的数据修改将是全部丢失的,所以在生产环境下,数据又很重 ...

  9. 浅谈SQL Server中的事务日志(三)----在简单恢复模式下日志的角色

    简介 在简单恢复模式下,日志文件的作用仅仅是保证了SQL Server事务的ACID属性.并不承担具体的恢复数据的角色.正如”简单”这个词的字面意思一样,数据的备份和恢复仅仅是依赖于手动备份和恢复.在 ...

随机推荐

  1. webpack4的迁移

    引用于https://www.jianshu.com/p/7ae252f27f09 感觉是突然之间,webpack4的消息就满天飞了,听说打包速度提高了很多,还有最大的噱头是实现了零配置,leader ...

  2. Mybatis的generator自动生成代码

    mybatis-generator有三种用法:命令行.ide插件.maven插件.本次使用maven生成 环境:IDEA,mysql8,maven (1):新建项目,本次以SpringBoot项目为例 ...

  3. iOS仿写有妖气漫画、视频捕获框架、启动页广告页demo、多种动画效果等源码

    iOS精选源码 以tableview的section为整体添加阴影效果/ta'b'le'vi'e'w顶部悬浮.... 一个可以轻松应用自定义过滤器的视频捕获框架. 基于UITableView的组件,旨 ...

  4. javascript 实现最简单的阶乘!

    <script type='text/javascript'>      window.onload =  get(5);   function  get(n){   document.w ...

  5. Yii框架的学习指南(策码秀才篇)1-3 我是这么学习的yii framework (不间断更新中)

    Ⅰ.基本概念一.入口文件入口文件内容:一般格式如下:<?php $yii=dirname(__FILE__).'/../../framework/yii.php';//Yii框架位置$confi ...

  6. servlet之间传递数据的方式

    Servlet传递数据方式 基本概述 Servlet传递数据的方式有很多,这里提供五种方式: 1.静态变量 2.HttpServletResponse的sendRedirect()方法 3.HttpS ...

  7. VS2010 常用的快捷键

    1.强迫智能感知:Ctrl+J:2.强迫智能感知显示参数信息:Ctrl-Shift-空格:3.格式化整个块:Ctrl+K+F4.检查括号匹配(在左右括号间切换): Ctrl +]5.选中从光标起到行首 ...

  8. django操作非ORM创建的表

    问题:django的ORM怎么连接已存在的表,然后进行增删查改操作?     工作中会遇见很多二次开发的时候,表都是已经创建好的,用django的ORM进行二次开发,怎么操作数据库中的表呢? 下面介绍 ...

  9. firefox45版本与seleniumIDE

    firefox45版本与seleniumIDE https://blog.csdn.net/seanlyly/article/details/80203896 seleniumIDE与firefox版 ...

  10. HDU-2802-F(N)

    看到这题讨论版里有说用公式的有说用循环节的,但是个人觉得这两种方法都不靠谱,比赛场上做这种题能直接推出公式需要很强数学功底,而循环节的方法如果循环节比较大就不太好发现了.这种已知通项公式的题还是用矩阵 ...