2009-02-04 6 views
24

В службе WCF у меня есть два класса с атрибутом [DataContract]. Один из этих классов имеет отношение «is-a» к другому, поэтому класс B может наследовать от класса A. Однако, когда я настраиваю наследование между этими двумя классами, оба обозначаются атрибутом [DataContract], метаданные не загружаются при тестировании услуг.В WCF, могут ли наследовать друг от друга классы контрактов данных?

Возможно ли это в WCF? Я пропустил еще один атрибут?

[DataContract] 
public class A 
{   
    [DataMember] 
    public MyCustomType AValue1{ get; set; } 

    [DataMember] 
    public MyCustomType AValue2 { get; set; } 
} 

[DataContract] 
public class B: A 
{  
    [DataMember] 
    public double BValue1{ get; set; } 

    [DataMember] 
    public double BValue2 { get; set; } 
} 

ПРИМЕЧАНИЕ. Пользовательские типы также определяются с использованием контрактов данных.

UPDATE: Ниже ошибка:

Error: Cannot obtain Metadata from http://localhost:8002/GISDataServices/mex If this is a Windows (R) Communication Foundation service to which you have access, please check that you have enabled metadata publishing at the specified address. For help enabling metadata publishing, please refer to the MSDN documentation at http://go.microsoft.com/fwlink/?LinkId=65455.WS-Metadata Exchange Error URI: http://localhost:8002/GISDataServices/mex Metadata contains a reference that cannot be resolved: ' http://localhost:8002/GISDataServices/mex '. Receivera:InternalServiceFault The server was unable to process the request due to an internal error. For more information about the error, either turn on IncludeExceptionDetailInFaults (either from ServiceBehaviorAttribute or from the <serviceDebug> configuration behavior) on the server in order to send the exception information back to the client, or turn on tracing as per the Microsoft .NET Framework 3.0 SDK documentation and inspect the server trace logs.HTTP GET Error URI: http://localhost:8002/GISDataServices/mex There was an error downloading ' http://localhost:8002/GISDataServices/mex '. The request failed with HTTP status 400: Bad Request.

UPDATE 2: Смотрите мой ответ ниже.

ответ

38

Да, но вам необходимо украсить базовый класс [KnownTypeAttribute], построив его с помощью типа производного класса. Например:

[DataContract] 
[KnownType(typeof(B))] 
public class A 
{ 
    [DataMember] 
    public string Value { get; set; } 
} 

[DataContract] 
public class B : A 
{ 
    [DataMember] 
    public string OtherValue { get; set; } 
} 
+0

Это не работает. Я по-прежнему получаю сообщение об ошибке при загрузке метаданных. –

+0

Какая ошибка вы видите? –

+0

Большой вопрос. Можете ли вы высказать свою точную ошибку? –

1

Основываясь на этом тесте, он должен работать нормально. У обоих классов есть конструкторы по умолчанию? Вы используете Авто-Свойства. Примечание. В этом базовом примере атрибуты не требуются. Кроме того, поскольку Дэвид Мортон упомянул вас в зависимости от того, какой элемент вы возвращаетесь, вам может понадобиться атрибут KnownType, я не 100%, но известный тип, возможно, придется выполнить по контракту.

class Program 
{ 
    static void Main(string[] args) 
    { 
     var serializer = new DataContractSerializer(typeof(Employee)); 

     var employee = new Employee() { Name="Joe", Salary=100000 }; 
     using (var ms = new MemoryStream()) 
     { 
      serializer.WriteObject(ms, employee); 

      ms.Position = 0; 

      var newEmployee = serializer.ReadObject(ms) as Employee; 
     } 

     Console.ReadKey(); 

    } 
} 

[DataContract] 
public class Employee : Person 
{ 
    [DataMember] 
    public decimal Salary { get; set; } 
} 

[DataContract] 
public class Person 
{ 
    [DataMember] 
    public string Name { get; set; } 
} 

[ServiceContract] 
interface IEmployeeService 
{ 
    [OperationContract] 
    Person GetPerson(); 

    [OperationContract] 
    Employee GetEmployee(); 

    [OperationContract] 
    [KnownType(typeof(Employee))] 
    Person GetEmployeeAsPerson(); 
} 
+0

Я использую авто-свойства, но они отлично работают, когда я удаляю конструкцию наследования. –

+0

Ни один из классов не использует явный конструктор. –

+0

Можете ли вы обновить сообщение с помощью проблемы наследования? – bendewey

7

Хорошо, я понял вопрос. Ответ ... Я идиот. Это не имело никакого отношения к наследованию. В базовом классе у меня был член контракта с данными без предложения свойства «set» - только «get». Doh !!! Включение в предложение «set» заставило его работать как шарм.

Извините за путаницу.