欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 新闻 > 资讯 > 用 F# 描述音乐领域

用 F# 描述音乐领域

2026/1/7 15:58:35 来源:https://blog.csdn.net/xiefeng240601/article/details/141370364  浏览:    关键词:用 F# 描述音乐领域

我最近的一个项目是创建一个软件,该软件可以根据一组预定义的规则自动生成音乐。我计划引入的随机性程度可以让我每次都能创作出不同的旋律,而我计划创建的规则集可以确保它听起来仍然很好听。您可以在此处访问完整的源代码。下面我们将更深入地了解它的功能细节。

一.域描述

我计划扩展的规则是功能和声的概念。这个概念基于每个和弦都有其自身功能的想法。给定和弦的功能取决于和弦“想要”接下来去哪里,因为和声进行有两个维度:和弦的音高及其相互作用方式(音程层次);以及它在整个和声环境中的功能。因此,功能和声经历了产生和释放张力的循环,因此,我们有稳定和不稳定的时刻,它们的强度各不相同。

最重要的三个功能是,

  • 主音:可以是或感觉非常稳定,通常是一段乐曲或一段乐段的最后一个和弦
  • 次属音:准备和声节奏并引入一定程度的不稳定性
  • 主导音:是最不稳定的和弦,想要解决另一个和弦

二.对域名进行编码

现在让我们将这些知识编码成代码。我选择 F# 来完成这项任务,因为它的类型系统对于表达各种领域来说非常方便。

让我们从基础开始,描述一下我们的调色板里有哪些和弦。

type ChordQuality =| Major| Minor

还有更多的和弦品质,但这已经足够满足我们的需要了。

现在,我们来描述一下从上一段获得的知识。

type HarmonyItem =| Tonic| SubDominant| Dominant

它们之间的转换如下所示。

type HarmonyTransition =| Dublicate| IncreaseTension| MaximizeTension| DecreaseTension| Resolve

现在让我们看看如何应用过渡。

let applyCommand command chord =match command with| Dublicate -> dublicate chord| IncreaseTension -> increaseTension chord| DecreaseTension -> decreaseTension chord| MaximizeTension -> maximizeTension chord| Resolve -> resolve chordlet dublicate harmonyItem =harmonyItemlet increaseTension harmonyItem =match harmonyItem with| Tonic -> SubDominant| SubDominant -> Dominant| Dominant -> Dominantlet decreaseTension harmonyItem =match harmonyItem with| Tonic -> Tonic| SubDominant -> Tonic| Dominant -> SubDominantlet maximizeTension harmonyItem =Dominantlet resolve harmonyItem =Tonic

话虽如此,让我们看看功能进程中每个项目的背后隐藏着什么。基本上,每个和弦都会有一个音质,并且音符与根音之间存在偏移。

type HarmonyItemValue = {value: intchordQuality: ChordQuality
}let getHarmonyItemValue item =match item with| Tonic -> { value = 0; chordQuality = Major }| SubDominant -> { value = 5; chordQuality = Major }| Dominant -> { value = 7; chordQuality = Major }

鉴于此,我们可以从每个和声项目中创建一个音调数组。

type Pitch = {midiNote: intduration: float
}let createChordFromRootNote rootNote item =let itemValue = getHarmonyItemValue itemmatch (itemValue.value, itemValue.chordQuality) with| (value, Major) -> [|{midiNote = rootNote + valueduration = 1.0};{midiNote = rootNote + value + 4duration = 0.125};{midiNote = rootNote + value + 7duration = 1.0}|]| (value, Minor) -> [|{midiNote = rootNote + valueduration = 1.0};{midiNote = rootNote + value + 4duration = 0.125};{midiNote = rootNote + value + 7duration = 1.0}|]

三.产生进展

因此,为了每次都能创建不同的进程,我们需要在过程中添加一些随机性。为了实现这一点,我们将某种程度与每次转换的概率相关联。假设我们处于主音和弦中,我们有 0.1 的概率会停留在那里进行下一个和弦,而增加张力的概率彼此相等,总计为 0.45。在这种情况下,让我们为每个转换分配一个阈值。假设主音为 0.1,次属音为 0.55,这是 0.1 + 主音概率的主音阈值,属音为 1.0,这是一组完整事件的概率。在这种情况下,一旦我们生成一个介于 0.0 和 1.0 之间的随机数,我们就可以选择阈值大于给定随机数的最小项。

其代码如下所示。

type HarmonyTransitionProbability = {transition: HarmonyTransitioncoinThreshold: float
}let regenerateHarmonyTransitionProbability currentHarmonyItem =match currentHarmonyItem with| Tonic ->[|{ transition = Dublicate; coinThreshold = 0.1 };{ transition = IncreaseTension; coinThreshold = 0.55 };{ transition = MaximizeTension; coinThreshold = 1.0 };|]| SubDominant ->[|{ transition = Dublicate; coinThreshold = 0.1 };{ transition = IncreaseTension; coinThreshold = 0.55 };{ transition = Resolve; coinThreshold = 1.0 };|]| Dominant ->[|{ transition = Dublicate; coinThreshold = 0.1 };{ transition = Resolve; coinThreshold = 0.9 };{ transition = DecreaseTension; coinThreshold = 1.0 };|]let rnd = Random()let generateNextChord currentChord coin =let probabilityMap = regenerateHarmonyTransitionProbability currentChordlet command = (Array.filter (fun x -> coin <= x.coinThreshold) probabilityMap).[0].transitionapplyCommand command currentChordlet generateProgression (initialChord: HarmonyItem) (length: int) : HarmonyItem array =let rec generate (currentChord: HarmonyItem) (remaining: int) (progression: HarmonyItem list) =if remaining = 0 thenList.toArray (List.rev progression)elselet coin = rnd.NextDouble()Console.WriteLine(coin)let nextChord = generateNextChord currentChord coingenerate nextChord (remaining - 1) (nextChord :: progression)generate initialChord (length - 1) [initialChord]

四.领域演化

到目前为止,我们只介绍了一些基本概念。但即使是一些更主流的进程,例如Axis of Awesome 4 chord wamp也基于替代的概念。替代是我们已经知道的谐波函数的副本,但不如其对应项那么明显。所以让我们也在我们的领域中介绍它们。

对我来说,这是用 F# 表达我的领域最有趣的部分,因为我必须记住在两个地方添加它:和谐项目以及它们之间的过渡。

type HarmonyItem =| Tonic| TonicSubstitute1| TonicSubstitute2| SubDominant| Dominanttype HarmonyTransition =| Dublicate| IncreaseTension| MaximizeTension| DecreaseTension| DecreaseTensionToFisrtSubstitute| DecreaseTensionToSecondSubstitute| Resolve| ResolveToFirstSubstitute| ResolveToSecondSubstitute

此时,在我应用模式匹配的任何地方,编译器都会向我发出有关模式匹配不完整的警告。因此,我只需添加缺失的案例,直到编译器满意为止,然后瞧:域的新版本就完成了。在某种程度上,这让我想起了经典之作“有效处理遗留代码”中的精益编译器技术。

五.产生声音

此时,我们可以生成 MIDI 音符的音高数组。为了从这些音符中创建声音,我使用了一种名为SuperCollider的专门编程语言。我不会在这里深入讨论细节,但如果您有兴趣,可以看看代码。注意,那里有很多分支,所有分支都包含一些有趣的代码。

六.结论

我已经支持 F# 很长时间了。因此,我不会再一次阐述其类型系统的强大功能,而是在这里留下一个我最喜欢的曲目的链接,该曲目是用本文中的代码创建的。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词