MVVM + EF czy dobrze?

MVVM + EF czy dobrze?
grzesiek51114
grzesiek51114
  • Rejestracja:ponad 11 lat
  • Ostatnio:ponad 4 lata
  • Postów:2442
0

Witajcie,

jak co niektórzy wiedzą uczę się Entity Framework i chcę go dobrze zastosować wespół z MVVM.

Teraz chcę się zapytać czy moje podejście jest dobre?

Wołam @DibbyDum i innych mvvm'owców.

PS: Wiem, że EF jest jaki jest ale mimo wszystko chcę go poznać. Później równolegle sobie może nHibernate zrobię.

edytowany 3x, ostatnio: grzesiek51114
DibbyDum
@fasadin jako człowiek pracujący na co dzień na pewno też coś dorzuci od siebie. :)
fasadin
niestety nie pracuje z EF. Jedynie do czego moglbym sie przyczepic to formatowanie w FindBook (nie podoba mi sie jak rozrozniasz if/else + linq wole rozdrabniac jezeli jest bardziej skomplikowane. Na pierwrzy rzut oka wyglada ok, ale nie wime jak sie z EF pracuje
T2
  • Rejestracja:ponad 10 lat
  • Ostatnio:7 miesięcy
  • Postów:194
1

Odnośnie ViewModelu:

  • Usuń magic stringi z property change
  • Operacje wyszukiwania oraz dodawania przesuń do osobnej klasy. Natomiast w ViewModelu używaj słabej referencji do tej klasy (Interfejsu). Zapewni to tobie większą możliwość testowania.
  • Uwaga od mnie ja bym przeniósł definicje akcji do osobnych method. Ułatwi to potem np. wprowadzenie Controllera.
edytowany 1x, ostatnio: teo215
DibbyDum
  • Rejestracja:ponad 12 lat
  • Ostatnio:ponad rok
  • Lokalizacja:Polska, Kraków
3

Słowem wstępu MVVM to wzorzec architektury dzielący ogólnikowo aplikacje na trzy warstwy odseparowane od siebie.

  1. View odpowiada za GUI, czyli za wygląd formatek za wszelakie wodotryski czy to napisane w XAML czy w c# code-behind bez różnicy.
  2. ViewModel tutaj leci logika aplikacji, głownie odpowiedzialna za pobierz dane z Model przekaż do View i z powrotem.
  3. Model tutaj powinna być logika biznesowa (domena), serwisy, repozytoria i inne szmery bajer które korzystają na przykład z bazy danych za pośrednictwem EF.

A więc wracając do twojego kodu:

  1. Wywalić z ViewModel ten "DbContext" i ubrać to w jakieś serwisy po stronie modelu.
  2. Model to nie mapowanie tabelki z bazy danych.
  3. "DbContext" jak "DbContext" jest okej nie ma co się nad nim zastanawiać.

View, ViewModel i Model to ogólniki pod nimi zawsze kryją się bardziej szczegółowe wzorce i implementacje.

P.S. nie przepracowałem ani pół sekundy jako WPF/MVVM developer więc równie dobrze mogę pieprzyć głupoty. :)


Yubby dibby dibby dibby dibby dibby dibby dum..
edytowany 2x, ostatnio: DibbyDum
0

To dopiero początek przygody z MVVM, pamiętam jak kląłem kiedy coś mi nie działało, MVVM jest "magiczne" :D

grzesiek51114
grzesiek51114
  • Rejestracja:ponad 11 lat
  • Ostatnio:ponad 4 lata
  • Postów:2442
0

Dzięki wszystkim za feedback. No ja juz próbuję mvvm używać dobre dwa latka :) Ciągle dowiaduję się czegoś nowego :)

grzesiek51114
grzesiek51114
  • Rejestracja:ponad 11 lat
  • Ostatnio:ponad 4 lata
  • Postów:2442
0

@DibbyDum, @teo215 Czy chodziło o taki serwis?

Kopiuj
using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Library.Model;

namespace Library.Services {
    public interface ILibraryService {
        ICollection<Book> GetBooks();
        ICollection<Author> GetAuthors();
        ICollection<BooksView> GetBooksView();

        bool InsertBook(Book book);
        bool InsertAuthor(Author author);
        bool Associate(int bookId, int authorId);
    }

    public class LibraryService : ILibraryService {
        public ICollection<Book> GetBooks() {
            using (var db = new LibraryContext()) {
                return db.Books.ToList();
            }
        }

        public ICollection<Author> GetAuthors() {
            using (var db = new LibraryContext()) {
                return db.Authors.ToList();
            }
        }

        public ICollection<BooksView> GetBooksView() {
            using (var db = new LibraryContext()) {
                return db.BooksView.ToList();
            }
        }

        public bool InsertBook(Book book) {
            try {
                using (var db = new LibraryContext()) {
                    db.Books.Add(book);
                    return db.SaveChanges() > 0;
                }
            }
            catch (Exception) {
                return false;
            }
        }

        public bool InsertAuthor(Author author) {
            try {
                using (var db = new LibraryContext()) {
                    db.Authors.Add(author);
                    return db.SaveChanges() > 0;
                }
            }
            catch (Exception) {
                return false;
            }
        }

        public bool Associate(int bookId, int authorId) {
            try {
                using (var db = new LibraryContext()) {
                    var book = db.Books.Single(b => b.Id == bookId);
                    var author = db.Authors.Single(a => a.Id == authorId);

                    book.Authors.Add(author);
                    author.Books.Add(book);

                    return db.SaveChanges() > 0;
                }
            }
            catch (Exception ex) {
                System.Windows.MessageBox.Show(ex.ToString());
                return false;
            }
        }
    }
}

Generalnie baza jest prosta. Taka tam... żeby ogarnąć działanie ORM:

61883b86f2.png

Klasy Book i Author to fizyczne tabele w bazie danych. Klasa BookView to klasa, która bierze dane z widoku zrobionego w bazie.
Czy chodziło o takie (albo podobne) podejście? :)

PS: Nie zwracajcie uwagi na ogolny Exception i to, że jest tam tylko return false. Chodzi generalnie o ideę serwisu. W viewmodelu będzie luźna referencja w postaci interfejsu.

edytowany 2x, ostatnio: grzesiek51114
DibbyDum
  • Rejestracja:ponad 12 lat
  • Ostatnio:ponad rok
  • Lokalizacja:Polska, Kraków
1

Koncepcja jest ogólnie słuszna tylko zamiast ObservableCollection zwracał bym IList.


Yubby dibby dibby dibby dibby dibby dibby dum..
grzesiek51114
grzesiek51114
  • Rejestracja:ponad 11 lat
  • Ostatnio:ponad 4 lata
  • Postów:2442
0

@DibbyDum, @teo215 robicie jeden duży serwis czy dzielicie serwis per tabela? A może jeszcze inaczej? Pytam, bo jak tych tabel jest dużo w bazie to per tabela porobi się mnóstwo serwisów.

DibbyDum
  • Rejestracja:ponad 12 lat
  • Ostatnio:ponad rok
  • Lokalizacja:Polska, Kraków
2

Wiele serwisów, ale nie per tabela tylko per logika biznesowa, takie SRP biznesowe. :D


Yubby dibby dibby dibby dibby dibby dibby dum..
grzesiek51114
grzesiek51114
  • Rejestracja:ponad 11 lat
  • Ostatnio:ponad 4 lata
  • Postów:2442
0

Czyli tak jak się w sumie domyślałem. :) I te serwisy plus to co jest z ORM'a to właśnie nasz model. :)

edytowany 1x, ostatnio: grzesiek51114
T2
  • Rejestracja:ponad 10 lat
  • Ostatnio:7 miesięcy
  • Postów:194
3

Również podpisuję się pod słowami @DibbyDum
Serwis powinien zwracać IList.Serwis powinien zawierać logikę biznesową a nie operować na pojedynczych tabelach (Wtedy był by tylko nakładą na EF).
Zazwyczaj tworzę kilka mniejszych serwisów gdzie każdy z nich jest odpowiedzialny za konkretną cześć logiki biznesowej.

  • Skoro już masz luźne wiązanie śmiało możesz wstrzyknąć tą zależność za pomocą konstruktora abyś nie tworzył instancji w ramach ViewModelu.
  • Ja jeszcze tworze sobie Interfejs dla ViewModelu. Wykorzystuje ten interfejs w widoku w ramach DesignInstance. Ułatwia mi to tworzenie kodu.
edytowany 1x, ostatnio: teo215
grzesiek51114
grzesiek51114
Wtedy był by tylko nakładą na EF hehe kiedyś zapętliłem się z Viewmodelem tak, że przesyłałem do niego kontrolki, przez co zrobiła się z tego nakładka na code-behind. Szybko się pokapowałem, że coś jest nie tak ale śmiesznie to wyglądało :)
grzesiek51114
grzesiek51114
  • Rejestracja:ponad 11 lat
  • Ostatnio:ponad 4 lata
  • Postów:2442
0
teo215 napisał(a):
  • Skoro już masz luźne wiązanie śmiało możesz wstrzyknąć tą zależność za pomocą konstruktora abyś nie tworzył instancji w ramach ViewModelu.

Rozwiń jak możesz. Teraz mam właśnie instancję, nazwijmy to IService w VM.

Kliknij, aby dodać treść...

Pomoc 1.18.8

Typografia

Edytor obsługuje składnie Markdown, w której pojedynczy akcent *kursywa* oraz _kursywa_ to pochylenie. Z kolei podwójny akcent **pogrubienie** oraz __pogrubienie__ to pogrubienie. Dodanie znaczników ~~strike~~ to przekreślenie.

Możesz dodać formatowanie komendami , , oraz .

Ponieważ dekoracja podkreślenia jest przeznaczona na linki, markdown nie zawiera specjalnej składni dla podkreślenia. Dlatego by dodać podkreślenie, użyj <u>underline</u>.

Komendy formatujące reagują na skróty klawiszowe: Ctrl+B, Ctrl+I, Ctrl+U oraz Ctrl+S.

Linki

By dodać link w edytorze użyj komendy lub użyj składni [title](link). URL umieszczony w linku lub nawet URL umieszczony bezpośrednio w tekście będzie aktywny i klikalny.

Jeżeli chcesz, możesz samodzielnie dodać link: <a href="link">title</a>.

Wewnętrzne odnośniki

Możesz umieścić odnośnik do wewnętrznej podstrony, używając następującej składni: [[Delphi/Kompendium]] lub [[Delphi/Kompendium|kliknij, aby przejść do kompendium]]. Odnośniki mogą prowadzić do Forum 4programmers.net lub np. do Kompendium.

Wspomnienia użytkowników

By wspomnieć użytkownika forum, wpisz w formularzu znak @. Zobaczysz okienko samouzupełniające nazwy użytkowników. Samouzupełnienie dobierze odpowiedni format wspomnienia, zależnie od tego czy w nazwie użytkownika znajduje się spacja.

Znaczniki HTML

Dozwolone jest używanie niektórych znaczników HTML: <a>, <b>, <i>, <kbd>, <del>, <strong>, <dfn>, <pre>, <blockquote>, <hr/>, <sub>, <sup> oraz <img/>.

Skróty klawiszowe

Dodaj kombinację klawiszy komendą notacji klawiszy lub skrótem klawiszowym Alt+K.

Reprezentuj kombinacje klawiszowe używając taga <kbd>. Oddziel od siebie klawisze znakiem plus, np <kbd>Alt+Tab</kbd>.

Indeks górny oraz dolny

Przykład: wpisując H<sub>2</sub>O i m<sup>2</sup> otrzymasz: H2O i m2.

Składnia Tex

By precyzyjnie wyrazić działanie matematyczne, użyj składni Tex.

<tex>arcctg(x) = argtan(\frac{1}{x}) = arcsin(\frac{1}{\sqrt{1+x^2}})</tex>

Kod źródłowy

Krótkie fragmenty kodu

Wszelkie jednolinijkowe instrukcje języka programowania powinny być zawarte pomiędzy obróconymi apostrofami: `kod instrukcji` lub ``console.log(`string`);``.

Kod wielolinijkowy

Dodaj fragment kodu komendą . Fragmenty kodu zajmujące całą lub więcej linijek powinny być umieszczone w wielolinijkowym fragmencie kodu. Znaczniki ``` lub ~~~ umożliwiają kolorowanie różnych języków programowania. Możemy nadać nazwę języka programowania używając auto-uzupełnienia, kod został pokolorowany używając konkretnych ustawień kolorowania składni:

```javascript
document.write('Hello World');
```

Możesz zaznaczyć również już wklejony kod w edytorze, i użyć komendy  by zamienić go w kod. Użyj kombinacji Ctrl+`, by dodać fragment kodu bez oznaczników języka.

Tabelki

Dodaj przykładową tabelkę używając komendy . Przykładowa tabelka składa się z dwóch kolumn, nagłówka i jednego wiersza.

Wygeneruj tabelkę na podstawie szablonu. Oddziel komórki separatorem ; lub |, a następnie zaznacz szablonu.

nazwisko;dziedzina;odkrycie
Pitagoras;mathematics;Pythagorean Theorem
Albert Einstein;physics;General Relativity
Marie Curie, Pierre Curie;chemistry;Radium, Polonium

Użyj komendy by zamienić zaznaczony szablon na tabelkę Markdown.

Lista uporządkowana i nieuporządkowana

Możliwe jest tworzenie listy numerowanych oraz wypunktowanych. Wystarczy, że pierwszym znakiem linii będzie * lub - dla listy nieuporządkowanej oraz 1. dla listy uporządkowanej.

Użyj komendy by dodać listę uporządkowaną.

1. Lista numerowana
2. Lista numerowana

Użyj komendy by dodać listę nieuporządkowaną.

* Lista wypunktowana
* Lista wypunktowana
** Lista wypunktowana (drugi poziom)

Składnia Markdown

Edytor obsługuje składnię Markdown, która składa się ze znaków specjalnych. Dostępne komendy, jak formatowanie , dodanie tabelki lub fragmentu kodu są w pewnym sensie świadome otaczającej jej składni, i postarają się unikać uszkodzenia jej.

Dla przykładu, używając tylko dostępnych komend, nie możemy dodać formatowania pogrubienia do kodu wielolinijkowego, albo dodać listy do tabelki - mogłoby to doprowadzić do uszkodzenia składni.

W pewnych odosobnionych przypadkach brak nowej linii przed elementami markdown również mógłby uszkodzić składnie, dlatego edytor dodaje brakujące nowe linie. Dla przykładu, dodanie formatowania pochylenia zaraz po tabelce, mogłoby zostać błędne zinterpretowane, więc edytor doda oddzielającą nową linię pomiędzy tabelką, a pochyleniem.

Skróty klawiszowe

Skróty formatujące, kiedy w edytorze znajduje się pojedynczy kursor, wstawiają sformatowany tekst przykładowy. Jeśli w edytorze znajduje się zaznaczenie (słowo, linijka, paragraf), wtedy zaznaczenie zostaje sformatowane.

  • Ctrl+B - dodaj pogrubienie lub pogrub zaznaczenie
  • Ctrl+I - dodaj pochylenie lub pochyl zaznaczenie
  • Ctrl+U - dodaj podkreślenie lub podkreśl zaznaczenie
  • Ctrl+S - dodaj przekreślenie lub przekreśl zaznaczenie

Notacja Klawiszy

  • Alt+K - dodaj notację klawiszy

Fragment kodu bez oznacznika

  • Alt+C - dodaj pusty fragment kodu

Skróty operujące na kodzie i linijkach:

  • Alt+L - zaznaczenie całej linii
  • Alt+, Alt+ - przeniesienie linijki w której znajduje się kursor w górę/dół.
  • Tab/⌘+] - dodaj wcięcie (wcięcie w prawo)
  • Shit+Tab/⌘+[ - usunięcie wcięcia (wycięcie w lewo)

Dodawanie postów:

  • Ctrl+Enter - dodaj post
  • ⌘+Enter - dodaj post (MacOS)