19.05.2012, 08:18
You're right by your assumption, field names can be used and this might be a simpler approach, however many ways are possible.
One: knowing what you query for and thereby knowing field indexes.
Adding a new field that you query for does not require you to modify hundreds of indexes in your code, what you can do is use a query which specifies all the field names that you are loading. For example:
Adding a new field to query for is simple, the new index will be 5 in a case where you append the new field name...
Field `health` has an index of 5, this means that in your callback, doing:
This code above assumes you have only one result and the query also uses the limit clause to make sure of this. So you're getting the 6th field of 1st row that the query returned and then storing it in a variable called health.
So what's the advantage to this behavior? You can easily do on-the-run development and modifications to your database!
Lets say you have such query:
SELECT * FROM players WHERE name = 'MP2' LIMIT 0,1
What does this mean? All fields in the table are queried for and will be returned in whatever callback you choose to use. If you are developing a new feature for your server (and this is surely most important if you're making a new gamemode from scratch and actually have it running publicly already), you might want to add a new field. But you can be sure that as soon as you add the new field somewhere in the table (not in the end of it), this can cause trouble.
This issue used to be worse in R6-2 where I used sscanf for parsing my database results. An extra unwanted field added to the parsed string instantly caused a lot of trouble.
This is not as bad in R7 with cache functions, but you should also take note that you do not always need all the fields. For example, you're making the query I mentioned above (SELECT * FROM players WHERE name = 'MP2' LIMIT 0,1) - you do not need to get all the data. You already know the player's name for example, you also might not need their password in case you're making an automatic login or if you compare the passwords in the query itself. So the less fields you have in your result, the faster it actually is. And I'm not saying that this is something that will be noticeable, but it is just about the coding practice and I doubt that specifying the required fields in the query itself is slower than eliminating the unwanted fields later on in your loading callback.
Two: Using field names
There's nothing wrong with this, yes. Use cache_get_field_content which is also documented and given an example in the first post of this topic.
An example for this would be:
This code is untested yet should give you the basic idea of this. Also note that you actually don't need to specify the fields you're querying, so your query can very well look like:
SELECT * FROM players WHERE name = 'MP2' LIMIT 0,1
Also, since you're starting out with your new gamemode, perhaps I can help you with your tables as well, do you mind posting your table syntax?
Have fun coding!
One: knowing what you query for and thereby knowing field indexes.
Adding a new field that you query for does not require you to modify hundreds of indexes in your code, what you can do is use a query which specifies all the field names that you are loading. For example:
pawn Code:
mysql_function_query(dbHandle, "SELECT level,kills,deaths,online_time,vehicle FROM players WHERE name = 'MP2' LIMIT 0,1", true, "OnPlayerDataLoad", "i", playerid);
// Field indexes: level = 0, kills = 1, deaths = 2, online_time = 3, vehicle = 4
pawn Code:
mysql_function_query(dbHandle, "SELECT level,kills,deaths,online_time,vehicle,health FROM players WHERE name = 'MP2' LIMIT 0,1", true, "OnPlayerDataLoad", "i", playerid);
pawn Code:
cache_get_row(0, 5, temp), health = strval(temp); // yeah, I did not make it a float
So what's the advantage to this behavior? You can easily do on-the-run development and modifications to your database!
Lets say you have such query:
SELECT * FROM players WHERE name = 'MP2' LIMIT 0,1
What does this mean? All fields in the table are queried for and will be returned in whatever callback you choose to use. If you are developing a new feature for your server (and this is surely most important if you're making a new gamemode from scratch and actually have it running publicly already), you might want to add a new field. But you can be sure that as soon as you add the new field somewhere in the table (not in the end of it), this can cause trouble.
This issue used to be worse in R6-2 where I used sscanf for parsing my database results. An extra unwanted field added to the parsed string instantly caused a lot of trouble.
This is not as bad in R7 with cache functions, but you should also take note that you do not always need all the fields. For example, you're making the query I mentioned above (SELECT * FROM players WHERE name = 'MP2' LIMIT 0,1) - you do not need to get all the data. You already know the player's name for example, you also might not need their password in case you're making an automatic login or if you compare the passwords in the query itself. So the less fields you have in your result, the faster it actually is. And I'm not saying that this is something that will be noticeable, but it is just about the coding practice and I doubt that specifying the required fields in the query itself is slower than eliminating the unwanted fields later on in your loading callback.
Two: Using field names
There's nothing wrong with this, yes. Use cache_get_field_content which is also documented and given an example in the first post of this topic.
An example for this would be:
pawn Code:
// Query somewhere
mysql_function_query(dbHandle, "SELECT kills,deaths,level FROM players WHERE name = 'MP2' LIMIT 0,1", true, "OnPlayerDataLoad", "i", playerid);
// The callback
forward OnPlayerDataLoad(playerid);
public OnPlayerDataLoad(playerid)
{
new rows, fields;
cache_get_data(rows, fields);
if(!rows)
{
// We did not get any rows - account does not exist?
return 1;
}
// Start storing the data
new temp[12];
cache_get_field_content(0, "kills", temp), gPlayerData[playerid][pKills] = strval(temp);
cache_get_field_content(0, "deaths", temp), gPlayerData[playerid][pDeaths] = strval(temp);
cache_get_field_content(0, "level", temp), gPlayerData[playerid][pLevel] = strval(temp);
return 1;
}
SELECT * FROM players WHERE name = 'MP2' LIMIT 0,1
Also, since you're starting out with your new gamemode, perhaps I can help you with your tables as well, do you mind posting your table syntax?
Have fun coding!