【LetMeFly】831.隐藏个人信息
力扣题目链接:https://leetcode.cn/problems/masking-personal-information/
给你一条个人信息字符串 s
,可能表示一个 邮箱地址 ,也可能表示一串 电话号码 。返回按如下规则 隐藏 个人信息后的结果:
电子邮件地址:
一个电子邮件地址由以下部分组成:
- 一个 名字 ,由大小写英文字母组成,后面跟着
- 一个
'@'
字符,后面跟着 - 一个 域名 ,由大小写英文字母和一个位于中间的
'.'
字符组成。'.'
不会是域名的第一个或者最后一个字符。
要想隐藏电子邮件地址中的个人信息:
- 名字 和 域名 部分的大写英文字母应当转换成小写英文字母。
- 名字 中间的字母(即,除第一个和最后一个字母外)必须用 5 个
"*****"
替换。
电话号码:
一个电话号码应当按下述格式组成:
- 电话号码可以由 10-13 位数字组成
- 后 10 位构成 本地号码
- 前面剩下的 0-3 位,构成 国家代码
- 利用
{'+', '-', '(', ')', ' '}
这些 分隔字符 按某种形式对上述数字进行分隔
要想隐藏电话号码中的个人信息:
- 移除所有 分隔字符
- 隐藏个人信息后的电话号码应该遵从这种格式:
"***-***-XXXX"
如果国家代码为 0 位数字"+*-***-***-XXXX"
如果国家代码为 1 位数字"+**-***-***-XXXX"
如果国家代码为 2 位数字"+***-***-***-XXXX"
如果国家代码为 3 位数字
"XXXX"
是最后 4 位 本地号码
示例 1:
输入:s = "LeetCode@LeetCode.com" 输出:"l*****e@leetcode.com" 解释:s 是一个电子邮件地址。 名字和域名都转换为小写,名字的中间用 5 个 * 替换。
示例 2:
输入:s = "AB@qq.com" 输出:"a*****b@qq.com" 解释:s 是一个电子邮件地址。 名字和域名都转换为小写,名字的中间用 5 个 * 替换。 注意,尽管 "ab" 只有两个字符,但中间仍然必须有 5 个 * 。
示例 3:
输入:s = "1(234)567-890" 输出:"***-***-7890" 解释:s 是一个电话号码。 共计 10 位数字,所以本地号码为 10 位数字,国家代码为 0 位数字。 因此,隐藏后的电话号码应该是 "***-***-7890" 。
示例 4:
输入:s = "86-(10)12345678" 输出:"+**-***-***-5678" 解释:s 是一个电话号码。 共计 12 位数字,所以本地号码为 10 位数字,国家代码为 2 位数字。 因此,隐藏后的电话号码应该是 "+**-***-***-7890" 。
提示:
s
是一个 有效 的电子邮件或者电话号码- 如果
s
是一个电子邮件:8 <= s.length <= 40
s
是由大小写英文字母,恰好一个'@'
字符,以及'.'
字符组成
- 如果
s
是一个电话号码:10 <= s.length <= 20
s
是由数字、空格、字符'('
、')'
、'-'
和'+'
组成
方法一:字符串解析
首先判断字符串中是否存在'@'
或'.'
,以此来区分字符串是电话还是邮箱。
邮箱
找到字符串中'@'
的下标,将答案字符串加上“字符串的第一个字符的小写形式”和“'@'
下标的前一个下标对应字符的小写形式”
之后从'@'
开始遍历字符串,若是大写字母则在答案字符串中加上其小写形式,否则直接加上原字母。
电话
首先统计字符串中数字的个数
c
n
t
N
u
m
cntNum
cntNum。依据
c
n
t
N
u
m
−
10
cntNum - 10
cntNum−10是0还是1还是2还是3分别在答案中加上打码号码的前缀***-***-
或+*-***-***-
或+**-***-***-
或+***-***-***-
之后解析出字符串中的最后四个数字并添加到答案的末尾。
- 时间复杂度 O ( l e n ( s ) ) O(len(s)) O(len(s))
- 空间复杂度 O ( 1 ) O(1) O(1)(或 O ( l e n ( s ) ) O(len(s)) O(len(s)))
AC代码
C++
class Solution {
public:
string maskPII(string s) {
bool isemail = false;
for (char c : s) {
if (c == '@' || c == '.') {
isemail = true;
break;
}
}
string ans;
if (isemail) {
ans += tolower(s[0]);
ans += "*****";
int locAt = 0;
while (s[locAt] != '@') {
locAt++;
}
ans += tolower(s[locAt - 1]);
while (locAt < s.size()) {
if (isupper(s[locAt])) {
ans += tolower(s[locAt]);
}
else {
ans += s[locAt];
}
locAt++;
}
}
else {
int cntNum = 0;
for (char c : s) {
cntNum += isdigit(c);
}
if (cntNum == 10) {
ans = "***-***-";
}
else if (cntNum == 11) {
ans = "+*-***-***-";
}
else if (cntNum == 12) {
ans = "+**-***-***-";
}
else {
ans = "+***-***-***-";
}
int cntTail = 0;
char tail[4];
for (int i = s.size() - 1; cntTail < 4; i--) {
if (isdigit(s[i])) {
tail[cntTail++] = s[i];
}
}
for (int i = 3; i >= 0; i--) {
ans += tail[i];
}
}
return ans;
}
};
Python
class Solution:
def maskPII(self, s: str) -> str:
if '@' in s: # email
locAt = s.find('@')
return s[0].lower() + '*****' + s[locAt - 1].lower() + s[locAt:].lower()
else:
cntNum = sum(c.isdigit() for c in s)
ans = ["***-***-", "+*-***-***-", "+**-***-***-", "+***-***-***-"][cntNum - 10]
allDigital = "".join(c if c.isdigit() else "" for c in s)
for i in range(len(allDigital) - 4, len(allDigital)):
ans += allDigital[i]
return ans
What’s more
写完后看了看官解,太sao了
transform、正则
class Solution {
public:
vector<string> country = {"", "+*-", "+**-", "+***-"};
string maskPII(string s) {
string res;
int at = s.find("@");
if (at != string::npos) {
transform(s.begin(), s.end(), s.begin(), ::tolower);
return s.substr(0, 1) + "*****" + s.substr(at - 1);
}
s = regex_replace(s, regex("[^0-9]"), "");
return country[s.size() - 10] + "***-***-" + s.substr(s.size() - 4);
}
};
class Solution:
def maskPII(self, s: str) -> str:
at = s.find('@')
if at >= 0:
return (s[0] + "*" * 5 + s[at - 1:]).lower()
s = "".join(i for i in s if i.isdigit())
return ["", "+*-", "+**-", "+***-"][len(s) - 10] + "***-***-" + s[-4:]
同步发文于CSDN,原创不易,转载请附上原文链接哦~
Tisfy:https://letmefly.blog.csdn.net/article/details/129893329