8

ここで少しパターンの方向性が必要です。C# 初心者。

Web サービスをラップするサードパーティの開発キットを使用しています。私が扱っている 2 つの特定のクラスは、比較的似ていますが、開発キットの 2 つの異なる名前空間にあり、共通の基本クラスはありません。ただし、両方の共通インターフェイスに対してプログラムしたいと思います。本質的にラッパーをラップする実装を無計画にまとめましたが、型キャストが絶え間なく行われるため、これが最も効率的な方法ではないことは確かです。

アダプター、インターフェース、拡張メソッドなどに関する記事を掘り下げてきましたが、時間が足りないので、一方向にプッシュできれば大歓迎です.

using ThirdParty.TypeA.Employee;
using ThirdParty.TypeB.Employee;

public class Employee
{
     private object genericEmployee;

     private EmployeeType empType;

     public enum EmployeeType
     {
          TypeA = 0;
          TypeB = 1;
     }   

     public Employee(Object employee, EmployeeType type)
     {
         genericEmployee = employee;
         empType = type;
     }

     public String Name
     {
         if (empType == EmployeeType.TypeA)
             return (ThirdParty.TypeA.Employee)genericEmployee.Name;
         else
             return (ThirdParty.TypeB.Employee)genericEmployee.Name;
     }

     public String Age
     {
         if (empType == EmployeeType.TypeA)
             return (ThirdParty.TypeA.Employee)genericEmployee.Age;
         else
             return (ThirdParty.TypeB.Employee)genericEmployee.Age;
     }
 }

リビジョン 2:

class EmployeeTypeAAdapter : TypeA, IEmployeeAdapter
{
    TypeA _employee;
    public EmployeeTypeAAdapter(TypeA employee) 
    { 
         _employee = employee
    }

     public String Name
     {
        get { return _employee.Name; }  
        set { _employee.Name = value; }
     } 

      public String Balance
      {
        get
        {
            if (_employee.Balance != null)
            {
                decimal c = _employee.Balance.Amount;
                return String.Format("{0:C}", c);
            }
            else
            {
                return "";
            }
          }
       }  

       //...
}

class EmployeeTypeBAdapter : TypeB, IEmployeeAdapter
{
    TypeB _employee;
    public EmployeeTypeAAdapter(TypeB employee) 
    { 
         _employee = employee
    }

     public String Name
     {
        get { return _employee.Name; }  
        set { _employee.Name = value; }
     } 

      public String Balance
      {
        get
        {
            if (_employee.Balance != null)
            {
                decimal c = _employee.Balance.Amount;
                return String.Format("{0:C}", c);
            }
            else
            {
                return "";
            }
          }
       }  

     //....
}
4

2 に答える 2

9

このアプローチを試してください:

public interface IEmployeeAdapter
{
    string Age { get; set; }
    string Name { get; set; }
}

class EmployeeTypeAAdapter : TypeA, IEmployeeAdapter
{
    public EmployeeTypeAAdapter(TypeA employee) { }
}

class EmployeeTypeBAdapter : TypeB, IEmployeeAdapter
{
    public EmployeeTypeBAdapter(TypeB employee) { }
}

public static class EmployeeAdapterFactory
{
    public static IEmployeeAdapter CreateAdapter(object employee, EmployeeType type)
    {
        switch (type)
        {
            case EmployeeType.TypeA: return new EmployeeTypeAAdapter((TypeA)employee);
            case EmployeeType.TypeB: return new EmployeeTypeBAdapter((TypeB)employee);
        }
    }

    // or without enum

    public static IEmployeeAdapter CreateAdapter(object employee)
    {
        if (employee is TypeA) return new EmployeeTypeAAdapter((TypeA)employee);
        if (employee is TypeB) return new EmployeeTypeABdapter((TypeB)employee);
    }

    // or better introduce sort of type map
}

もう 1 つの適切な名前は、お好みで EmployeeProxy です。

于 2013-01-23T18:54:16.920 に答える
3

あなたがやろうとしていることは、ダックタイピングとして知られています。これは、アダプター クラスと共有インターフェイスを使用して行うことができますが、これらのアダプターを手動で作成するには、多くの反復的なグルー コードが必要です。グルー コードの作成を回避する方法の 1 つは、アダプターの型を動的に構築することです。これは、IL Emit を介して自分で行うことができます (これまでにこれを試す機会がなかった場合は、検討する価値のある演習ですが、考慮すべき境界ケースがかなりある可能性があります)。ただし、このプロジェクトをチェックしてくださいスタート地点として。C# の「動的」型も使用できます (そして、同じコード生成を舞台裏で実行することになります) が、非動的コードにインターフェイスであるかのように渡すことができる参照を提供しません。タイプ。

于 2013-01-23T23:54:05.690 に答える