views:

293

answers:

6

Hi,

Is there a way to control the type conversion in C#? So for example, if I have two types with essentially the same details, but one is used for the internal working of my application and the other is a DTO used for communicating with non-.Net applications:

public sealed class Player
{
  public Player(string name, long score)
  {
    Name = name;
    Score = score;
    ID = Guid.NewGuid();
  }

  public string Name { get; private set; }

  public Guid ID { get; private set; }

  public long Score { get; private set; }
}

public sealed class PlayerDTO
{
  public PlayerDTO(string name, long score, string id)
  {
    Name = name;
    Score = score;
    ID = id;
  }

  public string Name { get; private set; }

  // the client is not .Net and doesn't have Guid
  public string ID { get; private set; }  

  public long Score { get; private set; }
}

Right now, I need to create a new instance of PlayerDTO from my Player instance every time and I'm looking for a better, cleaner way of doing this. One idea I had was to add an AsPlayerDTO() method to the player class, but would be nice if I can control the type conversion process so I can do this instead:

var playerDto = player as PlayerDTO; 

Anyone know if this is possible and how I might be able to do it?

Thanks,

+3  A: 

You can implement either implicit or explicit type conversion: http://msdn.microsoft.com/en-us/library/ms173105.aspx.

Alternately, if you want to avoid making each class having to know about the other, you can use either custom mapping or an existing mapping library like AutoMapper.

Pete
Overloading the implicit/explicit operator seems to work for me and it's reasonably clean too.Curious though, this doesn't apply to arrays, but AutoMapper seems to give that option based on Michael Meadows' comment below. Is there a way to implement this without using 3rd extensions like AutoMapper?
theburningmonk
Never mind, found the answer to my second question on another stackoverflow question http://stackoverflow.com/questions/1865031/why-wont-castdouble-work-on-ienumerableintArray.ConvertAll(playerList.ToArray(), p => (PlayerDTO)p);
theburningmonk
Also, if you're using Linq, you can use Select... playerArray.Select(player => (PlayerDTO) player).ToArray();
Michael Meadows
thanks Michael, +1 for that, skipped me entirely :-P
theburningmonk
A: 

What about conversion operator:

public static explicit operator PlayerDTO (Player value)...
public static implicit operator PlayerDTO (Player value)...
Dewfy
+5  A: 

You can implement an explicit convesion operator between the two types.

You could also consider using AutoMapper for the task.

Mark Seemann
+1 for AutoMapper
mhenrixon
Also +1 for AutoMapper. The "Getting Started" is almost exactly the OP's scenario.http://automapper.codeplex.com/wikipage?title=Getting%20Started
Michael Meadows
A: 

You'll still have to do the code to create the DTO object, but you can overload the cast operator to cast from your implementation object to the DTO object. You can even make an implicit cast.

see overloading cast and implicit

Pete McKinney
A: 

You can use explicit type conversions.

public sealed class Player {
  public static explicit operator PlayerDTO(Player p) {
    PlayerDTO dto;
    // construct
    return dto;
  }
}

public sealed class PlayerDTO {
  public static explicit operator Player(PlayerDTO dto) {
    Player p;
    // construct
    return p;
  }
}
gWiz
A: 

Use AutoMapper @ http://www.codeplex.com/AutoMapper

epitka