当前位置: 首页 > article >正文

算法训练:贪心与回溯

目录

1.手套(贪心算法)

2.字符串通配符(回溯算法)


1.手套

题目描述:

在地下室里放着n种颜色的手套,手套分左右手,但是每种颜色的左右手手套个数不一定相同。A先生现在要出门,所以他要去地下室选手套。但是昏暗的灯光让 他无法分辨手套的颜色,只能分辨出左右手。所以他会多拿一些手套,然后选出一双颜色相同的左右手手套。现在的问题是,他至少要拿多少只手套(左手加右 手),才能保证一定能选出一双颜色相同的手套。 给定颜色种数n(1≤n≤13),同时给定两个长度为n的数组left,right,分别代表每种颜色左右手手套的数量。数据保证左右的手套总数均不超过26,且一定存在至少一 种合法方案。

  

输入描述:第一行输入一个正整数n(n ≤ 1000) 第二行为n个数正整数x (x ≤ 1000)

 

输出描述:输出可以产生的幸运的袋子数 

 

测试样例:

 

4,[0,7,1,6],[1,5,0,6]

 

返回:10(解释:可以左手手套取2只,右手手套取8只)

思路分析:

我们从题目中可以看到,我们需要得出一个结果就是至少要拿多少只手套。这里一般采用贪心算法。我们来分析一下:

因为在拿手套的时候,是盲拿的,拿了多少不知道,拿的是右手还是左手的也不知道。因此,我们就必须保证,在拿手套的时候,拿到的数量最起码能够覆盖左手或者右手的所有颜色!基于此再拿一只另外一只手的手套,便是我们需要的结果。

比如测试样例中,我们需要拿到覆盖左手的手套颜色那么多的数量,那就将左手手套的数量全部加起来,然后减去最少的那个颜色的数量再加一,这就是覆盖了左手的颜色可拿到至少的左手手套的数量了!然后再从右手的手套随便取一个,那么就能够保证至少有一个颜色的手套可以匹配得上。

但是要注意0数量的情况,对于0数量的颜色的手套,我们需要将其额外拿出来,因为如果一种颜色的手套,它的右手或者左手的手套数量为0,即使覆盖了一只手的手套颜色的数量,也可能会使另外一只手的手套匹配不上的问题。

代码如下:

class Gloves
{
public:
	int FindMininum(int n, vector<int> left, vector<int>right)
	{
		int left_sum = 0, left_min = INT_MAX;
		int right_sum = 0, right_min = INT_MAX;

		int sum = 0;//用于统计另一只手的手套为零时,另一只手的手套的个数
		for (int i = 0; i < n; ++i)
		{
			if (left[i] * right[i] == 0)
			{
				sum += left[i] + right[i];//需要额外统计
			}
			else
			{
				left_sum += left[i];
				left_min = left_min < left[i] ? left_min : left[i];
				right_sum += right[i];
				right_min = right_min < right[i] ? right_min : right[i];
			}
		}

		//将它们加起来,sum + min(left_sum - left_min + 1, right_sum - right_min + 1)这一步
		//相当于把一只手的所有颜色的手套都拿了,然后再加个1,就是从另外一只手的手套堆取一只。
		//这样就必定能够至少匹配一双出来。
		return sum + min(left_sum - left_min + 1, right_sum - right_min + 1) + 1;
	}
};

2.字符串通配符

题目描述:

问题描述:在计算机中,通配符一种特殊语法,广泛应用于文件搜索、数据库、正则表达式等领域。现要求各位实现字符串通配符的算法。 要求: 实现如下2个通配符: *:匹配0个或以上的字符(注:能被*和?匹配的字符仅由英文字母和数字0到9组成,下同) ?:匹配1个字符 注意:匹配时不区分大小写。 输入: 通配符表达式; 一组字符串。 输出: 返回不区分大小写的匹配结果,匹配成功输出true,匹配失败输出false

 

输入描述:先输入一个带有通配符的字符串,再输入一个需要匹配的字符串

 

输出描述:返回不区分大小写的匹配结果,匹配成功输出true,匹配失败输出false

 

样例:

输入: te?*.*

输出: txtec.cpp

返回:false

 

输入: te?*.*

输出: tetedsdsdc.cpp

返回:true

思路分析

要匹配字符串,有以下三种情况:

①它是字符,并且相等。那么就直接往下走。

②它是'?',问号可以跟任何字符匹配。那么直接往下走。

③它是'*',星号有三种情况:

⭐它没得匹配,即已经走完,这个星号不需要匹配任何字符。

⭐它只需要匹配一个字符,那么直接往下走。

⭐它需要匹配很多个字符,那么这个字符串不需要往下走,需要匹配的字符串往下走。

往下走即使用回溯,即递归算法。

#include <iostream>
#include <string>
using namespace std;
bool strMach(const char* parten, const char* str)
{
    if (*parten == '\0' && *str == '\0')
    {
        return true;
    }
    if (*parten == '\0' || *str == '\0')
    {
        return false;
    }

    if (*parten == '?')
    {
        return strMach(parten + 1, str + 1);
    }
    else if (*parten == '*')
    {
        return strMach(parten + 1, str) || strMach(parten + 1, str + 1) || strMach(parten, str + 1);
    }
    else if (*parten == *str)
    {
        return strMach(parten + 1, str + 1);
    }
    return false;
}
int main() {

    string parten, str;
    cin >> parten >> str;
    bool ret = strMach(parten.c_str(), str.c_str());
    if (ret)
    {
        cout << "true" << endl;
    }
    else {
        cout << "false" << endl;
    }
    return 0;
}

http://www.kler.cn/a/5832.html

相关文章:

  • 12_Redis发布订阅
  • 代码随想录 哈希 test 8
  • 计算机网络(三)——局域网和广域网
  • mapbox基础,style样式汇总,持续更新
  • 新兴的开源 AI Agent 智能体全景技术栈
  • Elasticsearch:Query rules 疑难解答
  • debian 10编译安装gcc 9.5.0
  • 剪枝与重参第一课:修剪结构
  • Python类和对象的命名空间
  • 简述vue生命周期,以及每个周期做的事情?
  • 简单谈谈图形界面和命令行的区别
  • 大一被忽悠进了培训班
  • Java题目训练——统计每个月兔子的总数和字符串通配符
  • 2022年河南省高等职业教育技能大赛软件测试赛项竞赛任务书
  • 使用RegQueryValueEx作为可能为REG_DWORD或REG_SZ的注册表值
  • 蓝桥杯·3月份刷题集训Day03
  • GPT4国内镜像站
  • 第四十八天打卡
  • Date Time组件(下)
  • 【JavaEE】线程池
  • flutter安装自用笔记
  • 第一批00后,已经变成“职场老油条”了
  • TCP流套接字编程
  • Spark sql怎么使用Kafka Avro序列化器
  • 阿里云高级技术专家林立翔:基于阿里云弹性GPU服务的神龙AI加速引擎,无缝提升AI训练性能
  • 每日学术速递4.3