Использование iBATIS.NET с универсальными пользовательскими интерфейсами коллекций и Unity - PullRequest
1 голос
/ 02 марта 2009

Я пытаюсь использовать универсальный пользовательский интерфейс коллекции (для поддержки внедрения с помощью Microsoft Patterns and Practices Unity) в классе O / R, сопоставленном с iBATIS.NET. Кто-нибудь знает, возможно ли это и если да, то как это сделать?

У меня есть интерфейс IDataItemCollection , который я сопоставляю с SqlDataItemCollection , который расширяет CollectionBase. Я хочу использовать IDataItemCollection в своих классах, чтобы я мог поменять SqlDataItemCollection на другие классы, которые расширяют интерфейс через Unity. Файл отображения iBATIS.NET может напрямую ссылаться на конкретный класс, так как без другого не будет ни одного.

Ниже я привел очень упрощенный пример кода, базы данных и отображений. Я полностью новичок в iBATIS.NET и действительно хочу сейчас доказать его использование, поэтому, при необходимости, заново перенастройте XML-файл сопоставления.

Большое спасибо,

Пол


C # код

public interface IDataItem
{
    object Id { get; set; }
}

public class DataItem : IDataItem
{
    public object Id { get; set; }
}

public interface IDataItemCollection<T> : ICollection where T : IDataItem
{
    // Various Getters and Setters
...
}

public class SqlDataItemCollection<T> : CollectionBase, IDataItemCollection<T> where T : DataItem
{
    public SqlDataItemCollection() { }
    public SqlDataItemCollection(T injType) { }

    // Getters and Setters to implement interfaces
...
}

public class Foo : DataItem
{
    public Foo(IDataItemCollection<Bar> bars)
    {
        Bars = bars;
    }

    public IDataItemCollection<Bar> Bars { get; set; }
}

public class Bar : DataItem { }

База данных SQL Server 2005

CREATE TABLE Foo
(
    Id bigint IDENTITY(1,1)
)

CREATE TABLE Bar
(
    Id bigint IDENTITY(1,1)
)

CREATE TABLE FooBar
(
    FooId bigint,
    BarId bigint
)

iBATIS.NET mapping.xml

<resultMaps>
    <resultMap id="FooResult" class="Foo">
        <result property="Id" column="Id"/>
        <result property="Bars" column="Id" select="SelectBarsInFoo" lazyLoad="false"/>
    </resultMap>

    <resultMap id="BarResult" class="Bar">
        <result property="Id" column="Id"/>
    </resultMap>
</resultMaps>

<statements>
    <select id="SelectFoo" resultMap="FooResult">
        SELECT Id
        FROM Foo
    </select>

    <select id="SelectBarsInFoo" parameterClass="long" resultMap="BarResult" listClass="SqlDataItemCollection`1[Bar]" >
        SELECT Bar.Id
        FROM Bar
        JOIN FooBar ON Bar.Id = FooBar.BarId
        WHERE FooBar.FooId = #value#
    </select>
</statements>

1 Ответ

0 голосов
/ 01 июля 2010

У меня была такая же проблема после рефакторинга части моего приложения обратно на интерфейсы. Я справился с этим, явно реализовав определение интерфейса моей коллекции, а затем продублировав реализацию как ее конкретный класс. Это может не решить вашу проблему.

public interface IGroup { }
public class Group : IGroup { }
public class IGroupCollection : IList<IGroup> { }
public class GroupCollection : IGroupCollection { }

public interface IConcrete
{
    IGroupCollection Items { get; set; }
}

public class Concrete : IConcrete
{
    public GroupCollection Items { get; set; }
    IGroupCollection IConcrete.Items
    {
        get { return Items; }
        set { Items = value as GroupCollection; }
    }
}

Это позволяет iBATIS.NET добавлять элементы в коллекцию, не сталкиваясь с ошибкой преобразования типов, тогда как явная реализация интерфейса позволяет мне использовать IConcrete во всем приложении, не обращаясь к фактическому Concrete или фактическому * 1006. *.

...