给定一个正整数 N,找到并返回 N 的二进制表示中两个连续的 1 之间的最长距离。
如果没有两个连续的 1,返回 0 。
示例:
输入:22
输出:2
解释:
22 的二进制是 0b10110 。
在 22 的二进制表示中,有三个 1,组成两对连续的 1 。
第一对连续的 1 中,两个 1 之间的距离为 2 。
第二对连续的 1 中,两个 1 之间的距离为 1 。
答案取两个距离之中最大的,也就是 2 。
class Solution {
public int binaryGap(int N) {
int max = 0;
int tmp = 0;
while((N & 1) != 1) N >>= 1;
while(N > 0){
if((N & 1) == 1){
max = Math.max(max, tmp);
tmp = 0;
}
tmp++;
N >>= 1;
}
return max;
}
}
给定两个句子 A 和 B 。 (句子是一串由空格分隔的单词。每个单词仅由小写字母组成。)
如果一个单词在其中一个句子中只出现一次,在另一个句子中却没有出现,那么这个单词就是不常见的。
返回所有不常用单词的列表。
您可以按任何顺序返回列表。
示例:
输入:A = “this apple is sweet”, B = “this apple is sour”
输出:[“sweet”,“sour”]
总结
“可以理解成拼接字符串A+B,然后返回拼接后的字符串中只出现过一次的单词。”
真是醍醐灌顶
class Solution {
public String[] uncommonFromSentences(String A, String B) {
Map<String, Integer> count = new HashMap();
for (String word: A.split(" "))
count.put(word, count.getOrDefault(word, 0) + 1);
for (String word: B.split(" "))
count.put(word, count.getOrDefault(word, 0) + 1);
List<String> ans = new LinkedList();
for (String word: count.keySet())
if (count.get(word) == 1)
ans.add(word);
return ans.toArray(new String[ans.size()]);
}
}
爱丽丝和鲍勃有不同大小的糖果棒:A[i] 是爱丽丝拥有的第 i 块糖的大小,B[j] 是鲍勃拥有的第 j 块糖的大小。
因为他们是朋友,所以他们想交换一个糖果棒,这样交换后,他们都有相同的糖果总量。(一个人拥有的糖果总量是他们拥有的糖果棒大小的总和。)
返回一个整数数组 ans,其中 ans[0] 是爱丽丝必须交换的糖果棒的大小,ans[1] 是 Bob 必须交换的糖果棒的大小。
如果有多个答案,你可以返回其中任何一个。保证答案存在。
示例:
输入:A = [2], B = [1,3]
输出:[2,3]
Fastest Code
时空转移大法用到了极致
class Solution {
public int[] fairCandySwap(int[] A, int[] B) {
int sumA = 0;
int sumB = 0;
for(int a : A)
sumA += a;
int arr[] = new int[100001];
for(int b : B){
sumB += b;
arr[b] = 1;
}
int diff = (sumA - sumB) / 2;
for(int a : A){
if((a-diff) >= 0 && (a-diff) <= 100000 && arr[a-diff] == 1)
return new int[]{
a, a-diff};
}
return null;
}
}
Demo Code
class Solution {
public int[] fairCandySwap(int[] A, int[] B) {
Arrays.sort(A);
Arrays.sort(B);
int sumA = 0, sumB = 0;
for(int i = 0; i < A.length; i++)
sumA += A[i];
for(int i = 0; i < B.length; i++)
sumB += B[i];
int[] res = new int[2];
int diff = sumB - sumA;
int le = 0, ri = B.length-1;
for(int i = 0; i < A.length; i++) {
for(int j = B.length-1; j >= 0; j--) {
if(B[j] - A[i] == diff/2) {
res[0] = A[i];
res[1] = B[j];
break;
}
if(B[j] - A[i] < diff/2) continue;
}
}
return res;
}
}
在 N * N 的网格上,我们放置一些 1 * 1 * 1 的立方体。
每个值 v = grid[i][j] 表示 v 个正方体叠放在对应单元格 (i, j) 上。
请你返回最终形体的表面积。
总结
仔细想想还是很简单
class Solution {
public int surfaceArea(int[][] grid) {
int result=0;
for (int i = 0; i < grid.length; i++) {
for (int j = 0; j < grid[i].length; j++) {
if (grid[i][j]!=0)
//假设每个v=grid[i][j]都是独立的。
result+=grid[i][j]*4+2;
//减去面贴在一起的情况
if (i>0)
result -= Math.min(grid[i-1][j], grid[i][j])*2;
if (j>0)
result -= Math.min(grid[i][j-1], grid[i][j])*2;
}
}
return result;
}
}
对于字符串 S 和 T,只有在 S = T + … + T(T 与自身连接 1 次或多次)时,我们才认定 “T 能除尽 S”。
返回字符串 X,要求满足 X 能除尽 str1 且 X 能除尽 str2。
示例 1:
输入:str1 = “ABCABC”, str2 = “ABC”
输出:“ABC”
示例 2:
输入:str1 = “LEET”, str2 = “CODE”
输出:""
class Solution {
public String gcdOfStrings(String str1, String str2) {
if (!(str1 + str2).equals(str2 + str1))
return "";
return str1.substring(0, gcd(str1.length(), str2.length()));
}
public int gcd(int a, int b) {
return a == 0 ? b : gcd(b % a, a);
}
}
排排坐,分糖果。
我们买了一些糖果 candies,打算把它们分给排好队的 n = num_people 个小朋友。
给第一个小朋友 1 颗糖果,第二个小朋友 2 颗,依此类推,直到给最后一个小朋友 n 颗糖果。
然后,我们再回到队伍的起点,给第一个小朋友 n + 1 颗糖果,第二个小朋友 n + 2 颗,依此类推,直到给最后一个小朋友 2 * n 颗糖果。
重复上述过程(每次都比上一次多给出一颗糖果,当到达队伍终点后再次从队伍起点开始),直到我们分完所有的糖果。注意,就算我们手中的剩下糖果数不够(不比前一次发出的糖果多),这些糖果也会全部发给当前的小朋友。
返回一个长度为 num_people、元素之和为 candies 的数组,以表示糖果的最终分发情况(即 ans[i] 表示第 i 个小朋友分到的糖果数)。
示例:
输入:candies = 7, num_people = 4
输出:[1,2,3,1]
解释:
第一次,ans[0] += 1,数组变为 [1,0,0,0]。
第二次,ans[1] += 2,数组变为 [1,2,0,0]。
第三次,ans[2] += 3,数组变为 [1,2,3,0]。
第四次,ans[3] += 1(因为此时只剩下 1 颗糖果),最终数组变为 [1,2,3,1]。
class Solution {
public int[] distributeCandies(int candies, int num_people) {
int[] res = new int[num_people];
int idx = 0, ex = 1;
while(candies > 0) {
if(idx >= num_people) {
idx = 0;
ex += num_people;
}
res[idx] += Math.min(candies, idx+ex);
candies = Math.max(0, candies-idx-ex);
idx++;
}
return res;
}
}
class Solution {
public int[] distributeCandies(int candies, int num_people) {
int curr_give = 0;
int[] res = new int[num_people];
while (candies > 0) {
res[curr_give % num_people] += Math.min(++curr_give, candies);
candies -= curr_give;
}
return res;
}
}
给你一个由一些多米诺骨牌组成的列表 dominoes。
如果其中某一张多米诺骨牌可以通过旋转 0 度或 180 度得到另一张多米诺骨牌,我们就认为这两张牌是等价的。
形式上,dominoes[i] = [a, b] 和 dominoes[j] = [c, d] 等价的前提是 ac 且 bd,或是 ad 且 bc。
在 0 <= i < j < dominoes.length 的前提下,找出满足 dominoes[i] 和 dominoes[j] 等价的骨牌对 (i, j) 的数量。
示例:
输入:dominoes = [[1,2],[2,1],[3,4],[5,6]]
输出:1
class Solution {
public int numEquivDominoPairs(int[][] dominoes) {
int[] map = new int[100];
int res = 0;
for(int i = 0; i < dominoes.length; i++){
int m = dominoes[i][0];
int n = dominoes[i][1];
int k = m > n ? m * 10 + n : n * 10 + m;
map[k] ++;
}
for(int i = 0; i < 100; i++)
res += map[i] * (map[i] - 1) / 2; // 取 n 里面有多少个 2
return res;
}
}
我们来定义一个函数 f(s),其中传入参数 s 是一个非空字符串;该函数的功能是统计 s 中(按字典序比较)最小字母的出现频次。
例如,若 s = “dcce”,那么 f(s) = 2,因为最小的字母是 “c”,它出现了 2 次。
现在,给你两个字符串数组待查表 queries 和词汇表 words,请你返回一个整数数组 answer 作为答案,其中每个 answer[i] 是满足 f(queries[i]) < f(W) 的词的数目,W 是词汇表 words 中的词。
示例:
输入:queries = [“cbd”], words = [“zaaaz”]
输出:[1]
解释:查询 f(“cbd”) = 1,而 f(“zaaaz”) = 3 所以 f(“cbd”) < f(“zaaaz”)。
class Solution {
public int[] numSmallerByFrequency(String[] queries, String[] words) {
int[] num = new int[12];
for(int i = 0; i < words.length; i++)
num[helper(words[i])]++;
for(int i = 9; i >= 0; i--)
num[i] += num[i+1];
int[] res = new int[queries.length];
for(int i = 0; i < queries.length; i++)
res[i] = num[helper(queries[i])+1];
return res;
}
public int helper(String s){
int[] map = new int[26];
for(char c : s.toCharArray())
map[c-'a']++;
for(int m : map){
if(m != 0) return m;
}
return 0;
}
}
来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/fair-candy-swap
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。