移除K位数字

1.题目描述:

在这里插入图片描述

2.解题思想

基本:去除的数字需从左到右进行,而去除判断条件就是左边数字大于右边数字,则去除那个左边数字。
特殊情况: 1.整个字符串是升序排列
2.前导零的存在
3.去除k位数后成为空字符串
运用数据结构:栈

3.代码以及注释

package 移除K位数字;

import java.util.LinkedList;
import java.util.Scanner;

public class RemoveKNumber {

	public static void main(String[] args) {
		// TODO 自动生成的方法存根
		Scanner scanner = new Scanner(System.in);
		String num=scanner.next();
		int k=scanner.nextInt();
		scanner.close();
		System.out.println(removeKDigits(num,k));
	}
	public static String removeKDigits(String num,int k)
	{
		LinkedList<Character> stack = new LinkedList<Character>();
		for(char digit : num.toCharArray())
		{
			while(stack.size()>0 && k>0 && stack.peekLast()>digit)
			/*循环的条件:
			 * 1.栈不为空
			 * 2.还未移除k位数字
			 * 3.栈顶元素大于digit(也就是前一位数字大于后一位数字)
			 * */
			{
				stack.removeLast();
				k--;
			}
			stack.addLast(digit);
		}
		//处理 移除的数字仍未达到k位数字 情况
		for(int i=0;i<k;i++)
		{
			stack.removeLast();
		}
		//处理 类似(100200,1)得到 200  的情况 
		StringBuilder result = new StringBuilder();
		boolean flag=true;
		for(char digit:stack)
		{
			if(flag && digit == '0') continue;
			flag = false;
			result.append(digit);
		}
		//处理最后结果是空字符串 的情况
		if(result.length()==0) return "0";
		return result.toString();
	}
}

4.Summary

先分析去除数字的规律,再注意一下问题的特殊情况

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
以下是使用贪心算法实现的 C++ 代码: ```cpp #include <iostream> #include <string> using namespace std; string removeKDigits(string l, int k) { string res = ""; int n = l.length(); int keep = n - k; // 最终保留的数字数 for (char c : l) { while (k > 0 && res.length() > 0 && res.back() > c) { res.pop_back(); // 移除前面大于当前数字数字 k--; } res.push_back(c); } // 如果还需要移除数字,从后往前移除 while (k > 0 && res.length() > 0) { res.pop_back(); k--; } // 移除前导零 while (res.length() > 1 && res[0] == '0') { res.erase(res.begin()); } // 如果结果长度大于保留数,截取前keep if (res.length() > keep) { res = res.substr(0, keep); } return res; } int main() { string l; int k; cout << "请输入字符串l: "; cin >> l; cout << "请输入数字k: "; cin >> k; string res = removeKDigits(l, k); cout << "移除" << k << "数字后的最小数字为:" << res << endl; return 0; } ``` 算法思路: 1. 定义一个字符串 `res`,用于存储最终的结果; 2. 遍历输入字符串 `l` 中的每个字符 `c`,并将其依次加入到 `res` 中; 3. 如果当前数字 `c` 小于 `res` 中的最后一个数字,说明前面的数字大于当前数字,为了使最终数字最小,应该将前面的数字移除。我们可以使用一个 while 循环,不断地将 `res` 中大于当前数字数字移除,直到 `res` 中没有比当前数字大的数字或者已经移除了 `k` 个数字; 4. 将当前数字 `c` 加入到 `res` 中; 5. 如果还需要移除数字,从 `res` 的末尾开始移除,直到移除了 `k` 个数字或者 `res` 中已经没有数字; 6. 最后,移除 `res` 中的前导零,并且如果 `res` 的长度大于保留数 `keep`,则只保留前 `keep` 个数字。 时间复杂度:$O(n)$,其中 $n$ 是字符串 `l` 的长度。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Mandela688

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值