【发布时间】:2011-09-27 21:03:23
【问题描述】:
我正在尝试使用一种非常基本的程序方法来创建一个字谜求解器。我发现我可能应该使用课程来完成这项工作,但现在为时已晚,我的作业即将到期。任何关于如何解决这个问题的建议都会很棒!
基本上,这就是算法应该做的:
- 获取字典中的所有单词;将它们存放在容器中
- 从用户那里得到一个字;适当时退出
- 获取用户输入的单词的所有排列
- 从排列中去除用户输入的单词
- 去除排列集合中不在我在第 1 部分收集的字典中的所有单词
现在对于最后一步,我必须确保不显示重复的字谜(即包含相同字母的字谜,例如“循环”)。我似乎无法让这个检查工作,这在下面的 TODO 注释块下进行了说明。
任何建议都会很棒!
#include <iostream>
#include <fstream>
#include <string>
//
// Change size below to accomodate more anagrams and dictionary words
//
#define MAX_ANGM_SIZE 4096
#define MAX_WORD_SIZE 1048576
using namespace std;
//
// Determines whether anagram is valid or not; will not display word
// which user entered or words not contained in dictionary
//
bool isValidAnagram(string word, string userWord,
string dictionary[], unsigned int listIdx)
{
for(unsigned int idx = 0; idx < listIdx; ++idx)
{
if(word == userWord)
return false;
else if (word == dictionary[idx])
return true;
}
return false;
}
//
// Determines whether user's word is contained in the dictionary
// or not
//
bool isValidWord(string word, string dictionary[],
unsigned int listIdx)
{
for(unsigned int idx = 0; idx < listIdx; ++idx)
{
if(word == dictionary[idx])
return true;
}
return false;
}
//
// TODO:This function should test for duplicate anagrams and return
// true if duplicates are found.
//
bool isRepeated(string anagrams[], unsigned int anaIdx)
{
for(unsigned int idx = anaIdx; idx != 0; --idx)
{
if(anagrams[idx] == anagrams[anaIdx])
return true;
else
return false;
}
return false;
}
//
// Only display elements in array which aren't blank and don't
// display duplicate anagrams; notify user if no anagrams
// were found.
//
void displayAnagrams(string anagrams[], unsigned int next)
{
int flag = 0;
for (unsigned int idx = 0; idx < next; ++idx)
{
if((anagrams[idx] != "") || (!(isRepeated(anagrams, idx))))
{
if(idx == 1)
cout << " Anagrams: ";
if(idx > 0)
flag = 1;
cout << anagrams[idx] << " ";
}
else
continue;
}
if(flag == 0)
cout << " no anagrams found" << endl;
}
static void swap(char &c1, char &c2)
{
char temp = c1;
c1 = c2;
c2 = temp;
}
//
// Pass in word to be altered, the userWord for comparison, the array to store
// anagrams, the dictionary for comparison, the count for the number of anagrams
// and the count for number of dictionary words
//
static void permute(string word, string userWord, int k, string anagrams[],
string dictionary[], unsigned int &next, unsigned int listIdx)
{
if(k == word.length()-1)
{
if(isValidAnagram(word, userWord, dictionary, listIdx))
anagrams[next] = word;
++next;
}
else
{
for(int idx = k; idx < word.length(); ++idx)
{
swap(word[k], word[idx]);
permute(word, userWord, k+1, anagrams, dictionary, next, listIdx);
}
}
}
//
// Create container to store anagrams, validate user's word in dictionary, get all
// of the anagrams, then display all valid anagrams
//
void getAnagrams(string word, string dictionary[], unsigned int listIdx)
{
string anagrams[MAX_ANGM_SIZE];
unsigned int next = 0;
if(isValidWord(word, dictionary, listIdx))
{
permute(word, word, 0, anagrams, dictionary, next, listIdx);
}
else
{
cerr << " \"" << word << "\"" << " is not a valid word" << endl;
return;
}
displayAnagrams(anagrams, next);
}
//
// Read in dictionary file, store contents of file in a list, prompt
// the user to type in words to generate anagrams
//
int main()
{
string file;
string word;
string quit = "quit";
string dictionary[MAX_WORD_SIZE];
unsigned int idx = 0;
cout << "Enter a dictionary file: ";
cin >> file;
cout << "Reading file \"" << file << "\"" << endl;
cout << endl;
ifstream inFile(file.c_str());
if(!(inFile.is_open()))
{
cerr << "Can't open file \"" << file << "\""
<< endl;
exit(EXIT_FAILURE);
}
while(!inFile.eof())
{
inFile >> dictionary[idx];
++idx;
}
inFile.close();
while(true)
{
cout << "Enter a word: ";
cin >> word;
if(word == quit) break;
getAnagrams(word, dictionary, idx);
cout << endl;
}
return 0;
}
【问题讨论】:
-
为什么要避免使用 STL? (毕竟您使用的是
string,它符合STL 容器的大部分要求……) -
这其实是给STL上的一个类的,但这是第一次作业,我还没用过STL。此作业的目标是使用标准技术,然后在我们知道如何使用 STL 后在学期末做完全相同的作业。
-
啊,我明白了。好吧,只是一个 FYI——
std::string公开了一个主要是 STL 序列容器接口。 (但是,如果您想学究气,字符串库会早于 STL 包含在标准中一段时间) -
如果是作业,请加tag homework
-
@Dylan:很遗憾听到这是你们学校的学习方法(顺便说一句,我的学习方法是一样的)。你可能想向你的导师建议,反过来可能会更好:学习如何使用高级构造来解决问题,并在你这样做的同时学习语言和解决问题的基础知识。然后学习更难的低级细节,重新实现您已经手动完成的内容,目标不是解决问题(此时会知道算法),而是解决手动实现的细节。
标签: c++ string fstream anagram