【ACM算法竞赛日常训练】DAY4题解与分析【树】【子序列】| 组合数学 | 动态规划

DAY4共2题:

  • 树(组合数学)

  • 子序列(dp,数学)

? 作者:Eriktse
? 简介:19岁,211计算机在读,现役ACM银牌选手?力争以通俗易懂的方式讲解算法!❤️欢迎关注我,一起交流C++/Python算法。(优质好文持续更新中……)?
? 原文链接(阅读原文获得更好阅读体验):https://www.eriktse.com/algorithm/1095.html

题目传送门:https://ac.nowcoder.com/acm/problem/13611

通过观察条件“一个染色方案是合法的,当且仅当对于所有相同颜色的点对(x,y),x到y的路径上的所有点的颜色都要与x和y相同。”我们可以发现,当且仅当染色的点可以全部连通时可以满足条件。

所以现在问题是如何将n个点划分为k块。

我们可以发现在树上,任意删除一条边都会使得联通块个数 + 1

其实块数只要<= k即可,因为我们可以有一些颜色不使用。所以要划分为i块,只需要从n - 1条边中任选i - 1条进行删除即可,方案数是C(n - 1, i - 1)

假设现在我们得到了i (i <= k)个联通块,需要将i种颜色染上去,首先需要C(k, i)种方法取出颜色,然后A(i, i)一个全排列将颜色染上去。

所以答案公式如下:

\[ans=\sum_{i=1}^{k}C(n - 1, i - 1)C(k, i)i!
\]

可能涉及一些快速幂乘法逆元的知识,需要自行学习。

#include <bits/stdc++.h>
#define int long long
using namespace std;
const int maxn = 350, p = 1e9 + 7;

int fac[maxn];

int qmi(int a, int b)
{
    int res = 1;
    while(b)
    {
        if(b & 1)res = res * a % p;
        a = a * a % p, b >>= 1;
    }
    return res;
}

int inv(int x){return qmi(x, p - 2);}

int C(int n, int m)
{
    if(n < m || n < 0 || m < 0)return 0;
    return fac[n] * inv(fac[n - m] * fac[m] % p) % p;
}

signed main()
{
    int n, k;scanf("%lld %lld", &n, &k);
    fac[0] = 1;
    for(int i = 1;i <= n; ++ i)fac[i] = fac[i - 1] * i % p;
    
    int ans = 0;
    for(int i = 1;i <= n; ++ i)//分为i块
    {
        int tmp = C(n - 1, i - 1) * C(k, i) % p * fac[i] % p;
        ans = (ans + tmp) % p;
    }
    printf("%lld\n", ans);
    return 0;
}

子序列

题目传送门:https://ac.nowcoder.com/acm/problem/17065

小技巧:观察数据范围,比较小,应该可以容纳O(n^3)的复杂度,所以可以大胆考虑dp。

首先定义状态dp[i][j]表示以第i个元素结尾,且长度为j的序列的个数

再考虑一下转移,题目中的条件可以进行一些转换:

\[{a_{p_i}}^{p_j} < {a_{p_j}}^{p_i}
\]

等价于:

\[\frac{log(a_{p_i})}{p_i} < \frac{log(a_{p_j})}{p_j}
\]

我们可以记:

\[b_i = \frac{log(a_{p_i})}{p_i}
\]

也就是说对于选出的子序列中的每一个元素,他们满足一个偏序关系,只要我的b[j] > b[i],那么b[j]将会大于所有的b[k] (k < i)

所以我们可以考虑以下的转移:

\[dp_{i, j} = \sum_{k=1}^{i - 1}[b_i > b_k] \times dp_{k, j - 1}
\]

考虑初始化,当最后一个元素确定,序列长度为1(j = 1)时,方案仅有1种。

最后的答案是将所有情况加起来(注意取模,不过这道题数据较弱,不取模也可以过)。

#include <bits/stdc++.h>
#define int long long
using namespace std;
const int maxn = 109, p = 1e9 + 7;

//dp[i][j]表示以第i个元素结尾,长度为j的方案数
int a[maxn], dp[maxn][maxn];


signed main()
{
	int n;scanf("%lld", &n);
	for(int i = 1;i <= n; ++ i)scanf("%lld", a + i);
	
	for(int i = 1;i <= n; ++ i)
    {
        dp[i][1] = 1;
        for(int j = 1;j <= i; ++ j)
        {
            for(int k = 1; k < i; ++ k)
            {
                if(log(a[k]) / k < log(a[i]) / i)
                {
                    dp[i][j] += dp[k][j - 1];
                    dp[i][j] %= p;
                }
            }
        }
    }

	int ans = 0;
	for(int i = 1;i <= n; ++ i)
		for(int j = 1;j <= i; ++ j)
        {
			ans = (ans + dp[i][j]) % p;
        }
	printf("%lld\n", ans);
	return 0;
}

? 本文由eriktse原创,创作不易,如果对您有帮助,欢迎小伙伴们点赞?、收藏⭐、留言?

原文链接:https://www.cnblogs.com/eriktse/p/17262568.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:【ACM算法竞赛日常训练】DAY4题解与分析【树】【子序列】| 组合数学 | 动态规划 - Python技术站

(0)
上一篇 2023年4月18日
下一篇 2023年4月18日

相关文章

  • JavaScript数据结构yocto queue队列链表代码分析

    JavaScript数据结构yocto queue队列链表代码分析 什么是队列? 队列(Queue)是一种基础的数据结构,属于线性结构,它的特点是在队列尾插入元素,同时在队列头删除元素,遵循先进先出(FIFO)的原则。队列可以简单的理解为排队,先到达的先被服务,而后到达的则等在队列尾排队等待。队列的应用非常广泛,例如排队系统、消息队列等。 队列的实现方式 队…

    数据结构 2023年5月17日
    00
  • Python中八大图像特效算法的示例详解

    下面是关于“Python中八大图像特效算法的示例详解”的完整攻略。 1. 八大图像效法简介 图像特效算法是一种用于对图像进行处理的算法,可以使图像更加美观或者增强图像的表现力。在Python中,我们可以使用八大图像特效算法来对图像进行处理。这八大图像特效算法包括:灰度化二值化、反转、镜像、旋转、缩放、模糊和锐化。 2. Python实现八大图像特算法 2.1…

    python 2023年5月13日
    00
  • Java数据结构与算法实现递归与回溯

    Java数据结构与算法实现递归与回溯攻略 什么是递归与回溯 递归是指函数调用自己的过程。在递归过程中,一般需要包含两个部分:递归调用过程和递归出口。递归应用广泛,例如在计算机科学中,递归可应用于算法设计中的分治思想和动态规划。 回溯是指在解决问题时,尝试每一种可能的分步方法,当尝试后发现该方法不行时,取消当前尝试的分步方法,回到上一步,再使用其他可能的分步方…

    数据结构 2023年5月17日
    00
  • Java数据结构常见几大排序梳理

    Java数据结构常见几大排序梳理 在Java中,数据排序是非常常见的操作。下面我们详细讲解一下Java数据结构常见几大排序的梳理。 常见几大排序 Java数据结构中常见几种排序算法包括: 冒泡排序(Bubble Sort) 快速排序(Quick Sort) 插入排序(Insertion Sort) 选择排序(Selection Sort) 希尔排序(Shel…

    数据结构 2023年5月17日
    00
  • python查找与排序算法详解(示图+代码)

    下面是关于“Python查找与排序算法详解”的完整攻略。 1. 查找算法 1.1 线性查找算法 线性查找算法是一种简单的查找算法,它的基本思想是从数据集合的第一个元素开始逐个比较,直到找到目标元素或遍完整个数据集合。在Python中,我们可以使用线性查找算法来查找任意数据类型的元素。 下面使用Python实现性查算法: def linear_search(a…

    python 2023年5月13日
    00
  • 数据结构中的各种排序方法小结(JS实现)

    数据结构中的各种排序方法小结(JS实现) 本文将介绍常见的八种排序算法: 冒泡排序 插入排序 选择排序 快速排序 归并排序 堆排序 希尔排序 计数排序 下面进行详细讲解。 冒泡排序 冒泡排序是一种简单的排序算法,它重复地遍历数组,比较相邻的两个元素,并按大小交换位置,一直到整个数组排序完成。它的时间复杂度为O(n^2)。 示例代码: function bub…

    数据结构 2023年5月17日
    00
  • 详解Python牛顿插值法

    以下是关于“Python牛顿插值法”的完整攻略: 简介 牛顿插值法是一种用于插值的数值分析方法,它可以通过已知的数据点来构造一个多项式函数,从而在数据点之间进行插值。在本教程中,我们将介绍如何使用Python实现牛顿插值法,并提供两个示例说明。 实现牛顿插值法 以下是使用Python实现牛顿插值法的代码: def newton_interpolation(x…

    python 2023年5月14日
    00
  • 数据结构TypeScript之二叉查找树实现详解

    数据结构TypeScript之二叉查找树实现详解 什么是二叉查找树 二叉查找树(Binary Search Tree,简称BST)是一种基础的数据结构,也是一种常用的搜索算法。它通过以二叉树的形式表示各个结点之间的关系,实现了快速查找、添加、删除等操作。对于任何一个节点,其左子树上的节点值均小于该节点的值,右子树上的节点值均大于该节点的值。 二叉查找树的实现…

    数据结构 2023年5月17日
    00
合作推广
合作推广
分享本页
返回顶部