题目: P3392 涂国旗 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)
思路:
思路1
题中给出了几个限制条件,我想着去模拟一下这几个条件求解,用行头和行尾指针,每一行都求最优的方式去求解,当行数为3时三行都是确定的,大于三行就先把头行和尾行确定下来,再用头尾指针去不断确定两边要涂的颜色(找到该行最多的色块,并且确定该行是否可以涂该色(颜色顺序不能被改变,白蓝红)),但是实现起来发现条件太多,过于繁琐,因为头尾指针所在的行就算确定了要涂的色,但是要和整个国旗的统一起来就很难,不能有一个统一的方法去计算,需要用许多的条件判断等,于是放弃。
思路2
求出所有中间行(因为头行和尾行的色块是确定的)的排列,再拿求出的排列和题目给的色块比较,得出最小的值,输出即可,问题是求出所有的色块排列最大次数是3^48这样下去必然超时,于是放弃。
思路3
同样是暴力,为什么我就是想不起来这种方式呢!!!国旗总共三种颜色,需要确定两个色块的分界,有分界了就去求当前国旗中三个色块区域不同于当前区域颜色的色块数量(比如第一个是白色,那么不同于白色的就计数下来,其他区域相同),通过遍历不同分界下的需要变更色块的数量,不断更新min就行,最后输出count(每次循环要重置)就可以。由于最多50行,所以三层循环也没问题。
实现
思路3实现
在写完代码后发现一直只有28分,这道题而且没有数据可以下载,一开始我以为是循环的索引控制的问题(我用的是下标从0开始),后来我就用从1开始,结果也不可以,折腾了好久才发现是因为忘记把计数器count重置了,不然count的数据都累加了,还怎么找出最小的值呢,真是太马虎了。
提到索引的控制不妨再说一点,一般情况下要么是从0开始,要么从1开始,前者符合计算机的设计,后者便于理解。
最外层循环的索引从0开始呢,就代表i所在的是白色分界且该分界属于白色区域,这点要注意
package LOQ.暴力枚举;
import java.util.Scanner;
/**
* @Author Lunau
* @Create 2022-10-08 10:26
* @Description 遍历区域分界线,寻找最小变化的色块
*/
public class P3392涂国旗 {
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
int n = sc.nextInt(),m = sc.nextInt();
String[][] strs = new String[n][m];
sc.nextLine();
for (int i=0;i<n;i++) {
String s = sc.nextLine();
strs[i] = s.split("");
}
int min = Integer.MAX_VALUE;
//遍历白色区域可能的边界行
for (int i=0;i<n-2;i++) {
//遍历蓝色区域可能的边界行
for (int j=i+1;j<n-1;j++) {
int count = 0;
//分别遍历三个区域
for (int a=0;a<=i;a++) { //行
for (int tp=0;tp<m;tp++) { //列
if (!strs[a][tp].equals("W")) {
count ++;
}
}
}
for (int a=i+1;a<=j;a++) {
for (int tp=0;tp<m;tp++) { //列
if (!strs[a][tp].equals("B")) {
count ++;
}
}
}
for (int a=j+1;a<n;a++) {
for (int tp=0;tp<m;tp++) { //列
if (!strs[a][tp].equals("R")) {
count ++;
}
}
}
min = Math.min(min,count);
}
}
System.out.println(min);
}
}
思路1实现(半成品)
package LOQ.暴力枚举;
import java.util.Scanner;
/**
* @Author Lunau
* @Create 2022-10-08 10:26
* @Description
*/
public class P3392涂国旗 {
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
int n = sc.nextInt(),m = sc.nextInt();
String[][] strs = new String[n][m];
sc.nextLine();
for (int i=0;i<n;i++) {
String s = sc.nextLine();
strs[i] = s.split("");
}
//行头,行尾指针
int p = 0,q = n-1;
int count = 0;
//遍历所有行
for (int i=0;i<n/2;i++) {
//第一行最后一行先涂色
if (p==0&&q==n-1) {
for (int j=0;j<m;j++) {
//第一行涂白色
if (!strs[p][j].equals("W")) {
count++;
}
//最后一行涂红色
if (!strs[q][j].equals("R")) {
count++;
}
}
p++;
q--;
}
//中间行
if (q-p>1) {
}
}
}
//n当前行,m总列数,flag标识要该行要涂的成的色块
private static String searchMinColor(String[][] strs,int n,int m,String flag) {
int count = 0;//计数器
for (int i=0;i<m;i++) {
}
return flag;
}
}