I have a vector of objects passed into a function but it suddenly changes when i try to use them. Once the vector is passed in I use a print statement to see the contents and its all correct. But once I start to copy the elements to another vector I'm getting the wrong output.
void passCards(int numPlayers, std::vector<CardTemplate> deckOfCards, Player p1, Player p2, Player p3, Player p4)
{
//Distribute cards to number of Players
switch(numPlayers)
{
case 2:
//Test to check whats in deck
for(int i = 0; i < deckOfCards.size(); i++)
{
std::cout << deckOfCards[i].suit << " : " << deckOfCards[i].cardNum << std::endl;
}
std::cout<<std::endl;
std::cout<<std::endl;
std::cout<<std::endl;
for(int i = 0; i < deckOfCards.size() - 1; i++)
{
p1.playerHand.push_back(deckOfCards[i]);
//LOOK HERE
std::cout << p1.playerHand[i].suit << " : " << p1.playerHand[i].cardNum << std::endl;
i++;
p2.playerHand.push_back(deckOfCards[i]);
//LOOK HERE TOO
std::cout << "2----" << p2.playerHand[i].suit << " : " << p2.playerHand[i].cardNum << std::endl;
i++;
//Will not be used just to keep cards randomized for 2 players (Keep to let users select deck?)
//playerThree.push_back(deckOfCards[i]);
}
break;
I'm passing cards to different players from a deck of cards. I increment the i so the next player gets the next card in the deck.
Do you know what else could possibly cause this? I was also getting weird output with random code snippets along with numbers being printed out when i passed the players in my reference. (im guessing maybe memory addresses or something)
I'm skipping 1 card each turn. Its a 52 card deck and I'm giving each player 17 cards. Essentially the skipping should represent a third deck being passed to. The function incrementation was working fine for what i needed to do but that still doesn't explain why the vector elements are changing
Fair enough.
Just a thought - you're passing Player objects in by value.
void passCards(int numPlayers, std::vector<CardTemplate> deckOfCards, Player p1, Player p2, Player p3, Player p4)
Those copies will expire when the function returns.
This is a very uncommon design. Would it fix some things if you passed by reference? I.e.: void passCards(int numPlayers, std::vector<CardTemplate> deckOfCards, Player& p1, Player& p2, Player& p3, Player& p4)
The first bit in bold. Assuming that we're in the first iteration of the loop and that p2.playerHand is an empty vector. You push_back card #1.
Now the second bit in bold. You access element #1 of p2.playerHand to print out the suit. But there won't be anything in slot #1 yet - what you mean to do is access element #0 but your iterator's now out of sync with the for loop (because you manipulate it inside the for loop).
Next time round you'll be even further off - there'll be 2 elements in each playerHand but you'll be trying to access the values for elements #3 and #4. Etc.
You'd need a separate counter to keep track of how many cards you've dealt so you can access the right element.
Hope that helps.
Your previous post solved my bug. I was just wondering what the difference when passing the func(object) vs func(&object)? Don't they both use a pointer to the object?
#include <iostream>
class A
{
public:
A(int x) : myx(x) {}
int getmyx() { return myx; }
void setmyx(int x) { myx = x; }
private:
int myx;
}; // class A
// Parameter is a copy of object. There'll be no change in calling
// environment.
void bar(A theA)
{
theA.setmyx(17);
} // bar
// Parameter is a reference - changes will affect calling environment.
void foo(A& theA)
{
theA.setmyx(17);
} // foo
int main()
{
A someA(4);
std::cout << someA.getmyx() << "\n";
someA.setmyx(8);
std::cout << someA.getmyx() << "\n";
std::cout << "call bar()\n";
bar(someA);
std::cout << someA.getmyx() << "\n";
std::cout << "call foo()\n";
foo(someA);
std::cout << someA.getmyx() << "\n";
return 0;
} // main
If you want to make sure a function can't alter an object that's passed in by reference (and document to an end user that this won't happen), ask your compiler to enforce this by making that parameter const - as an example, try changing the function signature of foo() to: void foo(const A& theA). Your compiler should reject it.
In more idiomatic C++, your function header would maybe look like this (- note I make the vector a const ref): void passCards(int numPlayers, const std::vector<CardTemplate>& deckOfCards, Player& p1, Player& p2, Player& p3, Player& p4)
Thanks for your explanation! I mustve gotten confused with what my professor said. I sometimes get confused when i should use & lol but thanks for helping clarify!