我编写了一个包含User类的程序,该程序保存到文件中以存储用户信息。但是,可以有多个不同的用户,因此不是将文件保存为单个用户,而是用户列表。这是在XNA,因此是一个游戏,但这不应该对事情有任何影响。用户拥有硬币,皮肤,库存,用户名,密码(如果有),以及hasPassword bool。因为用户有硬币,我不希望这是一个可编辑的文本文件,因此我将其保存为二进制文件。我写了一些代码,测试过它,一切都很好,花花公子,直到我试图将多个用户保存到List。出于某种原因,每当我保存第二个用户并读入列表时,列表只有一个对象(第一个用户)。在这里,我将提供代码和示例,以便更容易理解:
[Serializable]
class User
{
#region Fields & Properties
public string Username = "";
public string Password = "";
public bool HasPassword = false;
public int Coins = 0;
public List<Achievement> AchievementsCompleted = new List<Achievement>();
public List<InventoryItem> Inventory = new List<InventoryItem>();
public List<string> Skins = new List<string>();
public string CurrentSkinAsset { get; set; }
const int SPACING = 10;
const string FILE_PATH = "Users.bin";
#endregion
#region Constructors
public User(string username, int coins, string skinPath, ContentManager content)
{
Username = username;
CurrentSkinAsset = skinPath;
}
public User(string username, string password, int coins, string skinPath, ContentManager content)
: this(username, coins, skinPath, content)
{
HasPassword = true;
Password = password;
}
#endregion
#region Public Methods
public static List<User> LoadUsers()
{
FileStream fileStream = null;
List<User> returnList = null;
try
{
if (File.Exists(FILE_PATH))
{
fileStream = new FileStream(FILE_PATH, FileMode.Open, FileAccess.Read);
BinaryFormatter deserializer = new BinaryFormatter();
returnList = (List<User>)deserializer.Deserialize(fileStream);
}
else
{
fileStream = File.Create(FILE_PATH);
List<User> users = new List<User>();
BinaryFormatter serializer = new BinaryFormatter();
serializer.Serialize(fileStream, users);
}
}
catch (Exception e)
{
Console.WriteLine("There's been an error. Here is the message: " + e.Message);
}
finally
{
if (fileStream != null)
{
fileStream.Close();
}
}
return returnList;
}
public void SerializeUser()
{
// Also works
FileStream fileStream = null;
List<User> users;
BinaryFormatter binaryFormatter = new BinaryFormatter();
try
{
if (File.Exists(FILE_PATH))
{
fileStream = File.Open(FILE_PATH, FileMode.Open);
//fileStream = new FileStream(FILE_PATH, FileMode.Open, FileAccess.ReadWrite);
users = (List<User>)binaryFormatter.Deserialize(fileStream);
}
else
{
fileStream = File.Create(FILE_PATH);
users = new List<User>();
}
for (int i = 0; i < users.Count; i++)
{
if (users[i].Username.ToLower() == this.Username.ToLower())
{
users.Remove(users[i]);
}
}
users.Add(this);
binaryFormatter.Serialize(fileStream, users);
}
catch (Exception e)
{
Console.WriteLine("There's been an error. Here is the message: " + e.Message);
}
finally
{
if (fileStream != null)
{
fileStream.Close();
}
}
}
#endregion
}
这是我写的测试代码(在Game1 LoadContent方法中):
User test = new User("Bob", 0, "skin1", Content)
test.SerializeUser();
我查看了Debug文件夹,发现了一个名为“Users.bin”的新文件。它有一堆随机字符,如预期的那样。我将测试的用户名参数从“Bob”更改为“Joe”,后者应该添加一个新用户。但是...
以下是阅读用户的代码:
List<User> testList = User.LoadUsers();
我在下一行代码中插入了一个断点,并在testList上方悬停,但唯一可见的用户是Bob,而Joe则不存在。有任何想法吗? 另外:我十三岁。我可能不理解你使用的所有术语,因为我没有编程很长时间。 提前谢谢!
答案 0 :(得分:3)
问题是,当您保存用户列表时,您打开FileStream
,在列表中读取(以替换当前用户),然后将新列表保存到FileStream
没有重置位置。初始读取将FileStream
推进到文件的末尾 - 保存然后将新列表附加到初始列表之后的FileStream。
然后会发生一个如下所示的文件:
*start of file*
List(User1)
List(User1,User2)
*end of file*
每次阅读时,您都只能获得初始列表。
试试这个:
users.Add(this);
filStream.Position = 0;
binaryFormatter.Serialize(fileStream, users);
对于它的价值,这是处理保存和加载用户的非常效率低下的方法。您实际上是在为任何用户的每次更改阅读和编写整个列表。它也不是线程安全的(意味着如果这是一个具有多个同步更新的多人游戏,您的用户文件可能会受到损坏,具体取决于您打开它的方式以及发生的更新)。您可能希望查看正确的数据库而不是普通文件。