Umumnya objek domain memiliki properti yang dapat diwakili oleh tipe bawaan tetapi nilai validnya adalah subset dari nilai yang mungkin diwakili oleh tipe itu.
Dalam kasus ini, nilai dapat disimpan menggunakan tipe bawaan tetapi perlu untuk memastikan bahwa nilai selalu divalidasi pada titik masuk, jika tidak kita akhirnya akan bekerja dengan nilai yang tidak valid.
Salah satu cara untuk mengatasi ini adalah dengan menyimpan nilai sebagai kebiasaan struct
yang memiliki private readonly
bidang dukungan tunggal dari tipe bawaan dan yang konstruktornya memvalidasi nilai yang diberikan. Kami kemudian dapat selalu yakin hanya menggunakan nilai yang divalidasi dengan menggunakan struct
tipe ini .
Kami juga dapat menyediakan operator cor dari dan ke tipe bawaan yang mendasarinya sehingga nilai-nilai dapat masuk dan keluar dengan mulus sebagai jenis yang mendasarinya.
Ambil contoh situasi di mana kita perlu mewakili nama objek domain, dan nilai yang valid adalah string apa pun yang panjangnya antara 1 dan 255 karakter. Kami dapat mewakili ini menggunakan struct berikut:
public struct ValidatedName : IEquatable<ValidatedName>
{
private readonly string _value;
private ValidatedName(string name)
{
_value = name;
}
public static bool IsValid(string name)
{
return !String.IsNullOrEmpty(name) && name.Length <= 255;
}
public bool Equals(ValidatedName other)
{
return _value == other._value;
}
public override bool Equals(object obj)
{
if (obj is ValidatedName)
{
return Equals((ValidatedName)obj);
}
return false;
}
public static implicit operator string(ValidatedName x)
{
return x.ToString();
}
public static explicit operator ValidatedName(string x)
{
if (IsValid(x))
{
return new ValidatedName(x);
}
throw new InvalidCastException();
}
public static bool operator ==(ValidatedName x, ValidatedName y)
{
return x.Equals(y);
}
public static bool operator !=(ValidatedName x, ValidatedName y)
{
return !x.Equals(y);
}
public override int GetHashCode()
{
return _value.GetHashCode();
}
public override string ToString()
{
return _value;
}
}
Contoh ini memperlihatkan string
pemain karena implicit
hal ini tidak akan pernah gagal tetapi dari string
pemain explicit
karena hal ini akan menghasilkan nilai yang tidak valid, tetapi tentu saja ini bisa berupa implicit
atau explicit
.
Perhatikan juga bahwa seseorang hanya dapat menginisialisasi struct ini dengan cara pemain dari string
, tetapi orang dapat menguji apakah pemain tersebut akan gagal di muka menggunakan IsValid
static
metode ini.
Ini tampaknya menjadi pola yang baik untuk menegakkan validasi nilai domain yang dapat diwakili oleh tipe sederhana, tetapi saya tidak melihatnya sering atau disarankan dan saya tertarik mengapa.
Jadi pertanyaan saya adalah: apa yang Anda lihat sebagai keuntungan dan kerugian dari menggunakan pola ini, dan mengapa?
Jika Anda merasa ini adalah pola yang buruk, saya ingin memahami mengapa dan apa yang Anda rasakan adalah alternatif terbaik.
NB Saya awalnya mengajukan pertanyaan ini pada Stack Overflow tapi itu ditunda terutama berdasarkan opini (ironisnya subjektif sendiri) - mudah-mudahan bisa menikmati lebih banyak kesuksesan di sini.
Di atas adalah teks asli, di bawah beberapa pemikiran lagi, sebagian sebagai jawaban atas jawaban yang diterima di sana sebelum ditunda:
- Salah satu poin utama yang dibuat oleh jawaban adalah sekitar jumlah kode pelat ketel yang diperlukan untuk pola di atas, terutama ketika banyak jenis seperti itu diperlukan. Namun dalam mempertahankan pola, ini sebagian besar bisa secara otomatis menggunakan template dan sebenarnya bagi saya sepertinya tidak terlalu buruk, tapi itu hanya pendapat saya.
- Dari sudut pandang konseptual, apakah tidak aneh ketika bekerja dengan bahasa yang sangat diketik seperti C # untuk hanya menerapkan prinsip yang sangat diketik untuk nilai komposit, daripada memperluasnya ke nilai-nilai yang dapat diwakili oleh turunan dari tipe bawaan?
Jawaban:
Ini cukup umum dalam bahasa gaya-ML seperti Standar ML / OCaml / F # / Haskell di mana lebih mudah untuk membuat jenis pembungkus. Ini memberi Anda dua manfaat:
ValidatedName
pernah berisi nilai yang tidak valid, Anda tahu kesalahan ada di dalamIsValid
metode.Jika Anda mendapatkan
IsValid
metode yang benar, Anda memiliki jaminan bahwa fungsi apa pun yang menerimaValidatedName
sebenarnya menerima nama yang divalidasi.Jika Anda perlu melakukan manipulasi string, Anda dapat menambahkan metode publik yang menerima fungsi yang mengambil String (nilai
ValidatedName
) dan mengembalikan String (nilai baru) dan memvalidasi hasil penerapan fungsi. Itu menghilangkan pelat untuk mendapatkan nilai String yang mendasarinya dan membungkusnya kembali.Penggunaan terkait untuk nilai pembungkus adalah untuk melacak asalnya. Misalnya API OS berbasis C terkadang memberikan pegangan untuk sumber daya sebagai bilangan bulat. Anda dapat membungkus OS API untuk menggunakan
Handle
struktur dan hanya menyediakan akses ke konstruktor ke bagian kode tersebut. Jika kode yang menghasilkan hurufHandle
s benar, maka hanya pegangan yang valid yang akan digunakan.sumber
Baik :
ValidatedString
membuatnya lebih jelas tentang semantik panggilan.Buruk :
IsValid
sebelum para pemeran agak tidak menyenangkan.ValidatedString
tidak valid / divalidasi.Saya telah melihat hal semacam ini lebih sering dengan
User
danAuthenticatedUser
hal-hal semacam itu, di mana objeknya benar-benar berubah. Ini bisa menjadi pendekatan yang baik, meskipun tampaknya tidak pada tempatnya di C #.sumber
Cara Anda cukup berat dan intensif. Saya biasanya mendefinisikan entitas domain seperti:
Di konstruktor entitas, validasi dipicu menggunakan FluentValidation.NET, untuk memastikan Anda tidak dapat membuat entitas dengan keadaan tidak valid. Perhatikan bahwa semua properti hanya dapat dibaca - Anda hanya dapat mengaturnya melalui konstruktor atau operasi domain khusus.
Validasi entitas ini adalah kelas yang terpisah:
Validator ini juga dapat dengan mudah digunakan kembali, dan Anda menulis lebih sedikit kode boilerplate. Dan keuntungan lain adalah itu bisa dibaca.
sumber
Saya suka pendekatan ini untuk tipe nilai. Konsepnya bagus, tapi saya punya beberapa saran / keluhan tentang implementasinya.
Casting : Saya tidak suka menggunakan casting dalam hal ini. Pemeran dari-string eksplisit tidak menjadi masalah, tetapi tidak ada banyak perbedaan antara
(ValidatedName)nameValue
dan yang baruValidatedName(nameValue)
. Jadi sepertinya tidak perlu. Cast ke-string implisit adalah masalah terburuk. Saya pikir bahwa mendapatkan nilai string yang sebenarnya harus lebih eksplisit, karena mungkin secara tidak sengaja ditugaskan ke string dan kompiler tidak akan memperingatkan Anda tentang kemungkinan "kehilangan presisi". Kehilangan presisi seperti ini harus eksplisit.ToString : Saya lebih suka menggunakan
ToString
overload hanya untuk keperluan debugging. Dan saya tidak berpikir mengembalikan nilai mentah karena itu adalah ide yang bagus. Ini adalah masalah yang sama dengan konversi implisit ke-string. Mendapatkan nilai internal harus operasi eksplisit. Saya percaya Anda mencoba untuk membuat struktur berperilaku sebagai string normal ke kode luar, tetapi saya pikir dalam melakukannya, Anda kehilangan beberapa nilai yang Anda dapatkan dari menerapkan jenis ini.Equals dan GetHashCode : Structs menggunakan kesetaraan struktural secara default. Jadi Anda
Equals
danGetHashCode
menduplikasi perilaku default ini. Anda dapat menghapusnya dan itu akan menjadi hal yang hampir sama.sumber