LeetCode 1585 Check If String Is Transformable With Substring Sort Operations
题意:一个只有0-9组成的字符串,每次选择任意一个子串,按照数字从小到大排序。问从源字符串能否经过若干次操作转换成目标字符串。
题解:首先题目只问了是否,而没有问多少次,所以可以判断出,我们不关心过程是怎么样的,只关心结果能否到达。所以不太可能是动态规划,搜索什么的。
既然是看结果,那估计是个找规律的题目。首先看目标字符串的第一个字符x,如果它是由源字符串的某个字符x经过排序之后,安排到了第一个位置,那么它一定满足,在源字符串中从起始位置到x的位置的子串中,x一定是最小的那个。否则x就无法移动到第一个位置。
一次内推,第二个字符x2, 由于第一个字符x已经确定了我们把x变成最大值,那么x2应该满足从源字符串起始位置到x2的位置的子串中,x2是最小的那个。
所以这道题目就变成了单点更新,求区间最值。
使用简单优雅的数据结构,树状数组,即可解决问题。
树状数组的区间最值,要比区间和复杂一点,且更新的时间效率是Log(n)*Log(n)
跟线段树比差一点。
class Solution {
public:
int c[100005];
int a[100005];
vector<int> pos[10];
int n;
bool isTransformable(string s, string t) {
for(int i = s.length()-1;i>=0;i--)
{
int num = s[i]-'0';
pos[num].push_back(i+1);
}
n = s.length();
for(int i=0;i<=n;i++)
{
c[i] = INT_MAX;
}
for(int i=0;i<s.length();i++)
{
add(i+1,s[i]-'0');
a[i+1]=s[i]-'0';
}
for(int i=0;i<t.length();i++)
{
if(pos[t[i]-'0'].size()==0)
return false;
int p = pos[t[i]-'0'].back();
pos[t[i]-'0'].pop_back();
int m = get(1,p);
if(m!=t[i]-'0')
return false;
add(p,INT_MAX);
}
return true;
}
int lowbit(int x)
{
return x&(-x);
}
void add(int index,int num)
{
a[index] = num;
while(index<=n)
{
int l = index-lowbit(index)+1;
int m = get(l, index-1);
if(l==index)
{
c[index]=a[index];
}
else if(min(m, a[index])!=c[index])
{
c[index]=min(m, a[index]);
}
else
{
break;
}
index+=lowbit(index);
}
}
int get(int l, int r)
{
if(l>r)
return INT_MAX;
int ans = INT_MAX;
while(r>=l)
{
if(r-lowbit(r) < l-1)
{
ans = min(ans, a[r]);
r--;
}
else
{
ans = min(ans, c[r]);
r-=lowbit(r);
}
}
return ans;
}
};
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】博客园社区专享云产品让利特惠,阿里云新客6.5折上折
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· Java 中堆内存和栈内存上的数据分布和特点
· 开发中对象命名的一点思考
· .NET Core内存结构体系(Windows环境)底层原理浅谈
· C# 深度学习:对抗生成网络(GAN)训练头像生成模型
· .NET 适配 HarmonyOS 进展
· 手把手教你更优雅的享受 DeepSeek
· 腾讯元宝接入 DeepSeek R1 模型,支持深度思考 + 联网搜索,好用不卡机!
· AI工具推荐:领先的开源 AI 代码助手——Continue
· 探秘Transformer系列之(2)---总体架构
· V-Control:一个基于 .NET MAUI 的开箱即用的UI组件库