我正在使用非常基本的声音合成在我的游戏中即时创建音频和效果。基本上,我有一些方法可以在给定频率、振幅和持续时间的情况下播放声音。
对于简短的短语和旋律,我想提出一个基本的符号,以便我可以轻松地重写或将新的旋律添加到代码中(最后也许我可以从文件中读取,但这可能有点过分了) .
但是我不确定如何实现它。
我首先创建了一个 EqualTemperamentTuning 枚举,其中包含所有 88 个基本钢琴音符以及一个 MIDI # 字段和一个频率字段。这至少意味着我可以处理音符名称而不是频率。
public enum EqualTemperamentTuning {
A_0 (1, 27.5),
A_SHARP_0 (2, 29.1352),
...
C_8 (88, 4186.01);
private int num;
private double frequency;
EqualTemperamentTuning(int num, double frequency){
this.num = num;
this.frequency = frequency;
}
public double getFrequency(){
return frequency;
}
public double getNum(){
return num;
}
}
然后我开始创建更多对象,首先是一个包含 EqualTemperamentTuning、振幅和长度的音符:
public class Note {
/** Note frequency */
private double frequency;
/** Note Amplitude */
private double amplitude;
/** Note length */
private int length;
public Note(EqualTemperamentTuning tuning, double amplitude, int length){
this.frequency = tuning.getFrequency();
this.amplitude = amplitude;
this.length = length;
}
public double getFrequency(){
return frequency;
}
public double getAmplitude(){
return amplitude;
}
public int getLength(){
return length;
}
}
最后为了定义我想要播放的旋律,我创建了一个 NotePhrase 类:
public class NotePhrase {
/** The arrayList of notes*/
private Note[] notes;
public NotePhrase(Note[] notes){
this.notes = notes;
}
public double getFrequency(int counter){
// for each note in the array
for (int i = 0; i< notes.length; i++){
// for each unit of length per note
for (int j=0; j<notes[i].getLength(); j++){
counter--;
// return the frequency at this point
if (counter <= 0) return notes[i].getFrequency();
}
}
return -1;
}
}
然后在我的音频生成类中,我有一个循环(带计数器)从波形发生器生成样本。每次我需要播放新样本时,它都会根据上面的 NotePhrase.getFrequency(int counter) 方法设置波形的频率。这应该(我还没有真正测试过!)只是根据频率和幅度(待添加)播放NotePhrase旋律。
问题是,它看起来不是很优雅,更具体地说,很难以任何一种清晰的方式“写”一段旋律。我必须编写一大堆新的 Note 对象,然后用它们的数组构造一个 NotePhrase 对象……我不太清楚我将如何硬编码这些旋律,然后稍后在它们之间轻松切换。
我真的很想创建一个旋律枚举或类似的东西,在这里我可以轻松地为每个不同的旋律硬编码一个人类可读的配置,然后当我想使用它们时,只需将枚举类型传递给音频播放器...
我得到的最好的是:
private static enum Melody {
NOKIA_RINGTONE ( new Note(EqualTemperamentTuning.E_5, 0.5, 1), new Note(EqualTemperamentTuning.D_5, 0.5, 1))
;
private Note[] notes = new Note[2];
Melody (Note note1, Note note2){
this.notes[0] = note1;
this.notes[1] = note2;
}
}
然后我会在运行时将其加载到 NotePhrase 对象中。这不是很好,因为我必须为具有不同音符数量的旋律创建一个新的构造函数。如果我反过来做,用一组音符构造枚举,那么我只是在别处和两部分中“写”旋律,这看起来更令人困惑......
所以我对如何正确构建它感到困惑?即要创建什么类以及它们应该包含什么信息......我想得到这个“正确”因为,我可能想在未来扩展符号以包括效果(回声等)并且我已经找到了我的非常很少有人知道正确的类、结构和关系(甚至名称)可以使我的程序非常容易或难以理解。
很抱歉这篇文章,这可能不是一个措辞很好(咳咳)的问题,但作为 Java 和 OOP 的初学者,任何提示都将非常受欢迎!
编辑**
感谢您的回答和建议,非常有帮助。考虑在这种情况下给出的答案促使我重新考虑我的一般音频实现,它现在非常不稳定。不过不确定我应该将谁标记为正确,因为我真的只是想采纳所有建议并尝试从那里开始。
最佳答案
不要对旋律使用枚举,因为旋律不代表真正的常量,而是音符数据的集合。我建议不要使用数组,而是使用 ArrayList<Note>哪个更灵活。
在 Melody 中,我会将音符与小节和节拍相关联,并且会给 Melody 类一个 addNote(Note, int measure, int beatFraction),删除音符也是如此。
考虑将您的 12 个核心笔记作为枚举,然后使用它们加上一个 Octave 整数来创建一个笔记对象。
...有很多方法可以玩这个练习。 “关键”是不断尝试。
关于java - 如何在 java 中创建一个简单但结构良好的乐谱类(乐谱)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10723832/
出于纯粹的兴趣,我很好奇如何按顺序创建PI,而不是在过程结果之后生成数字,而是让数字在过程本身生成时显示。如果是这种情况,那么数字可以自行产生,我可以对以前看到的数字实现垃圾收集,从而创建一个无限系列。结果只是在Pi系列之后每秒生成一个数字。这是我通过互联网筛选的结果:这是流行的计算机友好算法,类机器算法:defarccot(x,unity)xpow=unity/xn=1sign=1sum=0loopdoterm=xpow/nbreakifterm==0sum+=sign*(xpow/n)xpow/=x*xn+=2sign=-signendsumenddefcalc_pi(digits
我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h
如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby
我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%
exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby中使用两个参数异步运行exe吗?我已经尝试过ruby命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何rubygems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除
鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende
我正在为一个项目制作一个简单的shell,我希望像在Bash中一样解析参数字符串。foobar"helloworld"fooz应该变成:["foo","bar","helloworld","fooz"]等等。到目前为止,我一直在使用CSV::parse_line,将列分隔符设置为""和.compact输出。问题是我现在必须选择是要支持单引号还是双引号。CSV不支持超过一个分隔符。Python有一个名为shlex的模块:>>>shlex.split("Test'helloworld'foo")['Test','helloworld','foo']>>>shlex.split('Test"
我实际上是在尝试使用RVM在我的OSX10.7.5上更新ruby,并在输入以下命令后:rvminstallruby我得到了以下回复:Searchingforbinaryrubies,thismighttakesometime.Checkingrequirementsforosx.Installingrequirementsforosx.Updatingsystem.......Errorrunning'requirements_osx_brew_update_systemruby-2.0.0-p247',pleaseread/Users/username/.rvm/log/138121
我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/
这可能是个愚蠢的问题。但是,我是一个新手......你怎么能在交互式rubyshell中有多行代码?好像你只能有一条长线。按回车键运行代码。无论如何我可以在不运行代码的情况下跳到下一行吗?再次抱歉,如果这是一个愚蠢的问题。谢谢。 最佳答案 这是一个例子:2.1.2:053>a=1=>12.1.2:054>b=2=>22.1.2:055>a+b=>32.1.2:056>ifa>b#Thecode‘if..."startsthedefinitionoftheconditionalstatement.2.1.2:057?>puts"f