
lazy loading در ORMهای مختلف به روشهای مختلفی انجام میشود. مثلاً بعضیها از روش Inheritance استفاده میکنند. به این معنی که کلاس Business Entity شما بایستی از یک کلاس مخصوص ارثبری کند. این کلاس مخصوص خودش دارای متودهایی برای انجام عملیات lazy loading است. بعضی ORMهای دیگر از روش interface استفاده میکنند. در این روش کلاس Business Entity شما باید interface خاصی را implement کند. اما روش NHibernate استفاده از design pattern جالبی به اسم proxy است. در این روش NHiberntae به هنگام شروع به کار و ساختن session، به ازای هر کدام از کلاسهای Business Entity شما یک کلاس مخصوص میسازد و هر جا که برنامه شما نیاز به استفاده از آن کلاس پیدا کرد، NH آن کلاس proxy را تحویلش میدهد. این قضیه از دید برنامه مخفی است. یعنی برنامه فکر میکند همچنان در حال استفاده از کلاس خودش است. به این ترتیب NH این فرصت را پیدا میکند که درخواستهای دسترسی به propertyهای مختلف یک object را به جای آن که مستقیماً به دیتابیس بفرستد به آن کلاس proxy بفرستد. آن کلاس proxy هم خودش ترتیب lazy loading را میدهد. برای آن که یک کلاس proxy بتواند به وظیفهاش عمل کند باید کلیه propertyها و methodهای کلاس مبدا به صورت virtual تعریف شده باشد تا proxy factory بتواند همه آنها را در کلاس proxy ارث برده شده، override کند. NH برای تولید proxy از کتابخانههای مستقلی مثل Castle Dynamic Proxy استفاده میکند.
lazy loading در NH در سه سطح عمل میکند: الف- در سطح object. یعنی تا زمانی که واقعاً نیاز به خواندن اطلاعات objectهای دیگری که به صورت property تعریف شدهاند نباشد، اطلاعات آن از دیتابیس خوانده نخواهد شد. NH فعلا lazy loading در سطح propertyهای غیر آبجکتی را پشتیبانی نمیکند. ۲- در سطح کلی collection: تا زمانی که نیاز به اطلاعات یکی از اعضای آن collection نباشد، اطلات آن از دیتابیس بازیافت نخواهد شد. در این روش همیشه کل collection با هم خوانده میشوند. ۳- در سطح اعضای collection: درست مشابه روش قبل است با این تفاوت که lazy loading در سطح تک تک اعضای collection انجام میشود. مثلا وقتی که به عنصر چهارم یک collection نیاز هست فقط query همین یک رکورد به دیتابیس ارسال میشود نه دیگر اعضای collection. این حالت lazy loading به طور پیشفرض به خاطر جلوگیری از کندی سرعت غیر فعال است.
تنها مشکلی که با lazy loading وجود دارد، مشکل آن با session است. در مورد مثال بالا اگر obj1 در حین فعال بودن یک session از دیتابیس فراخوانی شود و آن session بدون فراخوانی obj1.obj2 بسته شود و فراخوانی پس از بسته شدن session انجام شود، خطا رخ میدهد چون آن object دیگر detach شده است. غلبه بر این مشکل خیلی سخت نیست. فقط کافیست Close شدن session تا حد امکان به تعویق انداخته شود، مثلا به آخرین خط متود یا try-catch منتقل شود. و اگر این امکانپذیر نبود قبل از بسته شدن session، آن property مورد دار به صورت الکی فراخوانی شود تا برنامه یک بار مجبور شود به دیتابیس مراجعه کرده و اطلاعات خود را به روز کند. مشکل sessionها بیشتر در برنامههای web مشکلساز هستند چون برنامهنویس مجبور است با هر بار Postback برنامه، session را ببندد.
دیدگاهها
مطلب خوبی بود.
من نمی دونم چرا هرچی سعی میکنم که قبول کنم که NH ارزش کار کردن داره نمیتونم.
تا وقتی EF هست واقعا چه نیازی به NH هست.البته بحث اپن سورس بودنش جدا ولی آیا واقعا ارزشش رو داره؟
مشکلات NH بعضی وقتا خیلی اذیت می کنن.
مثلا پیاده سازی NH با VB.Net یه خان خیلی سنگینه و کار زیادی می بره.
جناب برای بخش کامنت ها این استایل رو اضاقه کنید تا مشکل
RTL
حل بشه :
style="direction:rtl;"
@Anonymous: اولا که بابت حل مشکل RTL کامنت خیلی ممنون. راهنمایی خیلی با ارزشی بود.
ثانیا در مورد NH و EF: به نظر کسی میتونه راجع به این دو و بهتر یا بدتر بودن یکی از اونها نظر بده که با هر دو تا کار کرده باشه و به هر کدام یک تسلط نسبی داشته باشد. من خودم تا حالا یک بار هم EF کار نکرده و حتی یک نمونه کد هم با آن ندیدهام. دلیل شخصی خود من برای NH این است که تا حالا سه سال است که از دور و نزدیک با NH کار کردهام و این موضوع علاوه بر Open Source بودن آن انگیزه خوبی برای ادامه کار با آن به من میدهد.