mirror of https://github.com/me50/kukemuna.git
This commit is contained in:
commit
aee5eab564
|
|
@ -0,0 +1,217 @@
|
|||
#include <cs50.h>
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
#include <math.h>
|
||||
|
||||
// Max voters and candidates
|
||||
#define MAX_VOTERS 100
|
||||
#define MAX_CANDIDATES 9
|
||||
|
||||
// preferences[i][j] is jth preference for voter i
|
||||
int preferences[MAX_VOTERS][MAX_CANDIDATES];
|
||||
|
||||
// Candidates have name, vote count, eliminated status
|
||||
typedef struct
|
||||
{
|
||||
string name;
|
||||
int votes;
|
||||
bool eliminated;
|
||||
} candidate;
|
||||
|
||||
// Array of candidates
|
||||
candidate candidates[MAX_CANDIDATES];
|
||||
|
||||
// Numbers of voters and candidates
|
||||
int voter_count;
|
||||
int candidate_count;
|
||||
|
||||
// Function prototypes
|
||||
bool vote(int voter, int rank, string name);
|
||||
void tabulate(void);
|
||||
bool print_winner(void);
|
||||
int find_min(void);
|
||||
bool is_tie(int min);
|
||||
void eliminate(int min);
|
||||
|
||||
int main(int argc, string argv[])
|
||||
{
|
||||
// Check for invalid usage
|
||||
if (argc < 2)
|
||||
{
|
||||
printf("Usage: runoff [candidate ...]\n");
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Populate array of candidates
|
||||
candidate_count = argc - 1;
|
||||
if (candidate_count > MAX_CANDIDATES)
|
||||
{
|
||||
printf("Maximum number of candidates is %i\n", MAX_CANDIDATES);
|
||||
return 2;
|
||||
}
|
||||
for (int i = 0; i < candidate_count; i++)
|
||||
{
|
||||
candidates[i].name = argv[i + 1];
|
||||
candidates[i].votes = 0;
|
||||
candidates[i].eliminated = false;
|
||||
}
|
||||
|
||||
voter_count = get_int("Number of voters: ");
|
||||
if (voter_count > MAX_VOTERS)
|
||||
{
|
||||
printf("Maximum number of voters is %i\n", MAX_VOTERS);
|
||||
return 3;
|
||||
}
|
||||
|
||||
// Keep querying for votes
|
||||
for (int i = 0; i < voter_count; i++)
|
||||
{
|
||||
|
||||
// Query for each rank
|
||||
for (int j = 0; j < candidate_count; j++)
|
||||
{
|
||||
string name = get_string("Rank %i: ", j + 1);
|
||||
|
||||
// Record vote, unless it's invalid
|
||||
if (!vote(i, j, name))
|
||||
{
|
||||
printf("Invalid vote.\n");
|
||||
return 4;
|
||||
}
|
||||
}
|
||||
|
||||
printf("\n");
|
||||
}
|
||||
|
||||
// Keep holding runoffs until winner exists
|
||||
while (true)
|
||||
{
|
||||
// Calculate votes given remaining candidates
|
||||
tabulate();
|
||||
|
||||
// Check if election has been won
|
||||
bool won = print_winner();
|
||||
if (won)
|
||||
{
|
||||
break;
|
||||
}
|
||||
|
||||
// Eliminate last-place candidates
|
||||
int min = find_min();
|
||||
bool tie = is_tie(min);
|
||||
|
||||
// If tie, everyone wins
|
||||
if (tie)
|
||||
{
|
||||
for (int i = 0; i < candidate_count; i++)
|
||||
{
|
||||
if (!candidates[i].eliminated)
|
||||
{
|
||||
printf("%s\n", candidates[i].name);
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
// Eliminate anyone with minimum number of votes
|
||||
eliminate(min);
|
||||
|
||||
// Reset vote counts back to zero
|
||||
for (int i = 0; i < candidate_count; i++)
|
||||
{
|
||||
candidates[i].votes = 0;
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
// Record preference if vote is valid
|
||||
bool vote(int voter, int rank, string name)
|
||||
{
|
||||
for (int i = 0; i < candidate_count; i++)
|
||||
{
|
||||
if (strcmp(candidates[i].name, name) == 0)
|
||||
{
|
||||
preferences[voter][rank] = i;
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// Tabulate votes for non-eliminated candidates
|
||||
void tabulate(void)
|
||||
{
|
||||
for (int i = 0; i < voter_count; i++)
|
||||
{
|
||||
for (int j = 0; j < candidate_count; j++)
|
||||
{
|
||||
if (preferences[i][0] == j && !candidates[j].eliminated)
|
||||
{
|
||||
candidates[j].votes++;
|
||||
}
|
||||
}
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
// Print the winner of the election, if there is one
|
||||
bool print_winner(void)
|
||||
{
|
||||
for (int i = 0; i < candidate_count; i++)
|
||||
{
|
||||
if (candidates[i].votes > voter_count / 2)
|
||||
{
|
||||
printf("%s\n", candidates[i].name);
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// Return the minimum number of votes any remaining candidate has
|
||||
int find_min(void)
|
||||
{
|
||||
int min = candidates[0].votes;
|
||||
|
||||
for (int i = 0; i < candidate_count; i++)
|
||||
{
|
||||
if (candidates[i].votes < min && !candidates[i].eliminated)
|
||||
{
|
||||
min = candidates[i].votes;
|
||||
}
|
||||
}
|
||||
return min;
|
||||
}
|
||||
|
||||
// Return true if the election is tied between all candidates, false otherwise
|
||||
bool is_tie(int min)
|
||||
{
|
||||
int match = 0;
|
||||
|
||||
for (int i = 0; i < candidate_count; i++)
|
||||
{
|
||||
if (candidates[i].votes == min && !candidates[i].eliminated)
|
||||
{
|
||||
match++;
|
||||
}
|
||||
}
|
||||
if (match > 1)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// Eliminate the candidate (or candidates) in last place
|
||||
void eliminate(int min)
|
||||
{
|
||||
for (int i = 0; i < candidate_count; i++)
|
||||
{
|
||||
if (candidates[i].votes == min)
|
||||
{
|
||||
candidates[i].eliminated = true;
|
||||
}
|
||||
}
|
||||
return;
|
||||
}
|
||||
Loading…
Reference in New Issue