ASP.NET Web サイトから WCF サービスを呼び出すと、次の例外が発生します。どうすればそれを克服できますか?
注: サービス プロジェクトにブレーク ポイントを適用することにより、サービスが 2 つの有効なオブジェクトを返すことを確認しました。
注: サービスでは、IBankAccount のリストを返しています。[OperationContract]
List<IBankAccount> GetDataUsingDataContract(int userId);
IBankAccount はインターフェイスです。
例外には、「基になる接続が閉じられました: 接続が予期せず閉じられました」と表示されます。詳細なスタック トレースは、次の図で確認できます。
//Webサイト
using System;
using ServiceReference1;
public partial class _Default : System.Web.UI.Page
{
protected void Page_Load(object sender, EventArgs e)
{
Service1Client client = new Service1Client();
string result = client.GetData(1);
Response.Write(result);
client.GetDataUsingDataContract(1);
int d = 0;
}
}
//サービス インターフェイス
using System.Collections.Generic;
using System.ServiceModel;
using DTOProject;
namespace MyServiceApp
{
[ServiceContract]
public interface IService1
{
[OperationContract]
string GetData(int value);
[OperationContract]
List<IBankAccount> GetDataUsingDataContract(int userId);
}
}
//DTO
using System.Runtime.Serialization;
namespace DTOProject
{
public interface IBankAccount
{
int Duration { get; set; }
int AmountDeposited { get; set; }
}
}
using System.Runtime.Serialization;
namespace DTOProject
{
[DataContract]
public class FixedAccount : IBankAccount
{
[DataMember]
public int Duration { get; set; }
[DataMember]
public int AmountDeposited { get; set; }
}
}
using System.Runtime.Serialization;
namespace DTOProject
{
[DataContract]
public class SavingsAccount : IBankAccount
{
[DataMember]
public int Duration { get; set; }
[DataMember]
public int AmountDeposited { get; set; }
}
}
//サービスの実装
using System.Collections.Generic;
using DTOProject;
using BusinessLayer;
namespace MyServiceApp
{
public class Service1 : IService1
{
public string GetData(int value)
{
return string.Format("You entered: {0}", value);
}
public List<IBankAccount> GetDataUsingDataContract(int userId)
{
BusinessLayer.AccountManager accManager = new AccountManager();
List<IBankAccount> accounts = accManager.GetAllAccountsForUser(userId);
return accounts;
}
}
}
//ビジネス層
using System.Collections.Generic;
using DTOProject;
using DataAccessLayer;
namespace BusinessLayer
{
public class AccountManager
{
public List<IBankAccount> GetAllAccountsForUser(int userID)
{
DataAccessLayer.AccounutManagerDAL accountManager = new AccounutManagerDAL();
List<IBankAccount> accountList = accountManager.GetAllAccountsForUser(userID);
return accountList;
}
}
}
//データ アクセス層
using System;
using System.Collections.Generic;
using DTOProject;
namespace DataAccessLayer
{
public class DatabaseRecordSimulation
{
public string AccountType { get; set; }
public int Duration { get; set; }
public int DepositedAmount { get; set; }
}
public class AccounutManagerDAL
{
List<DatabaseRecordSimulation> dbRecords = new List<DatabaseRecordSimulation>()
{
new DatabaseRecordSimulation{AccountType="Savings",Duration=6,DepositedAmount=50000},
new DatabaseRecordSimulation{AccountType="Fixed",Duration=6,DepositedAmount=50000}
};
public List<IBankAccount> GetAllAccountsForUser(int userID)
{
List<IBankAccount> accountList = new List<IBankAccount>();
foreach (DatabaseRecordSimulation dbRecrod in dbRecords)
{
IBankAccount acc = AccountFactory.GetAccount(dbRecrod);
accountList.Add(acc);
}
return accountList;
}
}
public static class AccountFactory
{
public static IBankAccount GetAccount(DatabaseRecordSimulation dbRecord)
{
IBankAccount theAccount = null;
if ( String.Equals(dbRecord.AccountType, "Fixed"))
{
theAccount = new FixedAccount();
}
if (String.Equals(dbRecord.AccountType, "Savings"))
{
theAccount = new SavingsAccount();
}
return theAccount;
}
}
}
読み方: 1. WCF オブジェクトの設計 - OOP と SOA