牛客 NC51003 Matrix

文章介绍了如何通过行和列哈希技术处理二维矩阵,以高效计算子矩阵的哈希值,用于字符串匹配和子结构查找。关键点在于处理行和列前缀和的计算策略以及避免哈希冲突。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

牛客 NC51003 Matrix

二维Hash的板子题,我们先对每行的前缀维护一下Hash值,最对每列维护每行前缀的前缀值

void get_hash(int n,int m,int type)
{
    for (int i=1; i<=n; i++)
	for (int j=1; j<=m; j++)
	    if (type) hashs[i][j]=hashs[i][j-1]*base1+s[i][j];
	    else sub_hash[i][j]=sub_hash[i][j-1]*base1+s[i][j];
		
    for (int j=1; j<=m; j++)
	for (int i=1; i<=n; i++)
	    if (type) hashs[i][j]=hashs[i-1][j]*base2+hashs[i][j];
	    else sub_hash[i][j]=sub_hash[i-1][j]*base2+sub_hash[i][j];
}

为什么可以分别计算行和列的前缀和?

  1. 行哈希计算:首先,对于每一行,我们计算从第一个字符到当前字符的前缀和(使用base1作为基数)。这相当于将每一行看作一个独立的字符串,并计算其哈希值。这样,hashs[i][j]就存储了从第i行第1个字符到第i行第j个字符的哈希值。

  2. 列哈希计算:接着,我们利用已经计算好的行哈希值,进一步计算跨越多行的列哈希值。对于每一列,我们将每一行的哈希值作为新的“字符”,并使用base2作为基数来计算列的前缀和。这样,hashs[i][j]就存储了从第1行第j个字符到第i行第j个字符的哈希值(跨越了所有行)。

为什么直接通过矩阵的前缀和计算得到的答案是有问题的?

如果我们尝试直接计算整个矩阵的前缀和来作

### NC200369 四舍五入 解题思路 对于给定的小数字符串,需要按照指定的次数 `t` 进行四舍五入操作。每次四舍五入都是基于当前最右边的一位小数来决定是否进位。 #### 思路分析 1. **初始化处理** - 首先读取输入数据并解析成可操作的形式。 - 将原始分数转换为字符列表以便逐位修改[^2]。 2. **核心逻辑** - 对于每一次四舍五入: - 如果当前位置小于等于4,则直接截断后续部分; - 若大于等于5则向前一位加一,并继续检查前一位是否会再次触发进位直到不再发生为止。 - 更新剩余需处理的有效位数计数器 `t`,当其减至零时停止循环。 3. **特殊情况考虑** - 当所有有效数字都被处理完毕但仍存在未完成的四舍五入需求时,在结果前面补上相应数量的'1'。 #### 代码实现 以下是 Python 实现: ```python def round_number(n, t, num_str): # 转换为list方便操作 nums = list(num_str) dot_index = nums.index('.') start_pos = dot_index + min(t, len(nums) - 1) while t > 0 and '.' not in str(start_pos): current_digit = int(nums[start_pos]) if current_digit >= 5: carry = True for i in range(start_pos - 1, -1, -1): if nums[i].isdigit(): next_digit = (int(nums[i]) + 1) % 10 if next_digit != 0 or i == 0: carry = False nums[i] = str(next_digit) break if carry: nums.insert(0, '1') del nums[start_pos:] t -= 1 try: start_pos = max(dot_index + min(t, len(nums) - dot_index - 1), 0) except ValueError as e: pass result = ''.join([str(x) for x in nums]).rstrip('0').rstrip('.') or "0" return result if __name__ == "__main__": n, t = map(int, input().split()) initial_score = input() print(round_number(n, t, initial_score)) ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值