Unity wav2Audioclip知识点记录

news2025/1/23 10:28:33

链接:hanachiru/Wav2AudioClipSample: Sample project to convert Wav files to AudioClip. (github.com)

public static class Wav
{
    /// <summary>
    /// Create AudioClip
    /// </summary>
    /// <param name="fileBytes">.wav</param>
    /// <param name="audioClipName">file name</param>
    /// <returns>AudioClip</returns>
    public static AudioClip ToAudioClip(byte[] fileBytes, string audioClipName)
    {
        using var memoryStream = new MemoryStream(fileBytes);

        // RIFF
        var riffBytes = new byte[4];
        memoryStream.Read(riffBytes);
        if (riffBytes[0] != 0x52 || riffBytes[1] != 0x49 || riffBytes[2] != 0x46 || riffBytes[3] != 0x46)
            throw new ArgumentException("fileBytes is not the correct Wav file format.");
        
        // chunk size
        var chunkSizeBytes = new byte[4];
        memoryStream.Read(chunkSizeBytes);
        var chunkSize = BitConverter.ToInt32(chunkSizeBytes);
        
        // WAVE
        var wavBytes = new byte[4];
        memoryStream.Read(wavBytes);
        if (wavBytes[0] != 0x57 || wavBytes[1] != 0x41 || wavBytes[2] != 0x56 || wavBytes[3] != 0x45)
            throw new ArgumentException("fileBytes is not the correct Wav file format.");
        
        // fmt
        var fmtBytes = new byte[4];
        memoryStream.Read(fmtBytes);
        
        if (fmtBytes[0] != 0x66 || fmtBytes[1] != 0x6d || fmtBytes[2] != 0x74 || fmtBytes[3] != 0x20)
            throw new ArgumentException("fileBytes is not the correct Wav file format.");

        // fmtSize
        var fmtSizeBytes = new byte[4];
        memoryStream.Read(fmtSizeBytes);
        var fmtSize = BitConverter.ToInt32(fmtSizeBytes);
        
        // AudioFormat
        var audioFormatBytes = new byte[2];
        memoryStream.Read(audioFormatBytes);
        var isPCM = audioFormatBytes[0] == 0x1 && audioFormatBytes[1] == 0x0;
        
        // NumChannels   Mono = 1, Stereo = 2
        var numChannelsBytes = new byte[2];
        memoryStream.Read(numChannelsBytes);
        var channels = (int)BitConverter.ToUInt16(numChannelsBytes);
        
        // SampleRate
        var sampleRateBytes = new byte[4];
        memoryStream.Read(sampleRateBytes);
        var sampleRate = BitConverter.ToInt32(sampleRateBytes);
        
        // ByteRate (=SampleRate * NumChannels * BitsPerSample/8)
        var byteRateBytes = new byte[4];
        memoryStream.Read(byteRateBytes);
        
        // BlockAlign (=NumChannels * BitsPerSample/8)
        var blockAlignBytes = new byte[2];
        memoryStream.Read(blockAlignBytes);
        
        // BitsPerSample
        var bitsPerSampleBytes = new byte[2];
        memoryStream.Read(bitsPerSampleBytes);
        var bitPerSample = BitConverter.ToUInt16(bitsPerSampleBytes);

        // Discard Extra Parameters
        if(fmtSize > 16) memoryStream.Seek(fmtSize - 16, SeekOrigin.Current);
        
        // Data
        var subChunkIDBytes = new byte[4];
        memoryStream.Read(subChunkIDBytes);

        // If fact exists, discard fact
        if (subChunkIDBytes[0] == 0x66 && subChunkIDBytes[1] == 0x61 && subChunkIDBytes[2] == 0x63 &&
            subChunkIDBytes[3] == 0x74)
        {
            var factSizeBytes = new byte[4];
            memoryStream.Read(factSizeBytes);
            var factSize = BitConverter.ToInt32(factSizeBytes);
            memoryStream.Seek(factSize , SeekOrigin.Current);
            memoryStream.Read(subChunkIDBytes);
        }
        if (subChunkIDBytes[0] != 0x64 || subChunkIDBytes[1] != 0x61 || subChunkIDBytes[2] != 0x74 || subChunkIDBytes[3] != 0x61) throw new ArgumentException("fileBytes is not the correct Wav file format.");

        // dataSize (=NumSamples * NumChannels * BitsPerSample/8)
        var dataSizeBytes = new byte[4];
        memoryStream.Read(dataSizeBytes);
        var dataSize = BitConverter.ToInt32(dataSizeBytes);

        var data = new byte[dataSize];
        memoryStream.Read(data);
        memoryStream.Dispose();
        
        return CreateAudioClip(data, channels, sampleRate, bitPerSample, audioClipName);
    }

    private static AudioClip CreateAudioClip(byte[] data, int channels, int sampleRate, ushort bitPerSample, string audioClipName)
    {
        var audioClipData = bitPerSample switch
        {
            8 => Create8BITAudioClipData(data),
            16 => Create16BITAudioClipData(data),
            32 => Create32BITAudioClipData(data),
            _ => throw new ArgumentException($"bitPerSample is not supported : bitPerSample = {bitPerSample}")
        };

        var audioClip = AudioClip.Create(audioClipName, audioClipData.Length, channels, sampleRate, false);
        audioClip.SetData(audioClipData, 0);
        return audioClip;
    }

    private static float[] Create8BITAudioClipData(byte[] data)
        => data.Select((x, i) => (float) data[i] / sbyte.MaxValue).ToArray();

    private static float[] Create16BITAudioClipData(byte[] data)
    {
        var audioClipData = new float[data.Length / 2];
        var memoryStream = new MemoryStream(data);

        for(var i = 0;;i++)
        {
            var target = new byte[2];
            var read = memoryStream.Read(target);

            if (read <= 0) break;

            audioClipData[i] = (float) BitConverter.ToInt16(target) / short.MaxValue;
        }

        return audioClipData;
    }

    private static float[] Create32BITAudioClipData(byte[] data)
    {
        var audioClipData = new float[data.Length / 4];
        var memoryStream = new MemoryStream(data);

        for(var i = 0;;i++)
        {
            var target = new byte[4];
            var read = memoryStream.Read(target);

            if (read <= 0) break;

            audioClipData[i] = (float) BitConverter.ToInt32(target) / int.MaxValue;
        }

        return audioClipData;
    }
}

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/703360.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ajax根据经纬度 获取地址

address 参数格式&#xff0c;经度在前&#xff0c;维度在后&#xff0c;以逗号隔开。 开放地址&#xff0c;有可能失效 var address 30.67,104.06; //经纬度格式一 //var address11620.12’,3912.34’;//经纬度格式二 var url "http://api.map.baidu.com/geocoder/…

什么台灯护眼效果好?注意这些选灯细节!

每个人对一个事物的接受能力跟使用体验是不一样的&#xff0c;但是护眼灯可以说是大部分人使用下来都觉得好用的产品&#xff0c;我自己大学毕业后&#xff0c;面对电脑比较多&#xff0c;我也会买个护眼台灯用&#xff0c;可以舒缓眼部疲劳&#xff0c;给孩子使用也是&#xf…

【【51单片机的DS1302模块-9】】

51单片机的DS1302模块使用 DS1302时钟显示 显示年月日 时钟秒 再可以通过按键调节时间 有种当初买的小手表的感觉 其实我们芯片内部也是可以用计时的&#xff0c;但是过分占用了芯片内部的计时模块&#xff0c;我们不如通过外部的时间模块DS1302来操作 单片机的定时器不能掉电…

Linux教程——Linux用户和用户组(包含两者之间的关系)

Linux 是多用户多任务操作系统&#xff0c;换句话说&#xff0c;Linux 系统支持多个用户在同一时间内登陆&#xff0c;不同用户可以执行不同的任务&#xff0c;并且互不影响。 例如&#xff0c;某台 Linux 服务器上有 4 个用户&#xff0c;分别是 root、www、ftp 和 mysql&…

想知道通行密钥如何取代密码吗?看完这篇文章你就懂了

密码作为一种广泛应用的身份验证方式&#xff0c;已经成为我们日常生活中不可或缺的一部分。然而&#xff0c;随着技术的不断发展和安全威胁的增加&#xff0c;传统的用户名-密码组合逐渐显示出一些弊端。幸运的是&#xff0c;通行密钥作为一种新兴的身份验证方法&#xff0c;正…

Springboot 处理BigDecimal 数据科学计数格式,展示E+8的问题

如图&#xff1a; 处理方式&#xff1a; 我们基于springboot的 JsonSerializer &#xff0c;我们自定义一下 针对BigDecimal类型属性的JsonSerializer处理。 import com.fasterxml.jackson.core.JsonGenerator; import com.fasterxml.jackson.databind.JsonSerializer; import…

vue 3.0 + vite + flv 视频流播放

官方提供的 demo 地址&#xff0c;大家可以用自己的流地址&#xff0c;先试试是否符合需求&#xff1b; http://bilibili.github.io/flv.js/demo/ Flv.js API https://gitee.com/mirrors/flv.js/blob/master/docs/api.md 安装 Flv.js npm install --save flv.js更改 tscon…

CASAIM 与ABB 达成三维数字化测量技术合作,CASAIM 国有专业尺寸检测实验室助力机器人轨道检测

近期&#xff0c;CASAIM与ABB达成三维数字化测量技术合作&#xff0c;CASAIM 国有专业尺寸检测实验室助力ABB完成机器人轨道检测&#xff0c;提高机器人的运动精度和稳定性。 ABB集团位列全球500强企业&#xff0c;是全球领先的工业机器人制造商之一&#xff0c;致力于提供高质…

ModaHub魔搭社区:向量数据库Milvus部署运维问题教程(二)

目录 在 Windows 安装 pymilvus 报错&#xff0c;如何解决&#xff1f; 内网环境&#xff0c;即离线方式&#xff0c;能否部署 Milvus 服务&#xff1f; 在多个 Milvus 节点接入 Pushgateway 的情况下如何进行区分数据来源&#xff1f; 我应该使用 SQLite 还是 MySQL 进行元…

基于群组实现从 Azure AD 到极狐GitLab 的单点登录

目录 配置单点登录 在 Azure AD 中创建企业应用 SAML 基础配置 配置 Azure “Attributes & Claims” 配置用户同步 在极狐GitLab 创建 SCIM Token 配置 Azure Provisioning Azure 手动用户预配 测试单点登录 Azure 自动用户同步 配置群组同步 配置 SAML 群组链…

【算法基础】快速排序(模板)

&#x1f466;个人主页&#xff1a;Weraphael ✍&#x1f3fb;作者简介&#xff1a;目前正在学习c和算法 ✈️专栏&#xff1a;【C/C】算法 &#x1f40b; 希望大家多多支持&#xff0c;咱一起进步&#xff01;&#x1f601; 如果文章有啥瑕疵 希望大佬指点一二 如果文章对你有…

力扣题库刷题笔记17--电话号码的字母组合

1、题目如下&#xff1a; 2、个人Python代码实现&#xff1a; 还是先记录一下思路&#xff0c;首先这种类型的题&#xff0c;需要自定义一个字典对应题目中的电话号码和数字。其次&#xff0c;个人的思路是&#xff0c;先读取字符串第一个字符&#xff08;digits[0]&#xff09…

云原生(第二篇)k8s-二进制搭建

准备五台机器&#xff1a; master01&#xff1a;192.168.169.10 node01&#xff1a;192.168.169.40 node02&#xff1a;192.168.169.50 master02&#xff1a;192.168.169.60 负载均衡nginxkeepalive01&#xff08;master&#xff09;&#xff1a;192.168.169.20 负载均衡…

记一次Native memory leak排查过程 | 京东云技术团队

1 问题现象 路由计算服务是路由系统的核心服务&#xff0c;负责运单路由计划的计算以及实操与计划的匹配。在运维过程中&#xff0c;发现在长期不重启的情况下&#xff0c;有TP99缓慢爬坡的现象。此外&#xff0c;在每周例行调度的试算过程中&#xff0c;能明显看到内存的上涨…

Windows开启telnect

1、Telnet是什么&#xff1f; Telnet 是一种网络协议&#xff0c;用于通过网络远程登录到远程计算机或设备上。它允许用户在本地计算机上使用命令行界面&#xff08;命令提示符&#xff09;与远程主机进行交互&#xff0c;就像直接在远程主机上操作一样。Telnet 协议使用 TCP/I…

【需求实现】Tensorflow2的曲线拟合(三):Embedding层

文章目录 导读Embedding的维度问题Embedding的输入输出比较容易踩的坑input_shape与input_length的对应关系built属性 导读 这是填曲线拟合第一篇的坑&#xff0c;有关Embedding层的问题。 Embedding的维度问题 首先是上次我们提到的Embedding层&#xff0c;他确实能够做到将…

预约Oracle OCP认证考试的保姆式流程

Oracle OCP认证考试的预约流程涉及到Oracle的SLS培训记录&#xff0c;因此相当复杂。本文进行了详细地说明&#xff0c;每一步都有截图&#xff0c;有需要的同学建议收藏。 关于号主&#xff0c;姚远 Oracle ACE&#xff08;Oracle和MySQL数据库方向&#xff09;。Oracle MAA…

智能体重秤方案PCBA方案设计

智能体重秤是一款高精度、便捷、多功能的健康管理工具&#xff0c;旨在帮助用户监测和控制体重&#xff0c;达到健康管理与减肥的目的。该产品融合了先进的科技技术&#xff0c;结合了人体工程学设计&#xff0c;具有美观、易用的特点。以下将从结构、参数、原理和应用方面为大…

电涌(浪涌)保护器防雷保护级别

浪涌保护器实际就是压敏电阻&#xff0c;具有高通低阻的特性。当电网在不超过最大持续运行电压的情况下运行时&#xff0c;两个电极之间呈高阻状态。由于雷击的能量是非常巨大的&#xff0c;需要通过分级泄放的方法&#xff0c;将雷击能量逐步泄放到大地。 第一级防雷器可以对…