- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我有一个数组 A
连同 3 个变量 k
, x
和 y
.我必须找到无序对的数量 (i,j)
这样两个元素的总和 mod k
等于 x
和相同的两个元素的乘积 mod k
等于y
.对不必是不同的。换句话说,(i,j)
的数量这样
(A[i]+A[j])%k == x
和 (A[i]*A[j])%k == y
其中 0 <= i < j < size of A
.
例如,设A={1,2,3,2,1}
, k=2
, x=1
, y=0
.那么答案是 6,因为对是:(1,2)
, (1,2)
, (2,3)
, (2,1)
, (3,2)
, 和 (2,1)
.
我使用了蛮力方法,但显然这是 Not Acceptable 。
最佳答案
模运算有以下两个规则:
((a mod k) * (b mod k)) mod k = (a * b) mod k
((a mod k) + (b mod k)) mod k = (a + b) mod k
因此我们可以使用 separate chaining 将所有值排序到哈希表中和 k
桶。
查找m < k
,这样对于给定的 n < k
: (n + m) mod k = x
.这个问题只有一个解决方案:
n < x
: m < x
必须持有。因此 m = x - n
n == x
: m = 0
n > x
: 我们需要找到 m
这样 n + m = x + k
.因此 m = x + k - n
这样,我们可以很容易地为每个值列表确定相应的值,这样对于任何对 (a, b)
两个列表的叉积 (a + b) mod k = x
持有。
乘法有点棘手。幸运的是,我们已经得到了加法的匹配同余类(见上文),它也必须是乘法的匹配同余类,因为两个约束都需要成立。要验证给定的同余类匹配,我们只需要检查 (n * m) mod k = y
(n
和 m
定义如上)。如果这个表达式成立,我们就可以构建对,否则不存在匹配的元素。
这将是上述示例的工作 python 代码:
def modmuladd(ls, x, y, k):
result = []
# create tuples of indices and values
indices = zip(ls, range(0, len(ls)))
# split up into congruence classes
congruence_cls = [[] for i in range(0, k)]
for p in indices:
congruence_cls[p[0] % k].append(p)
for n in range(0, k):
# congruence class to match addition
if n < x:
m = x - n
elif n == x:
m = 0
else:
m = x + k - n
# check if congruence class matches for multiplication
if (n * m) % k != y or len(congruence_cls[m]) == 0:
continue # no matching congruence class
# add matching tuple to result
result += [(a, b) for a in congruence_cls[n] for b in congruence_cls[m] if a[1] <= b[1]]
result += [(a, b) for a in congruence_cls[m] for b in congruence_cls[n] if a[1] <= b[1]]
# sort result such according to indices of first and second element, remove duplicates
sorted_res = sorted(sorted(set(result), key=lambda p: p[1][1]), key=lambda p: p[0][1])
# remove indices from result-set
return [(p[0][0], p[1][0]) for p in sorted_res]
请注意,仅需要排序和消除重复项,因为此代码专注于同余类的使用,而不是完美优化。可以很容易地调整此示例以提供排序,而无需通过较小的修改进行排序。
print(modmuladd([1, 2, 3, 2, 1], 1, 0, 2))
输出:
[(1, 2), (1, 2), (2, 3), (2, 1), (3, 2), (2, 1)]
编辑:
该算法的最坏情况复杂度仍然是 O(n^2)
,由于构建所有可能的大小列表对 n
是O(n^2)
.但是,使用此算法可以将匹配对的搜索减少到O(k)
。与 O(n)
预处理。因此可以在 O(n)
中计算结果对用这种方法。假设数字在同余类上平均分布,该算法可以构建属于 O(n^2/k^2)
中解决方案集的所有对。 .
编辑 2:
一个只计算的实现会像这样工作:
def modmuladdct(ls, x, y, k):
result = 0
# split up into congruence classes
congruence_class = {}
for v in ls:
if v % k not in congruence_class:
congruence_class[(v % k)] = [v]
else:
congruence_class[v % k].append(v)
for n in congruence_class.keys():
# congruence class to match addition
m = (x - n + k) % k
# check if congruence class matches for multiplication
if (n * m % k != y) or len(congruence_class[m]) == 0:
continue # no matching congruence class
# total number of pairs that will be built
result += len(congruence_class[n]) * len(congruence_class[m])
# divide by two since each pair would otherwise be counted twice
return result // 2
每一对在结果中恰好出现两次:一次按顺序出现,一次按相反顺序出现。通过将结果除以二,这得到了纠正。运行时间是 O(n + k)
(假设字典操作是 O(1)
)。
关于algorithm - 具有给定总和和乘积的对数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42169327/
我希望我的问题有一个非常简单的解决方案。我只是找不到它: 假设您有两个向量(一个是列向量,一个是行向量)A、B: A = [1,2,3] B = [4;5;6] 如果我们按如下方式将它们相乘,我们会得
我有一个 Tuple 的列表: "dog", 25 "cat", 5 "cat", 7 "rat", 4 "dog", 10 我需要的 Linq 查询规则必须满足以下条件:我需要按字符串值对元组进行分
给定 2 个不同的 NDarray,A 和 B,形状相同但尺寸任意,我如何获得 NDarray C,其中 C 是 A 和 B(含)范围内所有整数的乘积。 我的意思是A是起始数组,B是结束数组,我想要数
假设我需要为某些输入构建一个真值表,它要求我提供逻辑和、算术和和逻辑乘积。它们之间有什么区别? 最佳答案 逻辑和 - 一种计算机加法,当一个或两个输入变量为 1 时,结果为 1;当输入变量均为 0 时
我正在尝试执行一个简单的矩阵乘法 vector 乘法,但出于某种原因,我在几次乘法的结果中得到了错误的符号。我不知道为什么会这样,任何指针将不胜感激。 这是我的全部代码,即矩阵 * vector 函数
我在上一个主题中找到了一些关于 cuda 矩阵 vector 积的代码: Matrix-vector multiplication in CUDA: benchmarking & performanc
我遇到的第一个问题是显示三个数字中的最小和最大。出现两个单独的警报 - 第一个警报说第二大数字是最大的(因为它还没有考虑第三个数字),第二个警报正确地指出三个中最大的数字是最大的.不确定为什么会这样—
我有两个矩阵 a = np.matrix([[1,2], [3,4]]) b = np.matrix([[5,6], [7,8]]) 我想得到元素乘积,[[1*5,2*6], [3*7,4*8]],等
我有一个数组和一个 vector : ArrayXd m1(3, 1337); ArrayXd v1(1, 1337); ArrayXd result(3, 1337); 现在我想将 m1 的每一行与
我有两个 3D 矩阵: a = np.random.normal(size=[3,2,5]) b = np.random.normal(size=[5,2,3]) 我想要每个切片分别沿 2 轴和 0
我正在创建一个 C++ 软件,我需要一个包装器,它基于 Eigen 库,实现类似于官方网页中解释的运算符* https://eigen.tuxfamily.org/dox/group__Matrixf
我正在尝试将张量 (m, n, o) 分解为矩阵 A(m, r)、B (n, r) 和 C (k, r)。这被称为 PARAFAC 分解。 Tensorly已经做了这种分解。 一个重要的步骤是将 A、
我目前正面临这个问题。我有两个矩阵 MatrixXf答: 0.5 0.5 0.5 0.50.694496 0.548501 0.680067 0.7171110
我有以下 df: df = pd.DataFrame({'A': ['foo', 'bar', 'dex', 'tru'], 'B': ['abc', 'def'
假设我们有 2 个 2X2 numpy 数组: X=np.array([[0,1],[1,0]]) 和 I=np.array([[1,0],[0,1]]) 考虑一下克罗内克产品 XX=X^X 我让符号
我想弄清楚这是 Eigen 中的错误还是我做错了什么。我只想要两个复数 vector [1,i] 和 [1,-i] 的点积。答案是 1*1 + i*(-i) = 2。但是 Eigen 给出的答案是零。
我的 C 代码有问题。我所做的就是这样: #include int main() { float zahlen[2]; for (int i = 0; i < 2; i++) {
为了找到数字的因数,我正在使用函数 void primeFactors(int n) # include # include # include # include using namespa
我是一名优秀的程序员,十分优秀!