给定N个数字和Q个查询,每个查询均由L和R组成。任务是编写一个程序,打印一个程序,该程序将给定范围LR中的所有数字相除。
例子 :
Input : a = {3, 4, 2, 2, 4, 6}
Q = 2
L = 1 R = 4
L = 2 R = 6
Output : 0
2
Explanation : The range 1-4 has {3, 4, 2, 2}
which does not have any number that divides all the
numbers in this range.
The range 2-6 has {4, 2, 2, 4, 6} which has 2 numbers {2, 2} which divides
all numbers in the given range.
Input: a = {1, 2, 3, 5}
Q = 2
L = 1 R = 4
L = 2 R = 4
Output: 1
0
天真的方法:针对每个查询从范围LR进行迭代,并检查index-i处的给定元素是否将范围中的所有数字相除。我们对所有元素进行计数,将所有数字相除。在最坏的情况下,每个查询的复杂度将为O(n 2 ) 。
下面是朴素方法的实现:
C++
// CPP program to Count elements which
// divides all numbers in range L-R
#include
using namespace std;
// function to count element
// Time complexity O(n^2) worst case
int answerQuery(int a[], int n,
int l, int r)
{
// answer for query
int count = 0;
// 0 based index
l = l - 1;
// iterate for all elements
for (int i = l; i < r; i++)
{
int element = a[i];
int divisors = 0;
// check if the element divides
// all numbers in range
for (int j = l; j < r; j++)
{
// no of elements
if (a[j] % a[i] == 0)
divisors++;
else
break;
}
// if all elements are divisible by a[i]
if (divisors == (r - l))
count++;
}
// answer for every query
return count;
}
// Driver Code
int main()
{
int a[] = { 1, 2, 3, 5 };
int n = sizeof(a) / sizeof(a[0]);
int l = 1, r = 4;
cout << answerQuery(a, n, l, r) << endl;
l = 2, r = 4;
cout << answerQuery(a, n, l, r) << endl;
return 0;
}
Java
// Java program to Count elements which
// divides all numbers in range L-R
import java.io.*;
class GFG
{
// function to count element
// Time complexity O(n^2) worst case
static int answerQuery(int a[], int n,
int l, int r)
{
// answer for query
int count = 0;
// 0 based index
l = l - 1;
// iterate for all elements
for (int i = l; i < r; i++)
{
int element = a[i];
int divisors = 0;
// check if the element divides
// all numbers in range
for (int j = l; j < r; j++)
{
// no of elements
if (a[j] % a[i] == 0)
divisors++;
else
break;
}
// if all elements are divisible by a[i]
if (divisors == (r - l))
count++;
}
// answer for every query
return count;
}
// Driver Code
public static void main (String[] args)
{
int a[] = { 1, 2, 3, 5 };
int n = a.length;
int l = 1, r = 4;
System.out.println( answerQuery(a, n, l, r));
l = 2; r = 4;
System.out.println( answerQuery(a, n, l, r));
}
}
// This code is contributed by anuj_67..
Python3
# Python 3 program to Count elements which
# divides all numbers in range L-R
# function to count element
# Time complexity O(n^2) worst case
def answerQuery(a, n, l, r):
# answer for query
count = 0
# 0 based index
l = l - 1
# iterate for all elements
for i in range(l, r, 1):
element = a[i]
divisors = 0
# check if the element divides
# all numbers in range
for j in range(l, r, 1):
# no of elements
if (a[j] % a[i] == 0):
divisors += 1
else:
break
# if all elements are divisible
# by a[i]
if (divisors == (r - l)):
count += 1
# answer for every query
return count
# Driver Code
if __name__ =='__main__':
a = [1, 2, 3, 5]
n = len(a)
l = 1
r = 4
print(answerQuery(a, n, l, r))
l = 2
r = 4
print(answerQuery(a, n, l, r))
# This code is contributed by
# Shashank_Sharma
C#
// C# program to Count elements which
// divides all numbers in range L-R
using System;
class GFG
{
// function to count element
// Time complexity O(n^2) worst case
static int answerQuery(int []a, int n,
int l, int r)
{
// answer for query
int count = 0;
// 0 based index
l = l - 1;
// iterate for all elements
for (int i = l; i < r; i++)
{
//int element = a[i];
int divisors = 0;
// check if the element divides
// all numbers in range
for (int j = l; j < r; j++)
{
// no of elements
if (a[j] % a[i] == 0)
divisors++;
else
break;
}
// if all elements are divisible by a[i]
if (divisors == (r - l))
count++;
}
// answer for every query
return count;
}
// Driver Code
public static void Main ()
{
int []a = { 1, 2, 3, 5 };
int n = a.Length;
int l = 1, r = 4;
Console.WriteLine(answerQuery(a, n, l, r));
l = 2; r = 4;
Console.WriteLine(answerQuery(a, n, l, r));
}
}
// This code is contributed by anuj_67..
PHP
输出:
1
0
高效的方法:使用细分树来解决此问题。如果元素将给定范围内的所有数字相除,则该元素是该范围内的最小数字,它是给定范围LR中所有元素的gcd。因此,鉴于最小值LR等于该范围的gcd,范围LR的最小值的计数将是我们对每个查询的答案。问题归结为使用细分树找到每个范围的GCD,MINIMUM和countMINIMUM 。在树的每个节点上,存储了三个值。
在查询给定范围时,如果gcd和给定范围的最小值相等,则返回countMINIMUM作为答案。如果它们不相等,则返回0作为答案。
以下是有效方法的实现:
// CPP program to Count elements
// which divides all numbers in
// range L-R efficient approach
#include
using namespace std;
#define N 100005
// predefines the tree with nodes
// storing gcd, min and count
struct node
{
int gcd;
int min;
int cnt;
} tree[5 * N];
// function to construct the tree
void buildtree(int low, int high,
int pos, int a[])
{
// base condition
if (low == high)
{
// initially always gcd and min
// are same at leaf node
tree[pos].min = tree[pos].gcd = a[low];
tree[pos].cnt = 1;
return;
}
int mid = (low + high) >> 1;
// left-subtree
buildtree(low, mid, 2 * pos + 1, a);
// right-subtree
buildtree(mid + 1, high, 2 * pos + 2, a);
// finds gcd of left and right subtree
tree[pos].gcd = __gcd(tree[2 * pos + 1].gcd,
tree[2 * pos + 2].gcd);
// left subtree has the minimum element
if (tree[2 * pos + 1].min < tree[2 * pos + 2].min)
{
tree[pos].min = tree[2 * pos + 1].min;
tree[pos].cnt = tree[2 * pos + 1].cnt;
}
// right subtree has the minimum element
else
if (tree[2 * pos + 1].min > tree[2 * pos + 2].min)
{
tree[pos].min = tree[2 * pos + 2].min;
tree[pos].cnt = tree[2 * pos + 2].cnt;
}
// both subtree has the same minimum element
else
{
tree[pos].min = tree[2 * pos + 1].min;
tree[pos].cnt = tree[2 * pos + 1].cnt +
tree[2 * pos + 2].cnt;
}
}
// function that answers every query
node query(int s, int e, int low, int high, int pos)
{
node dummy;
// out of range
if (e < low or s > high)
{
dummy.gcd = dummy.min = dummy.cnt = 0;
return dummy;
}
// in range
if (s >= low and e <= high)
{
node dummy;
dummy.gcd = tree[pos].gcd;
dummy.min = tree[pos].min;
if (dummy.gcd != dummy.min)
dummy.cnt = 0;
else
dummy.cnt = tree[pos].cnt;
return dummy;
}
int mid = (s + e) >> 1;
// left-subtree
node ans1 = query(s, mid, low,
high, 2 * pos + 1);
// right-subtree
node ans2 = query(mid + 1, e, low,
high, 2 * pos + 2);
node ans;
// when both left subtree and
// right subtree is in range
if (ans1.gcd and ans2.gcd)
{
// merge two trees
ans.gcd = __gcd(ans1.gcd, ans2.gcd);
ans.min = min(ans1.min, ans2.min);
// when gcd is not equal to min
if (ans.gcd != ans.min)
ans.cnt = 0;
else
{
// add count when min is
// same of both subtree
if (ans1.min == ans2.min)
ans.cnt = ans2.cnt + ans1.cnt;
// store the minimal's count
else
if (ans1.min < ans2.min)
ans.cnt = ans1.cnt;
else
ans.cnt = ans2.cnt;
}
return ans;
}
// only left subtree is in range
else if (ans1.gcd)
return ans1;
// only right subtree is in range
else if (ans2.gcd)
return ans2;
}
// function to answer query in range l-r
int answerQuery(int a[], int n, int l, int r)
{
// calls the function which returns
// a node this function returns the
// count which will be the answer
return query(0, n - 1, l - 1, r - 1, 0).cnt;
}
// Driver Code
int main()
{
int a[] = { 3, 4, 2, 2, 4, 6 };
int n = sizeof(a) / sizeof(a[0]);
buildtree(0, n - 1, 0, a);
int l = 1, r = 4;
// answers 1-st query
cout << answerQuery(a, n, l, r) << endl;
l = 2, r = 6;
// answers 2nd query
cout << answerQuery(a, n, l, r) << endl;
return 0;
}
输出:
0
2
时间复杂度:由于树的构建需要O(n),而找出gcd的需要花费O(log n),因此树的构建的时间复杂度为O(n logn )。在最坏的情况下,每个查询所花费的时间为O(log n * log n),因为内置函数__gcd需要O(log n)