ในปัญหานี้ เราได้รับอาร์เรย์ arr[] ซึ่งประกอบด้วยตัวเลข N และค่าจำนวนเต็ม x งานของเราคือ สร้างโปรแกรมเพื่อค้นหาองค์ประกอบแรกที่มากกว่าหรือเท่ากับ X ในผลรวมนำหน้าของตัวเลข N โดยใช้ Binary Lifting .
ผลรวมคำนำหน้า ขององค์ประกอบของอาร์เรย์คืออาร์เรย์ที่แต่ละองค์ประกอบเป็นผลรวมขององค์ประกอบทั้งหมดจนถึงดัชนีนั้นในอาร์เรย์เริ่มต้น
ตัวอย่าง − array[] ={5, 2, 9, 4, 1}
คำนำหน้าSumArray[] ={5, 7, 16, 20, 21}
มาดูตัวอย่างเพื่อทำความเข้าใจปัญหากัน
Input: arr[] = {5, 2, 9, 4, 1}, X = 19 Output: 3
แนวทางการแก้ปัญหา
ในที่นี้ เราจะมาแก้ปัญหาโดยใช้แนวคิด การยกแบบไบนารี . Binary Lifting เป็นการเพิ่มค่าของตัวเลขที่กำหนดโดยยกกำลัง 2 (ทำได้โดยการพลิกบิต) ตั้งแต่ 0 ถึง N
เราจะพิจารณาแนวคิดที่คล้ายกับการยกไบนารีทรี ซึ่งเราจะพบค่าเริ่มต้นของดัชนี 'P' สิ่งนี้เพิ่มขึ้นโดยการพลิกบิตเพื่อให้แน่ใจว่าค่าไม่มากกว่า X ตอนนี้เราจะพิจารณาการเพิ่มขึ้นพร้อมกับตำแหน่ง 'P' นี้
สำหรับสิ่งนี้ เราจะเริ่มการพลิกบิตของตัวเลข โดยที่การพลิกบิตที่ i จะไม่ทำให้ผลรวมมากกว่า X ตอนนี้ เรามีสองกรณีตามค่าของ 'P' −
ตำแหน่งเป้าหมายอยู่ระหว่าง 'ตำแหน่ง + 2^i ' และ 'ตำแหน่ง + 2^(i+1) ' โดยที่การยก ith เพิ่มมูลค่า หรือตำแหน่งเป้าหมายอยู่ระหว่าง 'ตำแหน่ง ' และ 'ตำแหน่ง + 2^i '.
เมื่อใช้สิ่งนี้ เราจะพิจารณาตำแหน่งดัชนี
ตัวอย่าง
โปรแกรมเพื่อแสดงการทำงานของโซลูชันของเรา
#include <iostream> #include <math.h> using namespace std; void generatePrefixSum(int arr[], int prefSum[], int n){ prefSum[0] = arr[0]; for (int i = 1; i < n; i++) prefSum[i] = prefSum[i - 1] + arr[i]; } int findPreSumIndexBL(int prefSum[], int n, int x){ int P = 0; int LOGN = log2(n); if (x <= prefSum[0]) return 0; for (int i = LOGN; i >= 0; i--) { if (P + (1 << i) < n && prefSum[P + (1 << i)] < x) { P += (1 << i); } } return P + 1; } int main(){ int arr[] = { 5, 2, 9, 4, 1 }; int X = 19; int n = sizeof(arr) / sizeof(arr[0]); int prefSum[n] = { 0 }; generatePrefixSum(arr, prefSum, n); cout<<"The index of first elements of the array greater than the given number is "; cout<<findPreSumIndexBL(prefSum, n, X); return 0; }
ผลลัพธ์
The index of first elements of the array greater than the given number is 3