【状态压缩】【动态规划】【C++算法】691贴纸拼词

作者推荐

【动态规划】【数学】【C++算法】18赛车

本文涉及知识点

状态压缩 动态规划

LeetCode:691 贴纸拼词

我们有 n 种不同的贴纸。每个贴纸上都有一个小写的英文单词。
您想要拼写出给定的字符串 target ,方法是从收集的贴纸中切割单个字母并重新排列它们。如果你愿意,你可以多次使用每个贴纸,每个贴纸的数量是无限的。
返回你需要拼出 target 的最小贴纸数量。如果任务不可能,则返回 -1 。
注意:在所有的测试用例中,所有的单词都是从 1000 个最常见的美国英语单词中随机选择的,并且 target 被选择为两个随机单词的连接。
示例 1:
输入: stickers = [“with”,“example”,“science”], target = “thehat”
输出:3
解释:
我们可以使用 2 个 “with” 贴纸,和 1 个 “example” 贴纸。
把贴纸上的字母剪下来并重新排列后,就可以形成目标 “thehat“ 了。
此外,这是形成目标字符串所需的最小贴纸数量。
示例 2:
输入:stickers = [“notice”,“possible”], target = “basicbasic”
输出:-1
解释:我们不能通过剪切给定贴纸的字母来形成目标“basicbasic”。

提示:
n == stickers.length
1 <= n <= 50
1 <= stickers[i].length <= 10
1 <= target.length <= 15
stickers[i] 和 target 由小写英文单词组成

封装类

假定target串有个n1字符,数量分别为m_vMax[0] m_vMax[1]…m_vMax[n1-1]。
对于某个stickers串,忽略target中没有的字符。target第0 1 2… 个字符在sticker的数量为vNum[0],vNum[1]…
则第0个字符的mask为1*vNum[0]
第1个字符的mask为(m_vMax[0]+1)*vNum[1]

第i个字符的mask为:Mul [ 0 , i − 1 ] j \Large^j_{[0,i-1]} [0,i1]j(vMax[j]+1)*vNum[i]
总mask 为各字符mask之和。
注意: 无论如何vNum[i]都大于等于0,小于等于m_vMax[i]

class CMask
{
public:
	void Add(int iMax)//当前最高位范围[0,iMax]
	{
		m_vMax.emplace_back(iMax);
		m_iMaskCount *= (iMax + 1);
	}
	vector<int> FromMask(int iMask)
	{
		vector<int> vNums;
		for (int i = 0; i < m_vMax.size(); i++)
		{
			vNums.emplace_back(iMask % (m_vMax[i] + 1));
			iMask /= (m_vMax[i] + 1);
		}
		return vNums;
	}
	int ToMask(const vector<int>& vNums,int iMul=1)
	{
		int iMask = 0;
		int iUnit = 1;
		for (int i = 0; i < m_vMax.size(); i++)
		{
			iMask += iUnit * min(m_vMax[i],vNums[i]* iMul);
			iUnit *= (m_vMax[i]+1);
		}
		return iMask;
	}
	int MaskSubVector(int iMask, const vector<int>& vNums, const int iMul = 1)
	{
		int iNewMask = 0;
		int iUnit = 1;
		for (int i = 0; i < m_vMax.size(); i++)
		{
			int cur = iMask % (m_vMax[i] + 1);
			cur -= vNums[i] * iMul;
			cur = max(0, cur);
			iNewMask += iUnit * cur;
			iMask /= (m_vMax[i] + 1);
			iUnit *= (m_vMax[i]+1);
		}
		return iNewMask;
	}
	int NeedGroupCount(const vector<int>& need, const vector<int>& has)
	{
		int iMax = 0;
		for (int i = 0; i < m_vMax.size(); i++)
		{
			if (has[i] <= 0)
			{
				continue;
			}
			iMax = max(iMax,need[i] / has[i]+ (0 != need[i] % has[i]));
		}
		return iMax;
	}
public:
	int MaskCount()const
	{
		return m_iMaskCount;
	}	
	int BitCount()const
	{
		return m_vMax.size();
	}
protected:
	int m_iMaskCount = 1;
	vector<int> m_vMax;
};

class CStrMask : public CMask
{
public:
	CStrMask(const string& target)
	{
		vector<int> vCount;
		for (const auto& ch : target)
		{
			if (mCharToIndex.count(ch))
			{
				vCount[mCharToIndex[ch]]++;
			}
			else
			{
				mCharToIndex[ch] = vCount.size();
				vCount.emplace_back(1);
			}
		}
		for (const auto& cnt : vCount)
		{
			CMask::Add(cnt);
		}
	}
	vector<int> GetVector(const string& s )
	{
		vector<int> vCharNums(m_vMax.size());
		for (const char& ch : s)
		{
			if (mCharToIndex.count(ch))
			{
				auto& i = vCharNums[mCharToIndex[ch]];
				if (i < m_vMax[mCharToIndex[ch]])
				{
					i++;
				}				
			}
		}
		return vCharNums;
	}
protected:
	unordered_map<char, int> mCharToIndex;
private:
	void Add(int iMax) {};
};

动态规划

动态规划的状态表示

pre[iMask] 表示利用前i-1个贴纸,达到未完成的字符状态为iMask的消耗的最少贴纸数。1000表示无法达成。
dp[iMask] 表示利用前i 个贴纸,达到未完成的字符状态为iMask的消耗的最少贴纸数。

动态规划的转移方程

iPreMask 当前贴纸 使用的贴纸数量 如果状态发生变化,则更新状态。

动态规划的填表顺序

对于每个合法的iPreMask,计算当前贴纸最多需要多少份。三层循环:第一层,枚举各贴纸。第二层,枚举pre各状态。第三层:枚举当前贴纸使用的数量。每次处理还要枚举各字符。
故总时间复杂度为:O(50*2151515)

动态规划的初始状态

pre[iMaskCount-1=0 其它值为1000。

动态规划的返回值

pre[0]

代码

核心代码

class Solution {
public:
	int minStickers(vector<string>& stickers, string target) {
		CStrMask mask(target);
		vector<vector<int>> vMaskToCounts(mask.MaskCount());
		for (int i = 0; i < mask.MaskCount(); i++)
		{
			vMaskToCounts[i] = mask.FromMask(i);
		}
		vector<int> vPre(mask.MaskCount(), 1000);
		vPre[mask.MaskCount() - 1] = 0;
		for (const auto s : stickers)
		{
			auto vCharNums = mask.GetVector(s);
			vector<int> dp = vPre;//不选择
			for (int iPre = 0; iPre < mask.MaskCount(); iPre++)
			{
				if (vPre[iPre] >= 1000)
				{
					continue;
				}
				const int iSelMax = mask.NeedGroupCount(vMaskToCounts[iPre], vCharNums);
				for (int iSel = 1; iSel <= iSelMax; iSel++)
				{
					const int iNewMask = mask.MaskSubVector(iPre, vCharNums, iSel);
					dp[iNewMask] = min(dp[iNewMask], vPre[iPre] + iSel);
				}
			}
			vPre.swap(dp);
		}
		return (vPre[0] >= 1000) ? -1 : vPre[0];
	}
};

测试用例

template<class T>
void Assert(const T& t1, const T& t2)
{
	assert(t1 == t2);
}

template<class T>
void Assert(const vector<T>& v1, const vector<T>& v2)
{
	if (v1.size() != v2.size())
	{
		assert(false);
		return;
	}
	for (int i = 0; i < v1.size(); i++)
	{
		Assert(v1[i], v2[i]);
	}

}





int main()
{
	

	vector<string> stickers;
	string target;
	{
		Solution sln;
		stickers = { "travel", "quotient", "nose", "wrote", "any" }, target = "lastwest";
		auto res = sln.minStickers(stickers, target);
		Assert(4, res);
	}
	{
		Solution sln;
		stickers = { "with", "example", "science" }, target = "thehat";
		auto res = sln.minStickers(stickers, target);
		Assert(3, res);
	}
	{
		Solution sln;
		stickers = { "notice", "possible" }, target = "basicbasic";
		auto res = sln.minStickers(stickers, target);
		Assert(-1, res);
	}

	{
		Solution sln;
		stickers = { "right", "ten", "year", "share", "period", "paper", "expect", "village", "home", "happen", "ring", "sat", "even", "afraid", "paint", "self", "range", "camp", "note", "read", "paragraph", "run", "basic", "fill", "week", "his", "star", "power", "any", "colony", "object", "free", "dark", "said", "chick", "true", "glad", "child", "room", "lost", "am", "cry", "quiet", "crease", "while", "race", "fun", "found", "dream", "once" }, target = "materialhalf";
		auto res = sln.minStickers(stickers, target);
		Assert(4, res);
	}

	{
		Solution sln;
		stickers = { "indicate","why","finger","star","unit","board","sister","danger","deal","bit","phrase","caught","if","other","water","huge","general","read","gold","shall","master","men","lay","party","grow","view","if","pull","together","head","thank","street","natural","pull","raise","cost","spoke","race","new","race","liquid","me","please","clear","could","reply","often","huge","old","nor" }, target = "fhhfiyfdcwbycma";
		auto res = sln.minStickers(stickers, target);
		Assert(9, res);
	}
	
}

改进简洁版,性能略差

不区分dp pre,因为一个贴纸可以无限使用。无论是pre[iMask]和dp[iMask] 都可增加贴纸。

动态规划的状态表示

状态压缩:如果target第i位已经贴好,则此位为1;否则此位为0。
dp[iMask] 已经完成iMask 消耗的最少贴纸数。

动态规划的转移方程。

对应每个mask,只需要考虑一个当前贴纸。比如: 目标串为aaa,贴纸为a。状态0只考虑一个贴纸。状态1只考虑1个贴纸,总共2个贴纸。状态3只考虑1个贴纸,总共3个贴纸。
如果目标串有多个相同字符,只需要匹配第一字符。“???” 第一张贴纸后,变成"a??" 不需要考虑“?a?"
方案一:第i步匹配第1个a,第j个步匹配第2个a。
方案二:第i步匹配第2个a,第j个步匹配第1个a。
如果方案二,能达到目标,那方案一显然也能达到目标。
iPreMask等于iNewMask 也不用排除,因为新值比旧值大1,必定被淘汰。

class Solution {
public:
	int minStickers(vector<string>& stickers, string target) {
		const int iMaskCount = 1 << target.size();
		vector<int> dp(iMaskCount, 1000);
		dp[0] = 0;
		for (const auto s : stickers)
		{
			int cnt1[26] = { 0 };
			for (const auto& ch : s)
			{
				cnt1[ch - 'a']++;
			}
			for (int iPreMask = 0; iPreMask < iMaskCount; iPreMask++)
			{
				int cnt[26] = { 0 };
				memcpy(cnt, cnt1, sizeof(cnt1));
				int iNewMask = iPreMask;
				for (int j = 0; j < target.size(); j++)
				{
					if (iPreMask & (1 << j))
					{
						continue;//已经拼成
					}
					if (cnt[target[j] - 'a'])
					{
						cnt[target[j] - 'a']--;
						iNewMask |= (1 << j);
					}
				}
				dp[iNewMask] = min(dp[iNewMask], dp[iPreMask] + 1);
			}		
		}
		return (dp.back() >= 1000) ? -1 : dp.back();
	}
};

2023年1月第一版

class Solution {
public:
int minStickers(vector& stickers, string target) {
m_target = target;
for (auto& ch : target)
{
m_mTarget[ch]++;
}
std::unordered_set hasMask;
hasMask.insert(0);
std::unordered_map<int,int> preMaskNum;
preMaskNum[0] = 0;
for (auto& s : stickers)
{
std::unordered_map<char, int> mCur;
for (auto& ch : s)
{
if (m_mTarget.count(ch))
{
mCur[ch]++;
}
}
std::unordered_map<int, int> dp = preMaskNum;
const int iCurMask = MakeMask(mCur);
if (hasMask.count(iCurMask))
{
continue;
}
hasMask.insert(iCurMask);
for (const auto& preMask : preMaskNum )
{
std::unordered_map<char, int> mPre;
ParseMask(mPre, preMask.first);
bool bAdd = true;
int iNum = preMask.second;
while (bAdd)
{
bAdd = false;
for (const auto it : mCur)
{
if (m_mTarget[it.first] > mPre[it.first])
{
bAdd = true;
mPre[it.first] = min(m_mTarget[it.first], mPre[it.first] + mCur[it.first]);
}
}
if (bAdd)
{
iNum++;
const int iMask = MakeMask(mPre);
if (dp.count(iMask))
{
dp[iMask] = min(dp[iMask], iNum);
}
else
{
dp[iMask] = iNum;
}
}
}
}
preMaskNum.swap(dp);
}
int iMaskTargt = MakeMask(m_mTarget);
if (preMaskNum.end() == preMaskNum.find(iMaskTargt))
{
return -1;
}
return preMaskNum[iMaskTargt];
}
int MakeMask(const std::unordered_map<char, int>& nums)const
{
int iMask = 0;
for (const auto& mm : m_mTarget )
{
auto it = nums.find(mm.first);
if ((nums.end() != it) && ( it->second > 0 ))
{
iMask = iMask * (mm.second + 1) + min(mm.second, it->second);
}
else
{
iMask = iMask * (mm.second + 1);
}
}
return iMask;
}
void ParseMask(std::unordered_map<char, int>& nums, int iMask)
{
int iMul = 1;
for (auto& it : m_mTarget)
{
iMul *= (it.second+1);
}
for (auto it = m_mTarget.begin(); it != m_mTarget.end(); ++it)
{
iMul /= (it->second+1);
nums[it->first] = iMask / iMul;
iMask %= iMul;
}
}
std::string m_target;
std::unordered_map<char, int> m_mTarget;
};

2023年1月第2版

class Solution {
public:
int minStickers(vector& stickers, string target) {
m_target = target;
Init();
vector<std::unordered_map<char, int>> mAllCharNum;
{
std::unordered_set hasMask;
hasMask.insert(0);
for (int i = stickers.size() - 1; i >= 0; i–)
{
std::unordered_map<char, int> mCur;
for (auto& ch : stickers[i])
{
if (m_mTarget.count(ch))
{
mCur[ch]++;
}
}
const int iCurMask = MakeMask(mCur);
if (hasMask.count(iCurMask))
{
continue;
}
hasMask.insert(iCurMask);
mAllCharNum.push_back(mCur);
}
}
std::unordered_set setPreMask,setHasDo;
setPreMask.insert(m_iTargetMask);
setHasDo.insert(m_iTargetMask);
for (int iOpeNum = 1; iOpeNum <= target.length(); iOpeNum++)
{
std::unordered_set dp;
for (auto& preMask : setPreMask)
{
std::unordered_map<char, int> mPre;
ParseMask(mPre, preMask);
for (auto& mCur : mAllCharNum)
{
const int iSubMask = GetSubMask(mPre, mCur);
if (0 == iSubMask)
{
continue;
}
const int iNewMask = preMask - iSubMask;
if (setHasDo.count(iNewMask))
{
continue;
}
if (iNewMask == 0 )
{
return iOpeNum;
}
setHasDo.insert(iNewMask);
dp.insert(iNewMask);
}
}
setPreMask.swap(dp);
vector<std::unordered_map<char, int>> mAllTmp;
for (auto& it : setPreMask)
{
std::unordered_map<char, int> tmp;
ParseMask(tmp, it);
mAllTmp.push_back(tmp);
}
}
return -1;
}
int GetSubMask(const std::unordered_map<char, int>& mPre, const std::unordered_map<char, int>& mCur)
{
int iSubMask = 0;
for (auto& cur : mCur)
{
auto pre = mPre.find(cur.first);
if (pre->second )
{
iSubMask += min(pre->second, cur.second) * m_vCharMul[cur.first-‘a’];
}
}
return iSubMask;
}
int MakeMask(const std::unordered_map<char, int>& nums)const
{
int iMask = 0;
for (const auto& mm : m_mTarget )
{
auto it = nums.find(mm.first);
if ((nums.end() != it) && ( it->second > 0 ))
{
iMask = iMask * (mm.second + 1) + min(mm.second, it->second);
}
else
{
iMask = iMask * (mm.second + 1);
}
}
return iMask;
}
void ParseMask(std::unordered_map<char, int>& nums, int iMask)
{
for (auto it = m_mTarget.begin(); it != m_mTarget.end(); ++it)
{
const int iMul = m_vCharMul[it->first-‘a’];
nums[it->first] = iMask / iMul;
iMask %= iMul;
}
}
void Init()
{
for (auto& ch : m_target)
{
m_mTarget[ch]++;
}
m_iTargetMask = MakeMask(m_mTarget);
int iMul = 1;
for (auto& it : m_mTarget)
{
iMul *= (it.second + 1);
}
for (auto it = m_mTarget.begin(); it != m_mTarget.end(); ++it)
{
iMul /= (it->second + 1);
m_vCharMul[it->first - ‘a’] = iMul;
}
}
std::string m_target;
int m_iTargetMask;
std::unordered_map<char, int> m_mTarget;
int m_vCharMul[26] ;
};

2023年1月第三版

class Solution {
public:
int minStickers(vector& stickers, string target) {
m_target = target;
Init();

	 vector<std::unordered_map<char, int>> mAllCharNum;
	 {
		 std::unordered_set<int> hasMask;
		 hasMask.insert(0);
		 for (int i = stickers.size() - 1; i >= 0; i--)
		 {
			 std::unordered_map<char, int> mCur;
			 for (auto& ch : stickers[i])
			 {
				 if (m_mTarget.count(ch))
				 {
					 mCur[ch]++;
				 }
			 }
			 const int iCurMask = MakeMask(mCur);
			 if (hasMask.count(iCurMask))
			 {
				 continue;
			 }
			 hasMask.insert(iCurMask);
			 mAllCharNum.push_back(mCur);
		 }
	 }
	
	 std::unordered_set<int> setPreMask,setHasDo;
	 setPreMask.insert(m_iTargetMask);
	 setHasDo.insert(m_iTargetMask);
	 for (int iOpeNum = 1; iOpeNum <= target.length(); iOpeNum++)
	 {
		 std::unordered_set<int> dp;
		 for (auto& preMask : setPreMask)
		 {
			 std::unordered_map<char, int> mPre;
			 ParseMask(mPre, preMask);
			 
			 char chFristNeedChar = 0;
			 for (auto& it : mPre)
			 {
				 if (it.second > 0)
				 {
					 chFristNeedChar = it.first;
					 break;
				 }
			 }

			 for (auto& mCur : mAllCharNum)
			 {
				 if ((0 == mCur.count(chFristNeedChar)) || (mCur[chFristNeedChar] <= 0 ))
				 {
					 continue;
				 }

				 const int iSubMask = GetSubMask(mPre, mCur);
				 if (0 == iSubMask)
				 {
					 continue;
				 }
				 const int iNewMask = preMask - iSubMask;
				 if (setHasDo.count(iNewMask))
				 {
					 continue;
				 }
				 if (iNewMask == 0 )
				 {
					 return iOpeNum;
				 }
				 setHasDo.insert(iNewMask);
				 dp.insert(iNewMask);
			 }
		 }
		 setPreMask.swap(dp);

		 vector<std::unordered_map<char, int>> mAllTmp;
		 for (auto& it : setPreMask)
		 {
			 std::unordered_map<char, int> tmp;
			 ParseMask(tmp, it); 
			 mAllTmp.push_back(tmp);
		 }
	 }
	 
	 
	 return -1;
 }
 int GetSubMask(const std::unordered_map<char, int>& mPre, const std::unordered_map<char, int>& mCur)
 {
	 int iSubMask = 0;
	 for (auto& cur : mCur)
	 {
		 auto pre = mPre.find(cur.first);
		 if (pre->second )
		 {
			 iSubMask += min(pre->second, cur.second) * m_vCharMul[cur.first-'a'];
		 }
	 }
	 return iSubMask;
 }
 int MakeMask(const std::unordered_map<char, int>& nums)const
 {
	 int iMask = 0;
	 for (const auto& mm : m_mTarget )
	 {
		 auto it = nums.find(mm.first);
		 if ((nums.end() != it) && ( it->second > 0 ))			
		 {
			 iMask = iMask * (mm.second + 1) + min(mm.second, it->second);
		 }
		 else
		 {
			 iMask = iMask * (mm.second + 1);
		 }
	 }
	 return iMask;
 }
 void ParseMask(std::unordered_map<char, int>& nums, int iMask)
 {
	 for (auto it = m_mTarget.begin(); it != m_mTarget.end(); ++it)
	 {
		 const int iMul = m_vCharMul[it->first-'a'];
		 nums[it->first] = iMask / iMul;
		 iMask %= iMul;
	 }
	
	
 }

 void Init()
 {
	 for (auto& ch : m_target)
	 {
		 m_mTarget[ch]++;
	 }
	 m_iTargetMask = MakeMask(m_mTarget);

	 int iMul = 1;
	 for (auto& it : m_mTarget)
	 {
		 iMul *= (it.second + 1);
	 }
	 for (auto it = m_mTarget.begin(); it != m_mTarget.end(); ++it)
	 {
		 iMul /= (it->second + 1);
		 m_vCharMul[it->first - 'a'] = iMul;
	 }
 }

 std::string m_target;
 int m_iTargetMask;
 std::unordered_map<char, int> m_mTarget;
 int m_vCharMul[26] ;

};

2023年1月 第4版

class CCTestHash
{
public:
std::size_t operator()(const std::unordered_map<char, int>& nums) const
{
return MakeMask(nums);
}
static int MakeMask(const std::unordered_map<char, int>& nums)
{
int iMask = 0;
for (const auto& mm : m_mTarget)
{
auto it = nums.find(mm.first);
if ((nums.end() != it) && (it->second > 0))
{
iMask = iMask * (mm.second + 1) + min(mm.second, it->second);
}
else
{
iMask = iMask * (mm.second + 1);
}
}
return iMask;
}

static void Init()
 {
	m_mTarget.clear();
	 for (auto& ch : m_target)
	 {
		 m_mTarget[ch]++;
	 }
	 m_iTargetMask = MakeMask(m_mTarget);

	 int iMul = 1;
	 for (auto& it : m_mTarget)
	 {
		 iMul *= (it.second + 1);
	 }
	 for (auto it = m_mTarget.begin(); it != m_mTarget.end(); ++it)
	 {
		 iMul /= (it->second + 1);
		 m_vCharMul[it->first - 'a'] = iMul;
	 }
 }

static std::string m_target;
static int m_iTargetMask;
static std::unordered_map<char, int> m_mTarget;
static int m_vCharMul[26];

};
std::string CCTestHash::m_target;
int CCTestHash::m_iTargetMask;
std::unordered_map<char, int> CCTestHash::m_mTarget;
int CCTestHash::m_vCharMul[26];

class Solution {
public:
int minStickers(vector& stickers, string target) {
CCTestHash::m_target = target;
CCTestHash::Init();

	 std::unordered_set<std::unordered_map<char, int>, CCTestHash> hasMask;
	 {			 
		 for (int i = stickers.size() - 1; i >= 0; i--)
		 {
			 std::unordered_map<char, int> mCur;
			 for (auto& ch : stickers[i])
			 {
				 if (m_testHask.m_mTarget.count(ch))
				 {
					 mCur[ch]++;
				 }
			 }
			 if (0 == mCur.size())
			 {
				 continue;
			 }
			 if (hasMask.count(mCur))
			 {
				 continue;
			 }
			 hasMask.insert(mCur);
		 }
	 }
	
	 std::unordered_set<std::unordered_map<char, int>, CCTestHash> setPreMask, setHasDo;
	 setPreMask.insert(CCTestHash::m_mTarget);
	 setHasDo.insert(CCTestHash::m_mTarget);
	 for (int iOpeNum = 1; iOpeNum <= target.length(); iOpeNum++)
	 {
		 std::unordered_set<std::unordered_map<char, int>, CCTestHash> dp;
		 for (auto& mPre : setPreMask)
		 {
			 char chFristNeedChar = 0;
			 for (auto& it : mPre)
			 {
				 if (it.second > 0)
				 {
					 chFristNeedChar = it.first;
					 break;
				 }
			 }

			 for (auto& mCur : hasMask)
			 {
				 if ((0 == mCur.count(chFristNeedChar)) || (mCur.find(chFristNeedChar)->second <= 0 ))
				 {
					 continue;
				 }

				 auto newMask = GetSubMask(mPre, mCur);
	
				 if (setHasDo.count(newMask))
				 {
					 continue;
				 }
				 if (CCTestHash::MakeMask(newMask) == 0)
				 {
					 return iOpeNum;
				 }
				 setHasDo.insert(newMask);
				 dp.insert(newMask);
			 }
		 }
		 setPreMask.swap(dp);
	 }
	 
	 
	 return -1;
 }
 std::unordered_map<char, int> GetSubMask(const std::unordered_map<char, int>& mPre, const std::unordered_map<char, int>& mCur)
 {
	 std::unordered_map<char, int> vRet;
	 for (auto& pre : mPre)
	 {
		 auto cur = mCur.find(pre.first);
		 if (mCur.end() != cur)
		 {
			 vRet[pre.first] = max(0, pre.second - cur->second);
		 }
		 else
		 {
			 vRet[pre.first] = pre.second;
		 }
	 }
	 return vRet;
 }

 /*
 int MakeMask(const std::unordered_map<char, int>& nums)const
 {
	 int iMask = 0;
	 for (const auto& mm : m_mTarget )
	 {
		 auto it = nums.find(mm.first);
		 if ((nums.end() != it) && ( it->second > 0 ))			
		 {
			 iMask = iMask * (mm.second + 1) + min(mm.second, it->second);
		 }
		 else
		 {
			 iMask = iMask * (mm.second + 1);
		 }
	 }
	 return iMask;
 }
 void ParseMask(std::unordered_map<char, int>& nums, int iMask)
 {
	 for (auto it = m_mTarget.begin(); it != m_mTarget.end(); ++it)
	 {
		 const int iMul = m_vCharMul[it->first-'a'];
		 nums[it->first] = iMask / iMul;
		 iMask %= iMul;
	 }
	
	
 }
 */
 CCTestHash m_testHask;

};

扩展阅读

视频课程

有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
https://edu.csdn.net/course/detail/38771

如何你想快

速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176

相关下载

想高屋建瓴的学习算法,请下载《喜缺全书算法册》doc版
https://download.csdn.net/download/he_zhidan/88348653

我想对大家说的话
闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。
子墨子言之:事无终始,无务多业。也就是我们常说的专业的人做专业的事。
如果程序是一条龙,那算法就是他的是睛

测试环境

操作系统:win7 开发环境: VS2019 C++17
或者 操作系统:win10 开发环境: VS2022 C++17
如无特殊说明,本算法用**C++**实现。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/333741.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

肇庆韶关异形件上门扫描服务龙岗3D抄数画图福田电脑抄数STL转STP

在当今的数字化时代&#xff0c;对于需要精确测量和设计的客户来说&#xff0c;CASAIM中科广电异形件上门扫描及抄数设计是一项非常重要的服务。这项服务不仅可以为客户提供高质量的测量和设计&#xff0c;还可以帮助他们减少时间和成本。 异形件是一种特殊的零件&#xff0c;…

力扣 | 11. 盛最多水的容器

双指针解法–对撞指针 暴力解法public int maxArea1(int[] height) {int n height.length;int ans 0;for (int i 0; i < n; i) {for (int j i 1; j < n; j) {int area Math.min(height[i], height[j]) * (j - i);ans Math.max(ans, area);}}return ans;}双指针解法…

java毕业设计 | springboot二手交易平台 闲置物品商城(附源码)

1&#xff0c;项目背景 1.1 当前的问题和困惑 随着社会发展&#xff0c;网上购物已经成为我们日常生活的一部分。但是&#xff0c;至今为止大部分电商平台都是从人们日常生活出发&#xff0c;出售都是一些日常用品比如&#xff1a;食物、服装等等&#xff0c;并未发现一个专注…

计算机网络-ACL实验

一、NAT实验配置 NAT实验配置 通过基本ACL匹配VLAN 10网段&#xff0c;然后在出口设备NAT转换只要匹配到VLAN10地址则进行转换。 核心交换机 # 配置VLAN和默认路由&#xff0c;配置Trunk和Access接口 interface Vlanif10ip address 192.168.10.254 255.255.255.0 # interface V…

源聚达科技:个人怎么开抖音店铺

随着互联网的发展&#xff0c;电商平台已经成为了人们购物的主要渠道之一。而抖音作为目前最受欢迎的短视频平台之一&#xff0c;也逐渐成为了一个新兴的电商市场。那么&#xff0c;个人怎么开抖音店铺呢?下面就来详细介绍一下。 第一步&#xff1a;注册抖音账号 首先&#xf…

深度学习进行数据增强(实战篇)

本文章是我在进行深度学习时做的数据增强,接着我们上期的划分测试集和训练集来做. 文章目录 前言 数据增强有什么好处&#xff1f; 一、构造数据增强函数 二、数据增强 总结 前言 很多人在深度学习的时候在对数据的处理时一般采用先数据增强在进行对训练集和测试集的划分,…

ORM Bee设计思想与功能思维导图

ORM Bee设计思想与功能思维导图 Bee&#xff0c;互联网新时代的Java ORM框架&#xff0c;支持Sharding&#xff1b;JDBC&#xff0c;Android&#xff0c;HarmonyOS&#xff1b;支持多种关系型数据库&#xff0c;还支持NoSQL的Cassandra&#xff0c;Mongodb等&#xff1b;更快、…

NVIDIA 大模型 RAG 分享笔记

文章目录 大语言模型在垂直领域落地的三个挑战&#xff1a;什么是 RAG以及为什么能解决大预言模型所带来的的这三个问题RAG 不是一项技术而是整体的 Pipeline非参数化 &#xff1a;数据库部分加载到数据库中检索阶段 提升检索效率的技术检索前&#xff1a;对query做处理use que…

redis缓存和本地缓存的应用设计

数据查询顺序 一级缓存&#xff1a;本地缓存 -》二级缓存&#xff1a;redis缓存 -》数据库 本地缓存和分布式缓存 本地缓存&#xff1a;基于jvm, 意思是程序放在哪&#xff0c;数据就存储在哪&#xff0c;不需要网络请求&#xff0c;特别快&#xff0c;但是需要占用jvm的内存…

Redis--Zset使用场景举例(滑动窗口实现限流)

文章目录 前言什么是滑动窗口zset实现滑动窗口小结附录 前言 在Redis–Zset的语法和使用场景举例&#xff08;朋友圈点赞&#xff0c;排行榜&#xff09;一文中&#xff0c;提及了redis数据结构zset的指令语法和一些使用场景&#xff0c;今天我们使用zset来实现滑动窗口限流&a…

Docker 仓库管理

Docker 仓库管理 仓库&#xff08;Repository&#xff09;是集中存放镜像的地方。以下介绍一下 Docker Hub。当然不止 docker hub&#xff0c;只是远程的服务商不一样&#xff0c;操作都是一样的。 Docker Hub 目前 Docker 官方维护了一个公共仓库 Docker Hub。 大部分需求…

Oracle命令大全

文章目录 1. SQL*Plus命令&#xff08;用于连接与管理Oracle数据库&#xff09;2. SQL数据定义语言&#xff08;DDL&#xff09;命令3. SQL数据操作语言&#xff08;DML&#xff09;命令4. PL/SQL程序块5. 系统用户管理6. 数据备份与恢复相关命令1. SQL*Plus命令&#xff08;用…

java-log4j日志冲突解决

一、概述 java日志框架较多&#xff0c;其中主流的slf4j和commons-logging是日志接口&#xff0c;log4j、log4j2和logback是真正的日志实现库。 二、具体库单独使用 2.1 log4j <dependency><groupId>log4j</groupId><artifactId>log4j</artifa…

CentOS stream 9配置网卡

CentOS stream9的网卡和centos 7的配置路径&#xff1a;/etc/sysconfig/network-scripts/ifcfg-ens32不一样。 CentOS stream 9的网卡路径&#xff1a; /etc/NetworkManager/system-connections/ens32.nmconnection 方法一&#xff1a; [connection] idens32 uuid426b60a4-4…

【鸿蒙4.0】详解harmonyos开发语言ArkTS

文章目录 一.什么是ArkTS&#xff1f;1.ArkTS的背景2.了解js&#xff0c;ts&#xff0c;ArkTS的演变js(Javascript)Javascript的简介Javascript的特点 ts(Typescript)ArkTS 二. ArkTS的特点 一.什么是ArkTS&#xff1f; 1.ArkTS的背景 如官方文档所描述&#xff0c;ArkTS是基…

《Linux C编程实战》笔记:Linux信号介绍

信号是一种软件中断&#xff0c;它提供了处理一种异步事件的方法&#xff0c;也是进程惟一的异步通信方式。在Linux系统中&#xff0c;根据POSIX标准扩展的信号机制&#xff0c;不仅可以用来通知某进程发生了什么事&#xff0c;还可以给进程传递数据。 信号的来源 信号的来源…

广东金牌电缆:法大大电子合同助力业务风险管控

广东金牌电缆集团股份有限公司&#xff08;以下简称“广东金牌电缆”&#xff09;成立于2013年&#xff0c;现为广东省电线电缆重点生产企业、广东省守合同重信用单位、国家专精特新小巨人企业、国家高新技术企业&#xff0c;拥有自主商标“夺冠”&#xff0c;“夺冠”商标被评…

一文读懂「Fine-tuning」微调

一、什么是微调&#xff1f; 1. 什么是微调&#xff1f; 微调是指在预训练模型&#xff08;Pre-trained model&#xff09;的基础上&#xff0c;针对特定任务或数据领域&#xff0c;对部分或全部模型参数进行进一步的训练和调整&#xff08;Fine Tune&#xff09;。预训练模型…

File 类的用法和 InputStream, OutputStream 的用法

1.File类的用法 下面就用几个简单的代码案例来熟悉File类里面函数的用法&#xff1a; public class IODemo1 {public static void main(String[] args) throws IOException {File f new File("./test2.txt");//File f new File("C:/User/1/test.txt");S…

redis数据安全(二)数据持久化 RDB

目录 一、RDB快照持久化 原理 二、RDB快照持久化配置&#xff08;redis.conf&#xff09;&#xff1a; 三、触发RDB备份&#xff1a; 1、自动备份&#xff0c;需配置备份规则&#xff1a; 2、手动执行命令备份&#xff08;save | bgsave&#xff09;&#xff1a; 3、flus…