diff --git a/TangDou/AcWing_TiGao/T5/QiWang/217.drawio b/TangDou/AcWing_TiGao/T5/QiWang/217.drawio
new file mode 100644
index 0000000..d9a22c8
--- /dev/null
+++ b/TangDou/AcWing_TiGao/T5/QiWang/217.drawio
@@ -0,0 +1 @@
+5VpLc9owEP41zLSHMtbLj2NDHj00M+mkbdLePFiAEmNRI4rJr6+MZbAlB0gCVgkzHLwraWV9+3m1WtFBvXF2lYaT0TWPaNyBTpR10HkHQoAh7OQ/J1oUGt8jhWKYskh1Witu2RNVSkdpZyyi01pHwXks2KSu7PMkoX1R04Vpyuf1bgMe12edhENqKG77YWxq71gkRkrrErxu+ELZcFRODdygaBmHZW+1lOkojPi8okIXHdRLORfF0zjr0ThHrwSmGHf5TOvqzVKaiF0GDB7cy3F2+fPp4QpP/izI7Ovi/pOy8jeMZ2rFD+ptxaLEQJqRcEvhbD5igt5Own7eMpcel7qRGMdSAvIxnE4KHwxYRuWsZ1OR8scVcFhqBjwRyssQlbKaqmFF5evRVNCsolIrvKJ8TEW6kF1Ua4AV2opvIFDyfO09olSjit9KXaj4MlxZXiMqHxSozQBf9JxfT78ZFL9/XH0PxfUw/PatAeAM2EbYgLMB9GcRdoM6wtCxjDA0EUZHjbDneTWEEbKMMDIRhkeNcKBxGNvmMDYRfjxqhH2/jjCxzWFiIJwcNcAu1AC2TWHXANjEN4k+5zmZlPpxOJ2yfh3WlM+SKMfz3NEQ9KRMMybuVVv+/Ct/7hIlnWeVpvNFKSRyaffK/FKojMrF9bClVI4z/VkshkZGumgkK1M+S/t0e84lwnRIxbbUwSTAjg5OaRwK9rf+uk1eVzPccCYXst6EHC1EehpximWqUdW0UzdENENQM1TgYBhaknC17Nfz0jN4edeQgckPUdTJWFCgx2OeSk3CkzwuDFgca6owZsMkZ7TkAZX6s/yzZvIE8Vk1jFkUxc8Flc2U30cg1uIE8M044TbQCB0qTvitxYku9CuhArwmULw4TORGb2jKJFY5GZa9LIQOaDN0BNoXLw3vJ3RApBk6cOgI2qMq2Zmp/8XWhK3yy8V1Wug5za788n3YdT3PdQMijWpWEfa6aHn4klsfQq7jtsq9sgB2ePKBN1BvFTYrKdUugdMCZZFVypZlz7dSNgBBTlnfgW7g5QyumSXY6WIMCCI+CggO2mWsWeo6VLh8wQlgA2e1MwHcwtmltJ+9fVOVZSuRiVUiBxtCpsztq40owK8jOXBg/XPBoN0zAzBrinum8u603HZU3SMD8TEwEDj6yVTPCnelmavXqFo+mQKzrnqogPmC/PK1W/q+8tJNtyj/NzNPIy81S9Ut7PJvyEs9WI+nEi+7+zw8Bi6/oyhrVv7vGi4IT6YACD3LBUBg3hTcNdwnvluHGJ8Wari6adcjTTVy8/7x3XrE9zSHNPxppF2HmEXywYcMfDwdl7hAcwm27RKzGCxdgk7IJZ7216rVRZ4tl5Tz110CT8gl+t6Obe/t0KwBSpc8npBL9DsrQg7mEimu/z1apMvrP+Gii38=
\ No newline at end of file
diff --git a/TangDou/AcWing_TiGao/T5/QiWang/217.md b/TangDou/AcWing_TiGao/T5/QiWang/217.md
new file mode 100644
index 0000000..f96f671
--- /dev/null
+++ b/TangDou/AcWing_TiGao/T5/QiWang/217.md
@@ -0,0 +1,204 @@
+##[$AcWing$ $217$. 绿豆蛙的归宿](https://www.acwing.com/problem/content/description/219/)
+
+### 一、题目描述
+给出一个有向无环的连通图,起点为 $1$ ,终点为 $N$,每条边都有一个长度。
+
+数据保证从起点出发能够到达图中所有的点,图中所有的点也都能够到达终点。
+
+绿豆蛙从起点出发,走向终点。
+
+到达每一个顶点时,如果有 $K$ 条离开该点的道路,绿豆蛙可以选择任意一条道路离开该点,并且走向每条路的概率为 $1/K$。
+
+现在绿豆蛙想知道,从起点走到终点所经过的路径总长度的 **期望** 是多少?
+
+**输入格式**
+第一行: 两个整数 $N,M$,代表图中有 $N$ 个点、$M$ 条边。
+
+第二行到第 $1+M$ 行: 每行 $3$ 个整数 $a,b,c$,代表从 $a$ 到 $b$ 有一条长度为 $c$ 的有向边。
+
+**输出格式**
+输出从起点到终点路径总长度的 **期望值**,结果四舍五入保留两位小数。
+
+**数据范围**
+$1≤N≤10^5,1≤M≤2N$
+
+**输入样例:**
+```cpp {.line-numbers}
+4 4
+1 2 1
+1 3 2
+2 3 3
+3 4 4
+```
+
+**输出样例:**
+```cpp {.line-numbers}
+7.00
+```
+
+### 二、数学期望
+
+首先明白一点:到达某个结果的期望值 = 这个结果 * 从起始状态到这个状态的概率
+
+**什么意思呢?**
+
+如图:
+
+
+我们计算从$1$号点到$3$号点的期望距离
+
+路径$1$. $\displaystyle 1 \rightarrow 3:E_1=2×\frac{1}{2}=1$
+
+路径$2$. $\displaystyle 1 \rightarrow 2 \rightarrow 3:E_2=1×\frac{1}{2}+3×\frac{1}{2}×1=2$
+
+这里路径$2$中从$1$到$2$概率为$\displaystyle \frac{1}{2}$,但单看从$2$到$3$概率就是$1$,但是从$1$到$3$那就是从($1$到$2$的概率)$\displaystyle \frac{1}{2}$×$1$($2$到$3$的概率)=$\displaystyle \frac{1}{2}$。
+
+所以从 点$1$ 到 点$3$ 的数学期望值=$1+2=3$
+>总结
+① 概率是叠乘的
+② 概率计算出来后,需要乘上边权值,做为本边权的贡献值
+③ 所有贡献值累加和就是期望
+
+
+
+本题有 **正推** 和 **倒推** 两种写法:
+
+#### 正推法
+
+
+设:
+- $a_1, a_2, a_3 … a_k$ 到 $j$ 的权值为 $w_1, w_2, w_3 … w_k$,
+- 从起点到这$k$个点的概率为:$p_1, p_2, p_3 … p_k$
+- 每个点的出度为:$out_1, out_2, out_3, … , out_k$
+
+这里的$1\sim k$个点的从起点的到该点的概率一定是确定的,也就是说这个点的概率是被更新完的,即此时这个点的入度为$0$!
+
+那么就有:
+$$f(i):表示从起点到i点的期望距离$$
+$$f(j)=\frac{f(1)+w_1\times p_1}{out_1}+\frac{f(2)+w_2\times p_2}{out_2}+\frac{f(3)+w_3\times p_3}{out_3}+...+\frac{f(k)+w_k\times p_k}{out_k} $$
+
+#### 正推代码
+```cpp {.line-numbers}
+#include
+using namespace std;
+const int N = 1e5 + 10, M = N << 1;
+
+// 邻接表
+int h[N], e[M], ne[M], w[M], idx;
+void add(int a, int b, int c) {
+ e[idx] = b, w[idx] = c, ne[idx] = h[a], h[a] = idx++;
+}
+
+int n, m; // n个顶点,m条边
+
+int out[N], in[N]; // 出度,入度
+double f[N], g[N]; // f:数学期望结果 g:概率
+
+void topsort() {
+ queue q;
+ // 起点为1,起点的概率为100%
+ q.push(1);
+ g[1] = 1.0;
+ f[1] = 0.0;
+
+ // DAG,执行拓扑序,以保证计算的顺序正确,确保递归过程中,前序数据都已处理完毕
+ while (q.size()) {
+ auto u = q.front();
+ q.pop();
+
+ for (int i = h[u]; ~i; i = ne[i]) { // 枚举的是每边相邻边
+ int v = e[i]; // 此边,一端是t,另一端是j
+ // 此边边条w[i]
+ f[v] += (f[u] + w[i] * g[u]) / out[u];
+ g[v] += g[u] / out[u]; // g[j]也需要概率累加
+ // 拓扑序的标准套路
+ in[v]--;
+ if (!in[v]) q.push(v);
+ }
+ }
+}
+
+int main() {
+ // 初始化邻接表
+ memset(h, -1, sizeof h);
+ cin >> n >> m;
+
+ while (m--) {
+ int a, b, c;
+ cin >> a >> b >> c;
+ add(a, b, c);
+ // 维护出度,入度
+ out[a]++, in[b]++;
+ }
+ // 拓扑序
+ topsort();
+
+ // 正向递推,输出结果,保留两位小数
+ printf("%.2lf", f[n]);
+
+ return 0;
+}
+```
+#### 倒推法
+现在学会了正推,来看看 **逆推**,即 **从终点找到起点**
+
+
+
+
+设 $f[x]$ 表示结点 $x$ 走到终点所经过的路径的期望长度。显然 $f[n]=0$ ,最后要求 $f[1]$ 。
+
+一般来说,**初始状态确定时可用顺推,终止状态确定时可用逆推**。
+
+设 $x$ 出发有 $k$ 条边,分别到达 $y_1,y_2...y_k$ ,边长分别为 $z_1,z_2...z_k$ ,根据数学期望的定义和性质,有:
+
+$$f[x]=\frac 1 k\times (f[y_1]+z_1)+\frac 1 k\times (f[y_2]+z_2)+...+\frac 1 k\times (f[y_k]+z_k)=\frac 1 k \times \sum_{i=1}^k(f[y_i]+z_i)$$
+根据设定已经确定是能够到达 $n$ 点了,概率为 $1$ 。
+
+$f[n]$ 已知,需要求解 $f[1]$ ,建立 **反向图**,按照 **拓扑序** 求解。
+
+#### 倒推代码
+```cpp {.line-numbers}
+#include
+using namespace std;
+const int N = 100010, M = N << 1;
+int n, m;
+int in[N], g[N]; // 入度,入度的备份数组,原因:in在topsort中会不断变小受破坏
+double f[N];
+
+// 链式前向星
+int e[M], h[N], idx, w[M], ne[M];
+void add(int a, int b, int c = 0) {
+ e[idx] = b, ne[idx] = h[a], w[idx] = c, h[a] = idx++;
+}
+
+void topsort() {
+ queue q;
+ q.push(n);
+ f[n] = 0; // n到n的距离期望是0
+ while (q.size()) {
+ int u = q.front();
+ q.pop();
+ for (int i = h[u]; ~i; i = ne[i]) { // 枚举每条入边(因为是反向图)
+ int v = e[i];
+ f[v] += (f[u] + w[i]) / g[v];
+ in[v]--;
+ if (in[v] == 0) q.push(v);
+ }
+ }
+}
+int main() {
+ memset(h, -1, sizeof h);
+ cin >> n >> m;
+
+ while (m--) {
+ int a, b, c;
+ cin >> a >> b >> c;
+ add(b, a, c); // 反向图,计算从n到1
+ in[a]++; // 入度
+ g[a] = in[a]; // 入度数量
+ }
+ topsort();
+ printf("%.2lf\n", f[1]);
+ return 0;
+}
+```
\ No newline at end of file
diff --git a/TangDou/AcWing_TiGao/T5/QiWang/217_DaoTui.cpp b/TangDou/AcWing_TiGao/T5/QiWang/217_DaoTui.cpp
new file mode 100644
index 0000000..c3378c1
--- /dev/null
+++ b/TangDou/AcWing_TiGao/T5/QiWang/217_DaoTui.cpp
@@ -0,0 +1,43 @@
+#include
+using namespace std;
+const int N = 100010, M = N << 1;
+int n, m;
+int in[N], g[N]; // 入度,入度的备份数组,原因:in在topsort中会不断变小受破坏
+double f[N];
+
+// 链式前向星
+int e[M], h[N], idx, w[M], ne[M];
+void add(int a, int b, int c = 0) {
+ e[idx] = b, ne[idx] = h[a], w[idx] = c, h[a] = idx++;
+}
+
+void topsort() {
+ queue q;
+ q.push(n);
+ f[n] = 0; // n到n的距离期望是0
+ while (q.size()) {
+ int u = q.front();
+ q.pop();
+ for (int i = h[u]; ~i; i = ne[i]) { // 枚举每条入边(因为是反向图)
+ int v = e[i];
+ f[v] += (f[u] + w[i]) / g[v];
+ in[v]--;
+ if (in[v] == 0) q.push(v);
+ }
+ }
+}
+int main() {
+ memset(h, -1, sizeof h);
+ cin >> n >> m;
+
+ while (m--) {
+ int a, b, c;
+ cin >> a >> b >> c;
+ add(b, a, c); // 反向图,计算从n到1
+ in[a]++; // 入度
+ g[a] = in[a]; // 入度数量
+ }
+ topsort();
+ printf("%.2lf\n", f[1]);
+ return 0;
+}
\ No newline at end of file
diff --git a/TangDou/AcWing_TiGao/T5/QiWang/217_ZhengTui.cpp b/TangDou/AcWing_TiGao/T5/QiWang/217_ZhengTui.cpp
new file mode 100644
index 0000000..c843c08
--- /dev/null
+++ b/TangDou/AcWing_TiGao/T5/QiWang/217_ZhengTui.cpp
@@ -0,0 +1,59 @@
+#include
+using namespace std;
+const int N = 1e5 + 10, M = N << 1;
+
+// 邻接表
+int h[N], e[M], ne[M], w[M], idx;
+void add(int a, int b, int c) {
+ e[idx] = b, w[idx] = c, ne[idx] = h[a], h[a] = idx++;
+}
+
+int n, m; // n个顶点,m条边
+
+int out[N], in[N]; // 出度,入度
+double f[N], g[N]; // f:数学期望结果 g:概率
+
+void topsort() {
+ queue q;
+ // 起点为1,起点的概率为100%
+ q.push(1);
+ g[1] = 1.0;
+ f[1] = 0.0;
+
+ // DAG,执行拓扑序,以保证计算的顺序正确,确保递归过程中,前序数据都已处理完毕
+ while (q.size()) {
+ auto u = q.front();
+ q.pop();
+
+ for (int i = h[u]; ~i; i = ne[i]) { // 枚举的是每边相邻边
+ int v = e[i]; // 此边,一端是t,另一端是j
+ // 此边边条w[i]
+ f[v] += (f[u] + w[i] * g[u]) / out[u];
+ g[v] += g[u] / out[u]; // g[j]也需要概率累加
+ // 拓扑序的标准套路
+ in[v]--;
+ if (!in[v]) q.push(v);
+ }
+ }
+}
+
+int main() {
+ // 初始化邻接表
+ memset(h, -1, sizeof h);
+ cin >> n >> m;
+
+ while (m--) {
+ int a, b, c;
+ cin >> a >> b >> c;
+ add(a, b, c);
+ // 维护出度,入度
+ out[a]++, in[b]++;
+ }
+ // 拓扑序
+ topsort();
+
+ // 正向递推,输出结果,保留两位小数
+ printf("%.2lf", f[n]);
+
+ return 0;
+}
\ No newline at end of file
diff --git a/TangDou/AcWing_TiGao/T5/QiWang/218.md b/TangDou/AcWing_TiGao/T5/QiWang/218.md
new file mode 100644
index 0000000..a4c716a
--- /dev/null
+++ b/TangDou/AcWing_TiGao/T5/QiWang/218.md
@@ -0,0 +1,152 @@
+##[$AcWing$ $218$. 扑克牌 ](https://www.acwing.com/problem/content/description/220/)
+
+### 一、题目描述
+$Admin$ 生日那天,$Rainbow$ 来找 $Admin$ 玩扑克牌。
+
+玩着玩着 $Rainbow$ 觉得太没意思了,于是决定给 $Admin$ 一个考验。
+
+$Rainbow$ 把一副扑克牌($54$张)随机洗开,倒扣着放成一摞。
+
+然后 $Admin$ 从上往下依次翻开每张牌,每翻开一张黑桃、红桃、梅花或者方块,就把它放到对应花色的堆里去。
+
+$Rainbow$ 想问问 $Admin$,得到 $A$ 张黑桃、$B$ 张红桃、$C$ 张梅花、$D$ 张方块需要翻开的牌的张数的期望值 $E$ 是多少?
+
+特殊地,如果翻开的牌是大王或者小王,$Admin$ 将会把它作为某种花色的牌放入对应堆中,使得放入之后 $E$的值尽可能小。
+
+由于 $Admin$ 和 $Rainbow$ 还在玩扑克,所以这个程序就交给你来写了。
+
+**输入格式**
+输入仅由一行,包含四个用空格隔开的整数,$A,B,C,D$。
+
+**输出格式**
+输出需要翻开的牌数的期望值 $E$,四舍五入保留 $3$ 位小数。
+
+如果不可能达到输入的状态,输出 `-1.000`。
+
+**数据范围**
+$0≤A,B,C,D≤15$
+
+**输入样例:**
+```cpp {.line-numbers}
+1 2 3 4
+```
+
+**输出样例:**
+```cpp {.line-numbers}
+16.393
+```
+
+### 二、题意分析
+
+$Q$:为什么从终止状态向起始状态递推?
+
+**答**:满足条件的终止状态较多,而起始状态唯一。考虑以终止状态为初值,起始状态为目标,进行动态规划。
+
+#### 状态表示
+$f[a][b][c][d][x][y]$ : 当前已翻开状态下,还需翻开牌的数量 **期望数**。
+
+- $a,b,c,d$ 为已翻开的各类牌 (黑红花片) 的数量
+- $x,y$代表大、小王的状态($0$为未翻开,$1$代表已翻开且当做黑桃,以此类推), 设 $rst$ 为剩余牌的数量。
+$$rst=54-a-b-c-d-(x!=0)-(y!=0)$$
+
+若 $a < 13$,则当前抽到黑桃的贡献为
+
+$$\frac{13−a}{rst} \times f[a+1][b][c][d][x][y]$$
+
+其余花色同理。若小王被抽取,取可转移状态期望最小的一个进行状态转移,其贡献为:
+
+$$\frac{1}{rst} \times \min_{1≤i≤4}f[a][b][c][d][i][y]$$
+
+大王同理。
+
+记忆化搜索求解,若无牌可抽仍未到达 $a > = A \&\& b > = B \&\& c > = C \&\& d > = D$ 的终止状态,则期望为正无穷,代表不合法的状态。
+
+
+#### 三、实现代码
+
+```cpp {.line-numbers}
+#include
+using namespace std;
+const int N = 15;
+const int INF = 0x3f3f3f3f;
+double f[N][N][N][N][5][5];
+int st[N][N][N][N][5][5];
+int A, B, C, D;
+
+//如果大小王翻出来放1里,则a++,放2里b++,...
+void add(int &a, int &b, int &c, int &d, int x) {
+ if (x == 1) a++;
+ if (x == 2) b++;
+ if (x == 3) c++;
+ if (x == 4) d++;
+}
+
+/*
+功能:计算当前状态f(a,b,c,d,x,y)下的期望值
+*/
+double dfs(int a, int b, int c, int d, int x, int y) {
+ //记忆化,同时因为f为double类型,不能使用传统的memset(0x3f)之类
+ //进行初始化并判断是否修改过,只能再开一个st数组
+ if (st[a][b][c][d][x][y]) return f[a][b][c][d][x][y];
+ st[a][b][c][d][x][y] = 1;
+
+ //递归出口:当前状态是否到达目标状态,目标状态的期望值是0
+ int ta = a, tb = b, tc = c, td = d; //抄出来
+ add(ta, tb, tc, td, x), add(ta, tb, tc, td, y); //大王小王会改变四个花色的数量
+ if (ta >= A && tb >= B && tc >= C && td >= D) return 0;
+
+ //当前状态下的剩余牌数量
+ int rst = 54 - ta - tb - tc - td;
+ if (rst == 0) return INF; //还没有完成目标,没有剩余的牌了,无解
+
+ //当前状态可以向哪些状态转移
+ // Q:v为什么要初始化为1?
+ // A:看题解内容
+ double v = 1;
+ if (a < 13) //黑桃有剩余,可能选出的是黑桃
+ v += dfs(a + 1, b, c, d, x, y) * (13 - a) / rst;
+ if (b < 13) //红桃有剩余,可能选出的是红桃
+ v += dfs(a, b + 1, c, d, x, y) * (13 - b) / rst;
+ if (c < 13) //梅花有剩余,可能选出的是梅花
+ v += dfs(a, b, c + 1, d, x, y) * (13 - c) / rst;
+ if (d < 13) //方块有剩余,可能选出的是方块
+ v += dfs(a, b, c, d + 1, x, y) * (13 - d) / rst;
+
+ //如果小王没有被选出
+ if (x == 0)
+ v += min(min(dfs(a, b, c, d, 1, y), dfs(a, b, c, d, 2, y)), min(dfs(a, b, c, d, 3, y), dfs(a, b, c, d, 4, y))) / rst;
+
+ //如果大王没有被选出
+ if (y == 0)
+ v += min(min(dfs(a, b, c, d, x, 1), dfs(a, b, c, d, x, 2)), min(dfs(a, b, c, d, x, 3), dfs(a, b, c, d, x, 4))) / rst;
+
+ return f[a][b][c][d][x][y] = v;
+}
+
+int main() {
+ cin >> A >> B >> C >> D;
+ //① 终点状态不唯一,起点是唯的的,所以以起点为终点,以终点为起点,反着推
+ //② AcWing 217. 绿豆蛙的归宿 需要建图,本题不用建图
+ double res = dfs(0, 0, 0, 0, 0, 0); //四种花色、大小王都还没有被抽取
+
+ if (res > INF / 2) //因为是浮点数,不能用等号判断是不是相等,简单的办法就是INF/2
+ puts("-1.000");
+ else
+ printf("%.3f\n", res);
+ return 0;
+}
+```
+
+### 四、期望值为什么初始化为$1$?
+
+$f[i]$: 从$i$卡牌状态到终点状态所需要的**期望卡牌数**
+
+每次抽一张牌变到下个状态,所以每条路径的权值为$1$
+
+$$\large f[v]=p_1×(f[1]+1)+p_2×(f[2]+1)+p_3×(f[3]+1)+…+p_k×(f[k]+1) = \\
+\sum_{i=1}^{k}p_i+\sum_{i=1}^{k}p_i \times f[i]
+$$
+
+ 因为$v$一定能到达下个局面,所以下个状态的概率和为$1$,这里的$\large \displaystyle \sum_{i=1}^{k}p_i=1$ 那么就有:$\displaystyle \large f[v]=1+\sum_{i=1}^{k}p_i \times f[i]$
+综上这里的$f[v]$可以初始化为$1$!
+
\ No newline at end of file
diff --git a/TangDou/AcWing_TiGao/T5/QiWang/218_dfs.cpp b/TangDou/AcWing_TiGao/T5/QiWang/218_dfs.cpp
new file mode 100644
index 0000000..cb9302e
--- /dev/null
+++ b/TangDou/AcWing_TiGao/T5/QiWang/218_dfs.cpp
@@ -0,0 +1,70 @@
+#include
+using namespace std;
+const int N = 15;
+const int INF = 0x3f3f3f3f;
+double f[N][N][N][N][5][5];
+int st[N][N][N][N][5][5];
+int A, B, C, D;
+
+//如果大小王翻出来放1里,则a++,放2里b++,...
+void add(int &a, int &b, int &c, int &d, int x) {
+ if (x == 1) a++;
+ if (x == 2) b++;
+ if (x == 3) c++;
+ if (x == 4) d++;
+}
+
+/*
+功能:计算当前状态f(a,b,c,d,x,y)下的期望值
+*/
+double dfs(int a, int b, int c, int d, int x, int y) {
+ //记忆化,同时因为f为double类型,不能使用传统的memset(0x3f)之类
+ //进行初始化并判断是否修改过,只能再开一个st数组
+ if (st[a][b][c][d][x][y]) return f[a][b][c][d][x][y];
+ st[a][b][c][d][x][y] = 1;
+
+ //递归出口:当前状态是否到达目标状态,目标状态的期望值是0
+ int ta = a, tb = b, tc = c, td = d; //抄出来
+ add(ta, tb, tc, td, x), add(ta, tb, tc, td, y); //大王小王会改变四个花色的数量
+ if (ta >= A && tb >= B && tc >= C && td >= D) return 0;
+
+ //当前状态下的剩余牌数量
+ int rst = 54 - ta - tb - tc - td;
+ if (rst == 0) return INF; //还没有完成目标,没有剩余的牌了,无解
+
+ //当前状态可以向哪些状态转移
+ // Q:v为什么要初始化为1?
+ // A:看题解内容
+ double v = 1;
+ if (a < 13) //黑桃有剩余,可能选出的是黑桃
+ v += dfs(a + 1, b, c, d, x, y) * (13 - a) / rst;
+ if (b < 13) //红桃有剩余,可能选出的是红桃
+ v += dfs(a, b + 1, c, d, x, y) * (13 - b) / rst;
+ if (c < 13) //梅花有剩余,可能选出的是梅花
+ v += dfs(a, b, c + 1, d, x, y) * (13 - c) / rst;
+ if (d < 13) //方块有剩余,可能选出的是方块
+ v += dfs(a, b, c, d + 1, x, y) * (13 - d) / rst;
+
+ //如果小王没有被选出
+ if (x == 0)
+ v += min(min(dfs(a, b, c, d, 1, y), dfs(a, b, c, d, 2, y)), min(dfs(a, b, c, d, 3, y), dfs(a, b, c, d, 4, y))) / rst;
+
+ //如果大王没有被选出
+ if (y == 0)
+ v += min(min(dfs(a, b, c, d, x, 1), dfs(a, b, c, d, x, 2)), min(dfs(a, b, c, d, x, 3), dfs(a, b, c, d, x, 4))) / rst;
+
+ return f[a][b][c][d][x][y] = v;
+}
+
+int main() {
+ cin >> A >> B >> C >> D;
+ //① 终点状态不唯一,起点是唯的的,所以以起点为终点,以终点为起点,反着推
+ //② AcWing 217. 绿豆蛙的归宿 需要建图,本题不用建图
+ double res = dfs(0, 0, 0, 0, 0, 0); //四种花色、大小王都还没有被抽取
+
+ if (res > INF / 2) //因为是浮点数,不能用等号判断是不是相等,简单的办法就是INF/2
+ puts("-1.000");
+ else
+ printf("%.3f\n", res);
+ return 0;
+}
\ No newline at end of file
diff --git a/TangDou/AcWing_TiGao/T5/QiWang/218_dp.cpp b/TangDou/AcWing_TiGao/T5/QiWang/218_dp.cpp
new file mode 100644
index 0000000..ca614d3
--- /dev/null
+++ b/TangDou/AcWing_TiGao/T5/QiWang/218_dp.cpp
@@ -0,0 +1,48 @@
+#include
+using namespace std;
+const int INF = 0x3f3f3f3f;
+const int N = 15;
+double f[N][N][N][N][5][5];
+int a, b, c, d;
+int main() {
+ memset(f, -1, sizeof f);
+ cin >> a >> b >> c >> d;
+
+ for (int i = 13; i >= 0; i--)
+ for (int j = 13; j >= 0; j--)
+ for (int k = 13; k >= 0; k--)
+ for (int w = 13; w >= 0; w--)
+ for (int x = 4; x >= 0; x--)
+ for (int y = 4; y >= 0; y--) {
+ double &v = f[i][j][k][w][x][y];
+ if (i + (x == 1) + (y == 1) >= a && j + (x == 2) + (y == 2) >= b
+ && k + (x == 3) + (y == 3) >= c && w + (x == 4) + (y == 4) >= d) {
+ v = 0;
+ continue;
+ }
+
+ v = 1;
+ int sum = i + j + k + w + (x != 0) + (y != 0);
+ if (i < 13) v += f[i + 1][j][k][w][x][y] * (13 - i) / (54 - sum);
+ if (j < 13) v += f[i][j + 1][k][w][x][y] * (13 - j) / (54 - sum);
+ if (k < 13) v += f[i][j][k + 1][w][x][y] * (13 - k) / (54 - sum);
+ if (w < 13) v += f[i][j][k][w + 1][x][y] * (13 - w) / (54 - sum);
+ if (x == 0) {
+ double t = INF;
+ for (int u = 1; u <= 4; u++) t = min(t, f[i][j][k][w][u][y] / (54 - sum));
+ v += t;
+ }
+ if (y == 0) {
+ double t = INF;
+ for (int u = 1; u <= 4; u++) t = min(t, f[i][j][k][w][x][u] / (54 - sum));
+ v += t;
+ }
+ }
+
+ if (f[0][0][0][0][0][0] > 54)
+ printf("-1.000");
+ else
+ printf("%.3lf", f[0][0][0][0][0][0]);
+
+ return 0;
+}
\ No newline at end of file