cs50 - CS50 复数 - 无法理解为什么使用 Candidate_count
问题描述
我试图理解为什么在 CS50 的复数(第 3 周)中使用候选人计数而不是选民计数。下面是我的代码。
如果我们假设我们有 3 个候选人(Alice、Bob、Charlie),并且每次我们遍历 bool 函数或 print_winner 函数时,如果我们有 10 个选民,我们会不会错过计票?根据我的理解,“我”只会迭代 3 次。我在理解为什么我们不会在底部的打印获胜者函数中使用 voter_count 时遇到一个概念问题。
我仍在尝试改进我的代码,所以部分可能仍然存在错误。我只是在寻找一些帮助来澄清这个问题的逻辑。
#include <cs50.h>
#include <stdio.h>
#include <string.h>
// Max number of candidates
#define MAX 9
// Candidates have name and vote count
typedef struct
{
string name;
int votes;
}
candidate;
// Array of candidates
candidate candidates[MAX];
// Number of candidates
int candidate_count;
// Function prototypes
bool vote(string name);
void print_winner(void);
int main(int argc, string argv[])
{
// Check for invalid usage
if (argc < 2)
{
printf("Usage: plurality [candidate ...]\n");
return 1;
}
// Populate array of candidates (number of arguments - 1 because the first arg is going to be plurality)
candidate_count = argc - 1;
if (candidate_count > MAX)
{
printf("Maximum number of candidates is %i\n", MAX);
return 2;
}
for (int i = 0; i < candidate_count; i++)
{
candidates[i].name = argv[i + 1]; //
candidates[i].votes = 0;
}
int voter_count = get_int("Number of voters: ");
// Loop over all voters
for (int i = 0; i < voter_count; i++)
{
string name = get_string("Vote: ");
// Check for invalid vote
if (!vote(name))
{
printf("Invalid vote.\n");
}
}
// Display winner of election
print_winner();
}
// Update vote totals given a new vote
bool vote(string name)
{
for (int i = 0; i < candidate_count; i++)
{
if (strcmp(candidates[i].name, name) == 0)
{
candidates[i].votes++;
return true;
}
}
return false;
}
// Print the winner (or winners) of the election
void print_winner(void)
{
int maxvotes = 0;
for (int i = 0; i < candidate_count; i++)
{
if (candidates[i].votes > maxvotes)
{
maxvotes = candidates[i].votes;
}
}
for (int i = 0; i < candidate_count; i++)
{
printf("the winner is %s\n!", candidates[i].name);
}
return;
}
解决方案
我认为您误解了我们正在循环的内容。candidate_count
用于循环遍历具有元素数量candidates
的数组。candidate_count
考虑这个简单的数组 -
int arr[] = {1, 2, 3, 4, 5};
我们将循环使用-
for (int i = 0; i < 5; i++)
{
printf("value at index %d is %d\n", i, arr[i]);
}
注意部分i < 5
。为什么是5?好吧,因为这是数组的长度arr
,再多一点,我们就会越界读取,少一点,我们就不会读取整个arr
.
现在替换arr
为candidates
,它也是一个数组,替换5
为candidate_count
,这显然是所述数组的长度。
vote
or函数中的循环print_winner
不会“计算选票”,它们会遍历候选人数组。这就是这些循环的目的。因此,要遍历数组,我们必须使用index < length_of_array
. 这正是它的作用。
只是为了解决“那么它在哪里计算选票?”的担忧。让我们来看看vote
真正的快速-
bool vote(string name)
{
for (int i = 0; i < candidate_count; i++)
{
if (strcmp(candidates[i].name, name) == 0)
{
candidates[i].votes++;
return true;
}
}
return false;
}
这个函数在用户输入后触发,假设用户被要求输入一个名字——他们选择了一些名字,假设这个名字与候选人匹配。所谓vote
的,它遍历候选数组。为什么要迭代?它需要找到与用户刚刚提供的姓名同名的候选人。
所以这个循环的目的,是遍历数组,而这个循环candidates
中的代码的目的是比较每个候选者的名字和用户提供的名字。如果匹配,它会增加该候选人的投票。
这条线——candidates[i].votes++;
这就是votes
计算的地方。就像增加一个计数器一样简单。
想象一个真实的场景,你就是程序。候选人都站成一排(一个阵列)。每个候选人以 0 张牌(票)开始。假设它们都没有相同的名称。
一个用户告诉你他们想投票给候选人 foo(只是一个名字)。
然后你去那一行候选人。你从第一个候选人开始,问他们“你叫什么名字?”。
候选人告诉你他们的名字。
如果他们的名字与用户给你的名字匹配,就是这样,你给他们一张代表投票的卡片。现在他们比以前多了 1 张卡
如果他们的名字不匹配,您将移动到下一个候选人并重复。
但是你在哪里停下来?很简单,你在最后一个候选人之前停下来。这最多需要多少次迭代?
candidate_count
,即站在那条线上的候选人人数。因此,最坏的情况是,您要找的人排在队伍的最后,所以您必须先询问candidate_count
人数,然后才能最终找到您要找的人。遍历数组。最后,你只需要计算每个候选人需要多少张牌才能知道谁是赢家。(
.votes
)
该print_winner
函数应该做同样的事情。虽然你的代码看起来有点歪。
for (int i = 0; i < candidate_count; i++)
{
if (candidates[i].votes > maxvotes)
{
maxvotes = candidates[i].votes;
}
}
这个循环相当于遍历整个候选人行,询问每个人他们有多少张卡片(选票)并找出谁拥有最多的卡片(选票)。
但是你还需要问那个人的名字maxvotes
!这就是你要找的。
但是你完全抛弃了maxvotes
你刚刚计算的,你永远不会使用它。
for (int i = 0; i < candidate_count; i++)
{
printf("the winner is %s\n!", candidates[i].name);
}
该循环将打印每个candidate
. 这相当于遍历所有候选人(数组),问每个人“你叫什么名字?” 并大喊“获胜者是[他们的名字]!”。但这不是真的!
您应该存储候选人的姓名,maxvotes
然后在第一个循环之后打印它 -
string winner_name;
...
for (int i = 0; i < candidate_count; i++)
{
if (candidates[i].votes > maxvotes)
{
maxvotes = candidates[i].votes;
winner_name = candidates[i].name;
}
}
printf("the winner is %s\n!", candidates[i].name);
当然,这不考虑多个获胜者,如果 2 个候选人的票数相同怎么办?那是你必须自己尝试的东西。
但我希望这能回答你关于混乱的问题。
推荐阅读
- java - 如果我们用 AES 密钥包装 256 位 AES 密钥,那么包装密钥的大小可以超过 32 个字节?
- xaml - 在这种情况下,哪种布局最合适?Xamarin 形式
- unity3d - Unity WebGL 资产包内存未释放
- java - 文件系统的Java和Junit集成测试
- angular - SheetJs自动转换日期角度7
- angular - 路由器导航 SlideIn 动画不起作用
- curl - 尽管 curl 代码中有单引号,但仍会出错
- php - 如果它遵循版本号格式,我如何正确检查字符串格式?
- mysql - mysql 加入条件行为
- java - Picasso使用mysql url在arraylist hashmap中加载图像