C#:向二进制文件写入和读取列表

标签 c# list file

我写了一个程序,有一个User类,保存到文件中用来存储用户信息。但是,可以有多个不同的用户,因此不是将文件保存为单个用户,而是一个用户列表。这是在 XNA 中,因此是一个游戏,但这应该不会对事情产生任何影响。用户有金币、皮肤、他们的库存、他们的用户名、他们的密码(如果他们有的话)和一个 hasPassword bool 值。因为用户有币,我不希望这个是一个可编辑的文本文件,所以我把它保存为一个二进制文件。我写了一些代码,对其进行了测试,一切都很好,很漂亮,直到我尝试将多个用户保存到列表中。出于某种原因,每当我保存第二个用户并在列表中读取时,列表只有一个对象(第一个用户)。在这里,我将提供代码和示例以使其更易于理解:

[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 不存在。有任何想法吗? 另外:我十三岁。我可能不理解您使用的所有术语,因为我已经很长时间没有编程了。 提前致谢!

最佳答案

问题是,当您保存用户列表时,您正在打开 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);

就其值(value)而言,这是处理保存和加载用户的非常效率低下的方法。您实际上是在为任何用户的每次更改读取和写入整个列表。它也不是线程安全的(这意味着如果这是一个同时进行多个更新的多人游戏,您的用户文件可能会损坏,具体取决于您打开它的方式和发生的更新)。您可能想要查看适当的数据库而不是普通文件。

关于C#:向二进制文件写入和读取列表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38547647/

相关文章:

c# - 将日期时间 js 转换为 c#

c# - 带有附加列的sql批量插入

c# - 如何将列表转换为 C# 中的 observablecollection?

python - 如何从列表中删除除 Python 中的一个元素之外的所有重复项?

list - 检查2个列表是否有相等的元素 Haskell

Java - 从文件中读取混合数据类型(char + int + 二进制)

windows - 使用应用程序注册的扩展列表

ruby - 在 ruby​​ 中按姓氏排序电子邮件地址文件?

C#:限制字符串的长度?

c# - 旋转 UILabel 的拉伸(stretch)宽度