
19/05/2026
19/05/2026
# nhập dữ liệu
a = list(map(int, input().split()))
k = int(input())
n = len(a)
dem = 0
#đếm số lượng đoạn con thỏa mãn
for i in range(n):
tong = 0
for j in range(i, n):
tong += a[j]
if tong % k == 0:
dem += 1
print(dem)
Khanh
19/05/2026
code này là đơn giản nha b, nếu b đang giải bài thi hsg tin thì b tham khảo thêm tài liệu gg hoặc code AI để tối ưu bài làm hơn nha
#include <bits/stdc++.h>
using namespace std;
int main()
{
long long n,k,a[100005],d[100005];
cin>>n>>k;
for(int i=1;i<=n;i++) cin>>a[i];
d[0]=0;
for(int i=1;i<=n;i++) d[i]=d[i-1]+a[i];
long long nmax=-1e18;
for(int i=k+1;i<=n;i++)
{
nmax=max(nmax,d[i]-d[i-k]);
}
cout<<nmax;
}
19/05/2026
Phân tích thuật toán
Để giải bài toán này một cách hiệu quả (độ phức tạp \(O(n)\)), chúng ta sử dụng phương pháp Tổng tiền tố (Prefix Sum) và Mảng băm/Tần suất dư:
1. Một đoạn con từ vị trí \(i\) đến \(j\) có tổng chia hết cho \(k\) khi và chỉ khi tổng tiền tố tại \(j\) và tổng tiền tố tại \(i-1\) có cùng số dư khi chia cho \(k\).
2. Ta duyệt qua dãy số, tính tổng tích lũy và lưu lại số lượng các số dư đã xuất hiện.
________________________________________
Mã nguồn Python
python
def count_subarrays_divisible_by_k(arr, k):
# Dictionary để lưu tần suất xuất hiện của các số dư
# Khởi tạo {0: 1} vì tổng bằng 0 mặc định chia hết cho k
remainder_counts = {0: 1}
current_sum = 0
count = 0
for x in arr:
current_sum += x
# Tính số dư của tổng tiền tố cho k
remainder = current_sum % k
# Nếu số dư này đã từng xuất hiện, cộng số lần xuất hiện vào kết quả
if remainder in remainder_counts:
count += remainder_counts[remainder]
remainder_counts[remainder] += 1
else:
remainder_counts[remainder] = 1
return count
# Ví dụ sử dụng:
# n = int(input("Nhập số lượng phần tử n: "))
# a = list(map(int, input("Nhập dãy số: ").split()))
# k = int(input("Nhập số nguyên dương k: "))
a = [4, 5, 0, -2, -3, 1]
k = 5
print(f"Số lượng đoạn con có tổng chia hết cho {k} là: {count_subarrays_divisible_by_k(a, k)}")
Use code with caution.
Giải thích:
• current_sum % k: Tìm số dư của tổng từ đầu đến vị trí hiện tại.
• remainder_counts: Lưu trữ số lần mỗi số dư xuất hiện. Nếu một số dư \(r\) xuất hiện \(m\) lần trước đó, điều đó có nghĩa là có \(m\) đoạn con kết thúc tại vị trí hiện tại có tổng chia hết cho \(k\).
• Độ phức tạp: \(O(n)\), tối ưu hơn rất nhiều so với cách duyệt lồng nhau \(O(n^2)\).
Nếu bạn muốn hỏi bài tập
Các câu hỏi của bạn luôn được giải đáp dưới 10 phút
CÂU HỎI LIÊN QUAN
30/05/2026
Top thành viên trả lời