ISerializable Interface in C#

What is the ISerializable Interface in C#?

The **ISerializable** interface in C# allows developers to **customize the serialization process** by defining how an object is serialized and deserialized. It is commonly used when working with **private fields, custom logic, or backward compatibility**.

Key Features of ISerializable Interface

  • Provides **full control** over serialization and deserialization.
  • Used for **handling private fields, encryption, and versioning**.
  • Works with **BinaryFormatter and other serialization methods**.
  • Requires implementing the **GetObjectData()** method.

Implementing ISerializable for Custom Serialization

To implement **ISerializable**, a class must define the **GetObjectData()** method and a **special constructor** to handle deserialization.

Example: Custom Serialization Using ISerializable

using System;
using System.IO;
using System.Runtime.Serialization;
using System.Runtime.Serialization.Formatters.Binary;

[Serializable]
class User : ISerializable
{
    public string Username { get; set; }
    private string Password { get; set; }

    public User() { }

    public User(string username, string password)
    {
        Username = username;
        Password = password;
    }

    // Custom Serialization
    protected User(SerializationInfo info, StreamingContext context)
    {
        Username = info.GetString("Username");
        Password = Decrypt(info.GetString("Password"));
    }

    public void GetObjectData(SerializationInfo info, StreamingContext context)
    {
        info.AddValue("Username", Username);
        info.AddValue("Password", Encrypt(Password));
    }

    private string Encrypt(string data) => $"encrypted({data})";
    private string Decrypt(string data) => data.Replace("encrypted(", "").Replace(")", "");

    public string GetPassword() => Password;
}

// Writing Custom Serialized Object
class Program
{
    static void Main()
    {
        User user = new User("Alice", "SuperSecret123");
        BinaryFormatter formatter = new BinaryFormatter();

        using (FileStream stream = new FileStream("user.dat", FileMode.Create))
        {
            formatter.Serialize(stream, user);
        }

        Console.WriteLine("User object serialized with encryption.");
    }
}

// Output:
// User object serialized with encryption.
        

The **ISerializable** interface allows encrypting passwords during serialization.

Deserializing ISerializable Objects

The **ISerializable** interface requires a **custom constructor** to reconstruct objects during deserialization.

Example: Deserializing a Custom ISerializable Object

using System;
using System.IO;
using System.Runtime.Serialization;
using System.Runtime.Serialization.Formatters.Binary;

class DeserializeExample
{
    static void Main()
    {
        BinaryFormatter formatter = new BinaryFormatter();
        using (FileStream stream = new FileStream("user.dat", FileMode.Open))
        {
            User user = (User)formatter.Deserialize(stream);
            Console.WriteLine($"Username: {user.Username}, Password: {user.GetPassword()}");
        }
    }
}

// Output:
// Username: Alice, Password: SuperSecret123
        

The **custom constructor** ensures that sensitive data like passwords are properly handled during deserialization.

Handling Versioning with ISerializable

When adding **new fields** to a class, the **ISerializable** interface can help maintain backward compatibility.

Example: Handling Versioning in ISerializable

using System;
using System.IO;
using System.Runtime.Serialization;
using System.Runtime.Serialization.Formatters.Binary;

[Serializable]
class Employee : ISerializable
{
    public string Name { get; set; }
    private int Salary { get; set; }

    public Employee() { }

    public Employee(string name, int salary)
    {
        Name = name;
        Salary = salary;
    }

    protected Employee(SerializationInfo info, StreamingContext context)
    {
        Name = info.GetString("Name");
        Salary = info.GetInt32("Salary", defaultValue: 50000); // Default value for older versions
    }

    public void GetObjectData(SerializationInfo info, StreamingContext context)
    {
        info.AddValue("Name", Name);
        info.AddValue("Salary", Salary);
    }
}

// Writing Versioned Serialized Object
class Program
{
    static void Main()
    {
        Employee emp = new Employee("Charlie", 70000);
        BinaryFormatter formatter = new BinaryFormatter();

        using (FileStream stream = new FileStream("employee.dat", FileMode.Create))
        {
            formatter.Serialize(stream, emp);
        }

        Console.WriteLine("Employee object serialized with version handling.");
    }
}

// Output:
// Employee object serialized with version handling.
        

**Default values** can be provided for older versions of serialized objects to ensure compatibility.

Best Practices for Using ISerializable

  • Use **ISerializable** when handling **private fields and encryption**.
  • Implement a **custom deserialization constructor** to properly reconstruct objects.
  • For **JSON serialization**, use **JsonConverter** instead of ISerializable.
  • Ensure **version compatibility** by providing default values for new fields.