راه حل های مسابقه شماره Quera ۲۱

1105

سلام دوباره! اولا کمال عذر خواهی را برای تاخیر در نوشتن راه حل‌ها به جا می‌آورم! به هر حال کنکوری بودن هم معایبی دارد! امید وارم کانتست بیشتر از اندازه ی کافی جذاب بوده باشد!

  • بولوف الکی!

    برای حساب کردن تعداد غلط‌ها کافی است اسپیس‌ها را کنار بگذاریم و حروف موجود در هر پرسش را با جواب آن چک کنیم(هر کدام آنها در یک خط نوشته می‌شوند). اگر مغایرت داشت جواب را یک واحد زیاد کنید. پیچیدگی زمانی O(Si)\mathcal{O}{(\sum |S_i|)} است.

     

    کد سی پلاس پلاس۱و کد سی پلاس پلاس ۲

  • خیلی قهوه ای یا باج یا خوشخوار!

    می‌توانیم ببینیم که در هر حرکتی که انجام می‌دهیم زوجیت جمع هر دو نوع کتاب ناورداست. پس برای این که دقیقا یک کتاب از یک نوع داشته باشیم لازم است که جمع تعداد دو نوع دیگر زوج و جمع تعداد این نوع با هر یک از دو نوع دیگر فرد باشد. مثلا اگر بخواهیم دقیقا یک کتاب از نوع یک داشته باشیم و زوجیت تعداد کتاب‌های هر نوع باید به صورت (0,1,1) یا (1,0,0) باشد.

    حال اگر زوجیت هر سه دسته به صورت‌های بالا باشد آنگاه می‌توانیم آنقد دو تا دوتا از یک نوع بفروشیم که تعداد هر نوع دقیقا برابر اعداد بالا شوند و در هر دو حالت بدیهی است که می‌توانیم به دقیقا یکی از نوع یک برسیم.  پیچیدگی زمانی O(1)\mathcal{O}{(1)} است.

    کد سی پلاس پلاس

  • هر که بامش بیش درسش بیشتر!

    می‌توان هر عددی را با حداکثر دو حرکت به هر جای جایگشت رساند. ابتدا اعداد از یک تا n2\frac{n}{2} را هر کدام حداکثر با دو حرکت سر جای خودشان می‌رسانیم. تا به حال حداکثر nn حرکت انجام داده ایم و n2\frac{n}{2} تا عدد اول سر جای خودشان حضور دارند. پس اکنون کافی است حداکثر n2+1\frac{n}{2}+1 تای آخر را درست کنیم. چون حداکثر فاصله n2+1\frac{n}{2}+1 است می‌توان از n2+1\frac{n}{2}+1 تاn1n-1 را هر کدام حداکثر با یک حرکت به جای خودشان برد و سپس عدد nn ام خود به خود سر جای خودش قرار می‌گیرد. حداکثر تعداد حرکت‌ها 3×n2\frac{3\times{n}}{2} است. پیچیدگی زمانی جواب O(n)\mathcal{O}{(n)} است.

    کد سی پلاس پلاس

  • تقلب رمزی

    به طور کلی بررسی کردن هر حرف به صورت جدا انجام می‌شود و هیچ حرفی روی حرف دیگر تاثیر نمی‌گذارد. می‌خواهیم بفهمیم آیا می‌توانیم از حرف cc شروع کنیم یا خیر؟

    فرض می‌کنیم که می‌خواهیم بدانیم اگر با حرفی خاص مانند cc شروع کنیم در i عملیات اول به چه حرفی می‌رسیم. برای این منظور  fcif_{c_i} را تعریف می‌کنیم. به سادگی معلوم است که fcif_{c_i} می‌شود convert(fci1,i)convert( f_{c_{i - 1}}, i)  که convert(a,i) convert(a, i) برابر حرفی است که با انجام عملیات iiام روی حرف aa بدست می‌آید.

    به همین ترتیب برای انجام برعکس عملیات‌ها gcig_{c_i} را تعریف می‌کنیم که یعنی اگر ii عملیات آخر را داشته باشیم و از حرف cc شروع کنیم به چه حرفی می‌رسیم. بطور مشابه داریم

    gci=fconvert(c,i)i+1 g_{c_i} = f_{convert(c,i)_{i+1}}  

    حال برای انجام برعکس عملیات‌ها نیز توابعی به نام ffff و gggg مانند ff و gg تعریف می‌کنیم و سپس می‌توانیم بفهمیم که آیا یک حرف خاص مثلا cc با حذف عملیات‌های ll تا rr یکه است یا نه. معلوم است که با حذف عملیات های ll تا rr حرف cc به حرف cc=g(fc(l1))r+1cc = g_{{(f_{c_{(l-1)}})}_{r+1}} تبدیل می‌شود. اگر حرف cccc با انجام برعکس عملیات ها و استفاده از ffff وgggg به همان cc رسیدیم پس cc در این کوئری یکه است. 

    از طرفی چون عملیات‌ها روی یک حرف دقیقا یک خروجی دارند (حروف مستقل از هم هستند) پس کافی است تا تعداد حروف یکه را بیابیم بدین ترتیب تعداد کلمات یکه هم بدست می‌آید و جواب هر کوئری برابر tanst^{ans} می‌شود که ansans تعداد حرف‌های یکه است و t طول داده شده در کوئری است. پیچیدگی زمانی O(n+q×K)\mathcal{O}{(n+q \times K)} است کهKK تعداد حروف الفباست.

    کد سی پلاس پلاس

  • درخت تقلب

    این مساله را به صورت آفلاین حل می‌کنیم
    یعنی ابتدا تمام کوئری‌هارا دریافت می‌کنیم و بعدا جواب هر یک را می‌دهیم.

    ابتدا تمام کوئری‌های نوع ۱ را دریافت می‌کنیم و راس‌ها را به درخت اضافه می‌کنیم. از راس ۱ دی اف اس میزنیم و روی هر راس زمان وارد شدن تابع dfsdfs به آن را می‌نویسیم در این صورت روی راس uu عدد sus_u نوشته خواهد شد.

    ما برای هر کوئری نوع دوم مانند v dv\ d بیشترین ارتفاع hh را در درخت می‌خواهیم بطوریکه راسی در آن ارتفاع باشد که فاصله اش از vv کمتر یا مساوی dd باشد. می‌دانیم اگر hvx1h_v\leq x-1 باشد و راسی در ارتفاع xx باشد که فاصله اش از vv کمتر یا مساوی dd باشد راسی در ارتفاع x1x - 1 نیز با خاصیت مذکور وجود دارد. پس می‌توانیم با جستجوی دودویی روی ارتفاع h را پیدا کنیم.

    حال می‌خواهیم بدانیم در ارتفاع hvxh_v\leq x نزدیک ترین راس به vv چه راسی است. دو تا از نزدیک ترین راس‌ها به v آن دوتاییست که در ارتفاع hh عدد منتاظرشان در آرایه ss  به svs_v نزدیک تر است. واضح است که در ارتفاع h فقط راس‌هایی را می‌بینیم که قبل از این کوئری وصل شده اند. و می‌توانیم این دو راس را با جست جوی دودویی دیگری پیدا کنیم.

    حال برای پیدا کردن فاصله دو راس در درخت جد مشترک (lca)(lca) آنها را پیدا می‌کنیم به این ترتیب فاصله دو راس نیز به راحتی بدست می‌آید. پیچیدگی زمانی این جواب O(n×log2(n)2)\mathcal{O}{(n\times log_2{(n)} ^ 2)} است. همین راه پیاده سازی سریع تر هم دارد.

     

  • منبع موثق

    فرض کنیم دو تابع داریم که مقادیر زیر را برای ما حساب می‌کنند :

    f(A,d,k,x,n)=0in10jx+i×k(A+d×j)f(A, d, k, x, n)=\sum_{0\leq i\leq {n-1}} \sum_{0\leq j\leq {x+i\times k}} (A+d\times j) g(A,d,k,x,n)=0in1(i1)×(x+i×k)ji×(x+i×k)(A+d×j)g(A, d, k, x, n)=\sum_{0\leq i\leq {n-1}} \sum_{{(i-1)}\times {(x+i\times k)}\leq j\leq i\times {(x+i\times k)}} (A+d\times j)

    می‌دانیم که ممکن است تعدادی از سطرهای ابتدایی جدول باشند  که فقط اعداد مربوط به دنباله حسابی دوم (دنباله حسابی ستون‌ها) روی خانه‌‌های آن‌ها نوشته می‌شود. به طرز مشابه تعدادی از سطرهای آخر هستند که فقط اعداد دنباله حسابی اول روی آن‌ها نوشته می‌شود. می‌توان مجموع آن سطر‌هارا حساب کرد و جدول را بدون آنها در نظر گرفت.
    حال p(i)p(i) را بدین گونه تعریف می‌کنیم : p(i)p(i) برابر کوچکترین jj ای است که i×b+a<j×d+ci \times b + a < j\times d + c معلوم است که p(i)+c=p(i+a)p(i) + c = p(i + a).
    حال مجموع خانه‌های p(i+k×a)p(i + k \times a) تا mm ام سطر i+k×ai + k\times a (به ازای هر kk ) را می‌توان با تابع ff حساب کرد
    همچین مجموع خانه‌های 1 تا p(i×k+a)p(i\times k + a) ام سطر i+k×ai + k\times a (به ازای هر kk ) را می‌توان با تابع gg حساب کرد.
    پس می‌توانیم مجموع تمام سطرهایی که باقیمانده اندیسشان بر aa برابر است را یک جا حساب کنیم.
    بنابر این می‌توان مجموع جدول را از O(a)\mathcal{O(a)} حساب کرد.
    یافتن فرمول برای دو تابع ff و gg نیز با کمی اطلاعات درباره ی دنباله‌های حسابی ساده می‌شود.

  • چمران

    tt را در نظر بگیریم کمترین زمانی که می‌توان چنین گشتی را طی کرد. اگر چنین عددی وجود نداشته باشد جواب "NO""NO" و گرنه "YES""YES" است. می‌خواهیم کوتاه ترین گشت ممکن را بررسی کنیم. برای سادگی راه حل فرض می‌کنیم شروع مسیر در جواب به سمت خانه‌ی nn بوده است(یک بار هم فرض می‌کنیم به سمت خانه‌ی 0 بوده است) و سوال را حل می‌کنیم.

    دنباله‌ی SlSl و SrSr را در نظر بگیرید. SlSl دنباله‌ای از اعداد ۱ تا nn است که تمام جاهایی است که در طول مسیر فرد مورد نظر به سمت خانه‌ی nn درحال حرکت بوده و در خانه‌ی SliSl_i  به سمت خانه‌ی صفر دور می‌زند( تغییر جهت می‌دهد). دورها به ترتیب زمان در SlSl ظاهر می‌شوند به این معنا که اگر دوری زودتر از دور دیگری انجام شود در دنباله در خانه‌ای با اندیس کمتر ظاهر می‌شود.SrSr هم مانند SlSl است با این تفاوت که دور‌ها از سمت خانه‌ی صفر به سمت خانه‌ی nn است.  اولین خانه ی مسیر را هم به SlSl می‌افزاییم چرا که به سمت خانه‌ی nn است. مقصد نیز به آن طرفی می‌افزاییم که به سمت آن تمام می‌شود.

    ابتدا ثابت می‌کنیم اگر tt مینیموم باشد SlSl دنباله ای اکیدا نزولی و SrSr اکیدا صعودی است. بدون از دست دادن کلیت روی SlSl بحث می‌کنیم (برای SrSr به قرینه ثابت می‌شود). فرض کنید SlSl نزولی اکید نباشد. شماره اولین دوری این ترتیب را به هم زده است را i در نظر بگیرید.  پس اکنون SliSli+1Sl_i \leq Sl_{i+1}. قسمتی از مسیر به این صورت است:

    Sli1Sl_{i-1} ا—> SriSr_i ا—->  SliSl_i ا—> Sri+1Sr_{i+1} ا—> Sli+1Sl_{i+1}.

    که جایگزین مناسب تر آن این مسیر است:  

    Sli1Sl_{i-1} ا—> min(Sri,Sri+1)\min(Sr_i,Sr_{i+1}) ا—> Sli+1Sl_{i+1}

    هم مسیر کوتاه تریست و خانه‌ها شانس بیشتری برای خواب ماندن دارند.

    چون tt کوتاه ترین گشت بود پس SlSl نزولی اکید و SrSr صعودی اکید است.tt حداکثر بیشترین عدد آرایه aa است. چون اولین خانه را فقط یک بار و در همان شروع می‌بینیم. می‌خواهیم ثابت کنیم اندازه‌ی SlSl حداکثر برابر maxai×2\sqrt{maxai}\times 2 است.

    این نامساوی را در نظر بگیرید:

    0ip(SliSri)<t\sum_{0\leq i\leq p} (|Sl_i - Sr_i|) < t

    چون SlSl نزولی و SrSr صعودی است پس هیچ کدام از SliSri|Sl_i - Sr_i|ها با هم برابر نیست پس اگر بخواهیم pp حداکثر باشد باید جمله‌های این جمع را حداقل کنیم:

    0i2×tit\sum_{0\leq i\leq 2\times \sqrt{t}} i \simeq t

    پسpmaxaip \leq \sqrt{maxai} یعنی تعداد دورها حداکثر maxai×4\sqrt{maxai} \times 4 است.

    حال کافی است دنباله‌های SlSl و  SrSr را تعیین کنیم. ابتدا برای خانه‌ی iiام bib_i را در نظر می‌گیریم برابر آخرین زمانی که تا الآن دیده شده است.

    سپس tt را فیکس می‌کنیم.  فرض می‌کنیم این کوتاه ترین زمانی است که می‌توان این گشت را طی کرد. همه ی درایه‌های bb منفی بینهایت است(به این معنا که تا به حال دیده نشده است). همانطور که از ابتدای راه حل اشاره کردیم از خانه‌ی nn شروع می‌کنیم.

    هر بار که می‌خواهیم تا جایی برویم و دور بزنیم محل دور را نزدیک ترین جایی که می‌توانیم انتخاب می‌کنیم.

    نزدیک ترین جایی را انتخاب می‌کنیم که خانه‌های از آن به بعد(از آن تا 0 یا از آن تا nn) تا آخر tt ثانیه خواب باشند یا به عبارت دیگر اینکه به ازای همه iهایی که از محل دور به بعد وجود دارد: tbiait-b_i \leq a_i

    و لازم به ذکر است که حتما باید تا آنجا برویم و این بهترین گشت برای کوتاه ترین زمان ممکن است.

    tmaxait \leq maxai و تعداد دورها حداکثر maxai×4\sqrt{maxai} \times 4 است و می‌توانیم بهترین مکان هر دور را با جست و جوی دو دویی پیدا کنیم. پس اردر زمانی مساله O(maxai×maxai×log2(n))\mathcal{O}{(maxai \times \sqrt{maxai} \times \log_2 (n))} است.

بازم سوالی بود بپرسید! ان‌شاءالله جواب میدیم.

التماس دعا!

خداحافظتون 🙂

آموزش برنامه نویسی با کوئرا کالج
کوئرا بلاگ

اشتراک در
اطلاع از
guest


4 دیدگاه‌
قدیمی‌ترین
تازه‌ترین بیشترین واکنش
بازخورد (Feedback) های اینلاین
View all comments
ناشناخته
ناشناخته
7 سال قبل

اگه میشه کدارو هم بزارین
بالاخره نحوه پیاده سازی هم مهمه

ناشناخته
ناشناخته
7 سال قبل
پاسخ به  کوئرا بلاگ

متشکر 🙂

پشمک
پشمک
7 سال قبل

کانتست جدید بزارین دیگه حوصلمون سر رفت یه ذره از کدفورسز یا بگیرید:(