ホームページ > バックエンド開発 > C#.Net チュートリアル > C# は単体テストのために EntityFramework を分離します

C# は単体テストのために EntityFramework を分離します

黄舟
リリース: 2017-02-28 11:18:24
オリジナル
2237 人が閲覧しました

1. まず第一に、EF のリポジトリにはインターフェイスで言及される抽象的な動作が必要です。

例:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

public interface IXXXContext : IDisposable

    {

        IXXXContext NewInstance();

// db sets

        DbSet<AAABBB> aaa { get; set; }

...

// common

Database Database { get; }

        DbContextConfiguration Configuration { get; }

        int SaveChanges();

 

 

        Task<int> SaveChangesAsync();

 

 

    // store pros

...

        IStorePro1 StorePro1 { get; }

    ...

}

ログイン後にコピー


その後、DataContext と TestDataContext を使用してこのインターフェイスを実装できます。このうちTestDataContextはUTで利用されており、DataContextは自動生成されます。

TestDataContext には、シミュレーション用に次のクラスも必要です。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

105

106

107

108

109

110

111

112

113

114

115

116

117

118

119

120

121

122

123

124

125

126

127

128

129

130

131

132

133

134

135

136

137

138

139

140

141

142

143

144

145

146

147

148

149

150

151

152

153

154

155

156

157

158

159

160

161

162

163

164

165

166

167

168

169

170

171

172

173

174

175

176

177

178

179

180

181

182

183

184

185

186

187

188

189

190

191

192

193

194

195

196

197

198

199

200

201

202

203

public class TestDbSet<TEntity> : DbSet<TEntity>, IQueryable, IEnumerable<TEntity>, IDbAsyncEnumerable<TEntity>

        where TEntity : class

   {

       ObservableCollection<TEntity> _data;

       IQueryable _query;

 

 

       public TestDbSet()

       {

           _data = new ObservableCollection<TEntity>();

           _query = _data.AsQueryable();

       }

 

 

       public override TEntity Add(TEntity item)

       {

           _data.Add(item);

           return item;

       }

 

 

       public override TEntity Remove(TEntity item)

       {

           _data.Remove(item);

           return item;

       }

 

 

       public override TEntity Attach(TEntity item)

       {

           _data.Add(item);

           return item;

       }

 

 

       public override TEntity Create()

       {

           return Activator.CreateInstance<TEntity>();

       }

 

 

       public override TDerivedEntity Create<TDerivedEntity>()

       {

           return Activator.CreateInstance<TDerivedEntity>();

       }

 

 

       public override ObservableCollection<TEntity> Local

       {

           get { return _data; }

       }

 

 

       Type IQueryable.ElementType

       {

           get { return _query.ElementType; }

       }

 

 

       Expression IQueryable.Expression

       {

           get { return _query.Expression; }

       }

 

 

       IQueryProvider IQueryable.Provider

       {

           get { return new TestDbAsyncQueryProvider<TEntity>(_query.Provider); }

       }

 

 

       System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()

       {

           return _data.GetEnumerator();

       }

 

 

       IEnumerator<TEntity> IEnumerable<TEntity>.GetEnumerator()

       {

           return _data.GetEnumerator();

       }

 

 

       IDbAsyncEnumerator<TEntity> IDbAsyncEnumerable<TEntity>.GetAsyncEnumerator()

       {

           return new TestDbAsyncEnumerator<TEntity>(_data.GetEnumerator());

       }

   }

 

 

   internal class TestDbAsyncQueryProvider<TEntity> : IDbAsyncQueryProvider

   {

       private readonly IQueryProvider _inner;

 

 

       internal TestDbAsyncQueryProvider(IQueryProvider inner)

       {

           _inner = inner;

       }

 

 

       public IQueryable CreateQuery(Expression expression)

       {

           return new TestDbAsyncEnumerable<TEntity>(expression);

       }

 

 

       public IQueryable<TElement> CreateQuery<TElement>(Expression expression)

       {

           return new TestDbAsyncEnumerable<TElement>(expression);

       }

 

 

       public object Execute(Expression expression)

       {

           return _inner.Execute(expression);

       }

 

 

       public TResult Execute<TResult>(Expression expression)

       {

           return _inner.Execute<TResult>(expression);

       }

 

 

       public Task<object> ExecuteAsync(Expression expression, CancellationToken cancellationToken)

       {

           return Task.FromResult(Execute(expression));

       }

 

 

       public Task<TResult> ExecuteAsync<TResult>(Expression expression, CancellationToken cancellationToken)

       {

           return Task.FromResult(Execute<TResult>(expression));

       }

   }

 

 

   internal class TestDbAsyncEnumerable<T> : EnumerableQuery<T>, IDbAsyncEnumerable<T>, IQueryable<T>

   {

       public TestDbAsyncEnumerable(IEnumerable<T> enumerable)

           : base(enumerable)

       { }

 

 

       public TestDbAsyncEnumerable(Expression expression)

           : base(expression)

       { }

 

 

       public IDbAsyncEnumerator<T> GetAsyncEnumerator()

       {

           return new TestDbAsyncEnumerator<T>(this.AsEnumerable().GetEnumerator());

       }

 

 

       IDbAsyncEnumerator IDbAsyncEnumerable.GetAsyncEnumerator()

       {

           return GetAsyncEnumerator();

       }

 

 

       IQueryProvider IQueryable.Provider

       {

           get { return new TestDbAsyncQueryProvider<T>(this); }

       }

   }

 

 

   internal class TestDbAsyncEnumerator<T> : IDbAsyncEnumerator<T>

   {

       private readonly IEnumerator<T> _inner;

 

 

       public TestDbAsyncEnumerator(IEnumerator<T> inner)

       {

           _inner = inner;

       }

 

 

       public void Dispose()

       {

           _inner.Dispose();

       }

 

 

       public Task<bool> MoveNextAsync(CancellationToken cancellationToken)

       {

           return Task.FromResult(_inner.MoveNext());

       }

 

 

       public T Current

       {

           get { return _inner.Current; }

       }

 

 

       object IDbAsyncEnumerator.Current

       {

           get { return Current; }

       }

   }

ログイン後にコピー


使用例:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

[TestMethod]

        public void TestMethod1()

        {

            var mockSet = new Mock<DbSet<BLACKLISTED_TICKET>>();

            var mockContext = new Mock<TicketDataContextTest>();

            mockContext.Setup(m => m.BLACKLISTED_TICKET).Returns(new TestDbSet<BLACKLISTED_TICKET>());

 

 

            var context = mockContext.Object;

 

 

            context.BLACKLISTED_TICKET.Add(new BLACKLISTED_TICKET()

            {

                TicketNumber = "aaa",

                CreatedDateTime = DateTime.Now,

                Id = 1,

                ModifiedDateTime = DateTime.Now,

                STATUS = "1"

            });

 

 

            Assert.IsTrue(context.BLACKLISTED_TICKET.First().Id == 1);

        }

ログイン後にコピー

ストアドプロシージャを使用する場合、ストアドプロシージャのインターフェースを追加定義する必要があります。
例:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

IStorePro {

...

}

 

 

StorePro : IStorePro{

...

}

 

 

StoreProFake: IStorePro{

 

 

}

ログイン後にコピー

その後、IDataContext はストアド プロシージャのインスタンスを返す責任があります

1

2

3

4

5

IDataContext{

    ...

    IStorePro GetStorePro();

    ...

}

ログイン後にコピー

上記は、単体テスト用に EntityFramework を分離する C# の内容です。さらに関連する内容については、PHP 中国語 Web サイトに注意してください。 (www.php.cn)!


関連ラベル:
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート