C#:在二进制文件

时间:2016-07-24 00:36:31

标签: c# list file

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

1 个答案:

答案 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);

对于它的价值,这是处理保存和加载用户的非常效率低下的方法。您实际上是在为任何用户的每次更改阅读和编写整个列表。它也不是线程安全的(意味着如果这是一个具有多个同步更新的多人游戏,您的用户文件可能会受到损坏,具体取决于您打开它的方式以及发生的更新)。您可能希望查看正确的数据库而不是普通文件。