Ocena programu w c# - komunikacja TCP - MVVM

Ocena programu w c# - komunikacja TCP - MVVM
Pixello
  • Rejestracja:prawie 10 lat
  • Ostatnio:4 miesiące
  • Lokalizacja:Podkarpacie
  • Postów:448
0

Witam,

naskrobałem sobie coś takiego https://github.com/pixellos/TCPConnector i chciałbym się dowiedzieć czy:
-dobrze wykorzystuję wzorzec MVVM
-czy klasy są logicznie zbudowane, poprawnie
-coś o komentarzach, czy dobrze opisuję itp
-czy obiektowość jest dobra, pomyślałem, że zamiast kilku parametrów w metodach mogę zrobić klasę, która będzie je przechowywała i nimi zarządzała - dobre podejście?
I co wam tam przyjdzie do głowy :)

edytowany 1x, ostatnio: Pixello
0

w Visual Studio jest narzędzie: testy jednostkowe... zamiast szukać pomocy prawdopodobnie bezskutecznie może warto byłoby zapoznać się z tym narzędziem... jest też w książce J.Matulewski Helion VS-2013 opisane za jedyne chyba 30 PLN ebooka se kupić można...

Pixello
  • Rejestracja:prawie 10 lat
  • Ostatnio:4 miesiące
  • Lokalizacja:Podkarpacie
  • Postów:448
0

Ale mi nie chodzi o "Nie działa, weźcie napiszcie za mnie lepiej, bo tak", tylko raczej
"Napisałem, sprawdziłem, działa, postarałem się obsłużyć wyjątki, napisać najlepszy kod na miarę moich możliwości i proszę o powiedzenie co mógłbym zrobić lepiej" ;)
Bardziej niż o techniczne sprawy chodzi mi o strukturę kodu, czy dobrze podzieliłem projekt na MVVM, czy kod jest w miarę logicznie nazwany, zrozumiały itp :)

edytowany 1x, ostatnio: Pixello
katelx
  • Rejestracja:prawie 10 lat
  • Ostatnio:4 miesiące
  • Lokalizacja:Hong Kong
1

tak przeklikalam przez pare plikow, uwagi (skupie sie na wadach bo chyba to cie interesuje, nie?):

  • mieszasz rozne konwencje nazewnictwa, do poczytania https://msdn.microsoft.com/en-us/library/ms229042.aspx
  • masz polskie nazwy
  • klasy powinny byc sealed, pola powinny byc readonly, w niektorych miescach nawet const
  • nie znalazlam przydatnego komentarza, lepiej je w ogole wywal
  • 'zjadasz' wyjatki zamiast je obsluzyc
  • w ogole nie uzywasz Dispose()/using
  • upubliczniasz listenery/readery/itp, po co?

wpf'a nie znam wiec sie nie wypowiem

Pixello napisał(a):

pomyślałem, że zamiast kilku parametrów w metodach mogę zrobić klasę, która będzie je przechowywała i nimi zarządzała - dobre podejście?
I co wam tam przyjdzie do głowy :)
czasem tak, czesciej zdecydowanie nie bo tworzysz sobie globalny god object ktorego skomplikowanie i zabugowanie bedzie roslo wykladniczo w miare rozwoju projektu. postaraj sie minimalizowac ilosc parametrow i odpowiedzialnosc kazdej klasy/metody. jesli widzisz ze dana rzecz robi zbyt wiele rzeczy, to znak ze nalezy ja podzielic na mniejsze, bardziej wyspecjalizowane.

edit: dobra ksiazka do przeczytania dla ciebie - http://www.amazon.com/Essential-Edition-Microsoft-Windows-Development/dp/0321877586

edytowany 1x, ostatnio: katelx
Pixello
  • Rejestracja:prawie 10 lat
  • Ostatnio:4 miesiące
  • Lokalizacja:Podkarpacie
  • Postów:448
0

Dzięki za odpowiedź ;) Zaraz się zabieram do poprawiania. Książkę na razie muszę odłożyć sobie do listy "do przeczytania", jeszcze nie skończyłem poprzednich.

katelx
  • Rejestracja:prawie 10 lat
  • Ostatnio:4 miesiące
  • Lokalizacja:Hong Kong
0

nie ma za co. co do ksiazki ktora podalam to polecam na poczatek (jak juz znajdziesz czas), tak zeby poznac podstawy platformy, procentuje przy nauce innych rzeczy i ogolnie w codziennej praktyce.

Pixello
  • Rejestracja:prawie 10 lat
  • Ostatnio:4 miesiące
  • Lokalizacja:Podkarpacie
  • Postów:448
0

Jeszcze mam pytanko, może bardziej do działu newbie pasuje, ale jak w ifach sprawdzam wartość, która jest bool?, to lepiej

Kopiuj
IsConnected().Value.Equals(true);

czy

Kopiuj
IsConnected() == true

?

katelx
  • Rejestracja:prawie 10 lat
  • Ostatnio:4 miesiące
  • Lokalizacja:Hong Kong
1

wystarczy

Kopiuj
IsConnected() ?? false

np. if(IsConnected() ?? false)
{
//ok, polaczony, tu cos robimy
}

Kopiuj
https://msdn.microsoft.com/en-us/library/ms173224.aspx

edit: odpowiadajac na pytanie, 2 sposob jest lepszy (i dziala), ale porownywanie z true/false jest brzydkie, w przykladzie ktory ci dalam mozesz dodatkowo zdefiniowac co jest defaultem (w moim przypadku false)
edytowany 1x, ostatnio: katelx
Pixello
Będę raczej używał Value.Equals, bo zrównanie false z nullem, szczególnie w przypadku obiektów, to dla mnie dodatkowe wyjątki do obsłużenia, a tak wyłapię nieprawidłowości ifami, a wyjątki zostawię dla nieprzewidzianych sytuacji :)
katelx
@Pixello Value.Equals to kiepski pomysl. nie wiem o czym mowisz z tymi dodatkowymi wyjatkami do obsluzenia, jakis przyklad? polecam poczytac o typach nullable
Pixello
Próbowałem czytać zmienne nienullable, i wyskakiwał wyjątek, że referencja nie jest ustawiona na obiekt, ostatecznie poradziłem sobie w ten sposób, że najpierw sprawdzam, czy referencja != null, a potem wykonuje instrukcje. Np jak chciałem czytać TcpClient.IsConnected();
somekind
Od tego jest HasValue, a nie porównywanie z null.
somekind
Moderator
  • Rejestracja:około 17 lat
  • Ostatnio:około 9 godzin
  • Lokalizacja:Wrocław
3
  • Namespace Common_Files - nazwa bez sensu, bo przestrzenie nazw zawierają klasy, nie pliki.
  • Wiele klas w jednym pliku.
  • Klasa o nazwie: abstract_Connector - po co ten abstract_ w nazwie?
  • Metoda ChangeParametrs - to nie lepiej po prostu zrobić nowy obiekt, skoro ma robić coś innego?
  • W projekcie Common mieszasz "silnik" swojej aplikacji z funkcjami pomocniczymi dla GUI? To jest bałagan... Bałagan, który zawsze tworzy się w projektach o nazwie Common. :)
Pixello napisał(a):

Jeszcze mam pytanko, może bardziej do działu newbie pasuje, ale jak w ifach sprawdzam wartość, która jest bool?, to lepiej

Kopiuj
IsConnected().Value.Equals(true);

czy

Kopiuj
IsConnected() == true

?

Jeśli IsConnected() zwraca Nullable<bool>, to myślę, że lepiej napisać if(connected.HasValue && connected.Value) (oczywiście wydzielając wcześniej zmienną, a nie pisać jakieś spaghetti w ifie), bo od razu widać o co chodzi i dlaczego.
Pytanie zasadnicze, czemu IsConnected zwraca taki typ - przecież albo jest się połączonym, albo nie jest, po prostu nie ma trzeciej możliwości. A jak chcesz tworzyć trzeci możliwy stan, to powinieneś zrobić enum.

somekind
Mógłbyś skopiować te swoje pytania do posta? Przecież dotyczą one głównego tematu wątku, a komentarze są do offtopu.
Pixello
  • Rejestracja:prawie 10 lat
  • Ostatnio:4 miesiące
  • Lokalizacja:Podkarpacie
  • Postów:448
0

W zamyśle miało zwracać null, gdy chodź jeden z obiektów jest niezainicjowany, potem zmieniłem na zawsze false oprócz sytuacji gdzie jest wszystko ok. Rozumiem, że 1 plik - 1 klasa? Zaraz wyodrębnię metody pomocnicze dla GUI do innego pliku. To, że zmieniam parametry zamiast tworzenia nowego obiektu to jakiś wielki błąd?

dam1an
  • Rejestracja:prawie 12 lat
  • Ostatnio:prawie 3 lata
  • Lokalizacja:Warszawa
  • Postów:1589
0

-dobrze wykorzystuję wzorzec MVVM?

Nie.
Przekazujesz jakieś labele z VM do View, do tego wykorzystujesz ButtonClicked i TextChanged w code behind.
W code behind powinieneś co najwyżej ustawić DataContext i to powinien być koniec grzebania w tym pliku. Zamiast ButtonClicked możesz zbindować się do Command, zamiast TextChanged nie pamiętam do czego się bindowało, pogogluj.
W MVVM ViewModel powinien jedynie udostępniać dane dla widoku, nie sam je modyfikować. A w widoku wyświetlasz je poprzez bindowanie.

edytowany 1x, ostatnio: dam1an
somekind
Moderator
  • Rejestracja:około 17 lat
  • Ostatnio:około 9 godzin
  • Lokalizacja:Wrocław
1
Pixello napisał(a):

W zamyśle miało zwracać null, gdy chodź jeden z obiektów jest niezainicjowany

Czy gdybyś pracował w sklepie, i ktoś by Cię spytał, czy masz chleb, to odpowiedziałbyś: "sprzątaczka dzisiaj nie przyszła"?
Bo to mniej więcej robi ta metoda obecnie. Klienta klasy nie obchodzi jej wewnętrzna implementacja, co on sobie ma zrobić z informacją "obiekt nie działa" w sytuacji gdy pyta o stan połączenia? Po prostu inicjalizuj wszystkie obiekty i nie pisz cieknącego kodu.

BTW: http://lambda-the-ultimate.org/node/3186

Rozumiem, że 1 plik - 1 klasa?

No raczej.

Zaraz wyodrębnię metody pomocnicze dla GUI do innego pliku.

Nie pliku tylko klasy. I trzymaj ją w projekcie GUI, albo pomocniczym GUI, a nie z rdzeniem aplikacji.

To, że zmieniam parametry zamiast tworzenia nowego obiektu to jakiś wielki błąd?

W ten sposób wprowadzasz niepotrzebną mutowalność, której efektem będzie najprawdopodobniej więcej dziwnych błędów wynikającej z przypadkowej zmiany stanu obiektu. Kod trzeba pisać tak, żeby błędów było jak najmniej.
No i w tej sytuacji trudno to nazwać programowaniem obiektowym, bo dla Ciebie klasa to jedynie zbiór funkcji.

Pixello
  • Rejestracja:prawie 10 lat
  • Ostatnio:4 miesiące
  • Lokalizacja:Podkarpacie
  • Postów:448
0

Z grubsza zacząłem poprawiać kod względem tego co napisaliście, szukałem trochę informacji na blogach i SO i zauważyłem, że bardzo często zmienne prywatne zapisują _zmienna, czy to jest jakaś przyjęta konwencja u "lepszych" programistów?

somekind
Moderator
  • Rejestracja:około 17 lat
  • Ostatnio:około 9 godzin
  • Lokalizacja:Wrocław
1

Jest to dość często spotykana konwencja nazewnictwa pól klasy (zakładam, że to je nazwałeś "zmiennymi prywatnymi").

Pixello
  • Rejestracja:prawie 10 lat
  • Ostatnio:4 miesiące
  • Lokalizacja:Podkarpacie
  • Postów:448
0

Mnie się jeszcze mocno C trzyma, dlatego ciężko mi się przyzwyczaić do tej obiektowości, ale się staram. Używacie czegoś takiego jak PDSA developer units (prowadzący w tucie microsoftu używał)? Poczytałem trochę więcej, jak tworzyć prawdziwy MVVM w WPF, i troszkę się przeraziłem ile to dodatkowego kodu ;) Co sądzicie o bibliotekach typu MVVM light/Caliburn.Micro, spróbować jej użyć, czy na początek uczyć się "na piechotę"?

Pixello
  • Rejestracja:prawie 10 lat
  • Ostatnio:4 miesiące
  • Lokalizacja:Podkarpacie
  • Postów:448
0

Dobra, ogarnąłem mniej więcej te Bindingi, jutro zacznę przerabiać program :) Myślałem już, że nigdy się nie kapnę o co chodzi :D

Pixello
  • Rejestracja:prawie 10 lat
  • Ostatnio:4 miesiące
  • Lokalizacja:Podkarpacie
  • Postów:448
0

O takie coś chodziło? https://github.com/pixellos/TCPConnector/tree/MVVM

Wywaliłem cały code-behind z widoków, pobindowałem do ICommand oraz do właściwości, odświeżanie odbioru przez BackgroundWorkera, reszta kodu mały regress zaliczyła, ale będę dalej pracował nad nim.

@Edit
Poprawiłem kod, dodałem transmisję w obie strony, starałem się poprawić kod i zastosować wzorzec MVVM.
Mógłby ktoś zerknąć, czy jest lepiej według was?

https://github.com/pixellos/TCPConnector

edytowany 2x, ostatnio: Pixello
Pixello
  • Rejestracja:prawie 10 lat
  • Ostatnio:4 miesiące
  • Lokalizacja:Podkarpacie
  • Postów:448
0

user image

https://github.com/pixellos/TCPConnector

Tak mniej więcej teraz wygląda, nie wywala przy co drugiej akcji. Jakbyście mogli przejrzeć i powiedzieć o tym kilka słów byłbym wdzięczny :)

Chciałbym się dowiedzieć, czy jest lepiej, starałem się dobrze nazywać funkcje i zmienne, pisać kod czytelny itp.

dam1an
  • Rejestracja:prawie 12 lat
  • Ostatnio:prawie 3 lata
  • Lokalizacja:Warszawa
  • Postów:1589
1

Jeśli chodzi o wzorzec MVVM to teraz jest dużo lepiej. Wywal może tylko te nieużywane eventy w MainWindow.
:ES i :AYT? nie wiadomo co to jest,
Mogę też zasugerować przeniesienie defaultowych tekstów z textboxów np "127.0.0.1" do settingsów. Dzięki temu przy otwieraniu okna będziesz miał ostatnią wartość którą tam wpisałeś przy poprzednim uruchomieniu.

Degusto
  • Rejestracja:ponad 11 lat
  • Ostatnio:około rok
  • Lokalizacja:Piła
  • Postów:70
0

Zamiast:

Kopiuj
public bool IsStreamDescribed()
    {
        if (newStream == null)
        {
            return false;
        }
        else return true;
    }

Możesz napisać tak:

Kopiuj
public bool IsStreamDescribed()
    {
        return newStream != null;
    }

Zwróciłbym też uwagę na nazwy plików np. tutaj: https://github.com/pixellos/TCPConnector/blob/master/Common%20Files/IActionClass.cs
Nazwa pliku sugeruje, że to interfejs, a w środku jest klasa(z inną nazwą).

edytowany 1x, ostatnio: Degusto
Pixello
  • Rejestracja:prawie 10 lat
  • Ostatnio:4 miesiące
  • Lokalizacja:Podkarpacie
  • Postów:448
0

Kiedyś był tam interfejs i tak zostało :D Zaraz spróbuję coś z tym zrobić. Dzięki wam za odpowiedzi :).
Zmieniłem IsStreamDescribed na właściwość z akcesorem get.
@dam1an, mógłbyś rozwinąć? Nie rozumiem jak mógłbym przechowywać wartość z poprzedniego uruchomienia w właściwości.

dam1an
  • Rejestracja:prawie 12 lat
  • Ostatnio:prawie 3 lata
  • Lokalizacja:Warszawa
  • Postów:1589
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)