目录

1 问题描述

2 解决方案

2.1 蛮力轮询法

2.2 素数相乘法

2.3 位运算法


1 问题描述

给定一长字符串A和一短字符串B。请问,如何最快地判断出短字符串B中的所有字符是否都在长字符串A中?请编写一个判断函数实现此功能。

为简单起见,假设输入的字符串只包含小写英文字母。下面举几个例子。

(1)如果字符串A是”abcd”,字符串B是”bad”,答案是包含,因为字符串B中的字母都在字符串A中,或者说B是A的真子集。

(2)如果字符串A是”abcd”,字符串B是”bce”,答案是不包含,因为字符串B中的字母e不在字符串A中。

(3)如果字符串A是”abcd”,字符串B是”aab”,答案是包含,因为字符串B中的字母a包含在字符串A中。


2 解决方案

2.1 蛮力轮询法

判断字符串B中的字符是否都在长字符串A中,最直观的思路则是:轮询B中每一个字符,逐个与A中每个字符进行比较,看是否都在字符串A中。

具体代码如下:

package com.liuzhen.string_1;

public class StringContain {
//方法1:蛮力轮询
/*
* 参数A:给定的长字符串A
* 参数B:给定的短字符串B
* 函数功能:如果B中所有字符在A中均出现过,则返回true,否则返回false
*/
public boolean bruteContain(String A,String B){
boolean result = false;
char[] arrayA = A.toCharArray();
char[] arrayB = B.toCharArray();
int testLen = 0; //用于计算B中与A匹配字符个数
for(int i = 0;i < arrayB.length;i++){
for(int j = 0;j < arrayA.length;j++){
if(arrayB[i] == arrayA[j]){
testLen++;
break;
}
}
}
if(testLen == arrayB.length) //当B个所有字符均和A中字符匹配时
result = true;
return result;
} public static void main(String[] args){
StringContain test = new StringContain();
String A = "abcd";
String B = "aab";
if(test.bruteContain(A, B))
System.out.println("使用蛮力轮询法得到结果:A字符串包含B字符串");
else
System.out.println("使用蛮力轮询法得到结果:A字符串不包含B字符串");
}
}

运行结果:

使用蛮力轮询法得到结果:A字符串包含B字符串

2.2 素数相乘法

思路如下:

(1)按照从小到大的顺序,用26个素数分别代替长字符串A中的所有字母。

(2)遍历字符串A,求得A中所有字母对于的素数的乘积。

(3)遍历短字符串B,判断上一步得到的乘积能否被B中的字母对于的素数整除。

(4)输出结果。

具体代码如下:

package com.liuzhen.string_1;

public class StringContain {

    //方法2:素数相乘
/*
* 参数A:给定的长字符串A
* 参数B:给定的短字符串B
* 函数功能:如果B中所有单个字符对应素数能被A中所有字符对应素数之积整除,则返回true,否则返回false
*/
public boolean primeContain(String A,String B){
boolean result = true;
int[] primes = {2,3,5,7,11,13,17,19,23,29,31,37,41,43,47,53,59,61,67,71,73,79,83,89,97,101};
long mulSum = 1;
char[] arrayA = A.toCharArray();
char[] arrayB = B.toCharArray();
for(int i = 0;i < arrayA.length;i++)
mulSum *= primes[arrayA[i] - 'a'];
for(int j = 0;j < arrayB.length;j++){
int temp = (int) (mulSum % primes[arrayB[j] - 'a']);
if(temp != 0){ //此时,A中不包含arrayB[j]字符
result = false;
return result;
}
}
return result;
} public static void main(String[] args){
StringContain test = new StringContain();
String A = "abcd";
String B = "aab";
if(test.placeContain(A, B))
System.out.println("使用素数相乘法得到结果:A字符串包含B字符串");
else
System.out.println("使用素数相乘法得到结果:A字符串不包含B字符串");
}
}

运行结果:

使用素数相乘法得到结果:A字符串包含B字符串

2.3 位运算法

用位运算(26位整数表示)为长字符串A计算出一个“签名”(利用位或运算),再逐一将短字符串B中的字符放到A中进行查找(PS:利用位与运算)。

具体代码如下:

package com.liuzhen.string_1;

public class StringContain {

    //方法3:位运算法
/*
* 参数A:给定的长字符串A
* 参数B:给定的短字符串B
* 函数功能:如果B中每个字符进行处理后的对应二进制值与A中所有字符进行处理对应二进制值的求或运算
* ,在单独进行求与运算,一旦出现0,则返回false,否则返回true
*/
public boolean placeContain(String A,String B){
boolean result = true;
char[] arrayA = A.toCharArray();
char[] arrayB = B.toCharArray();
int hash = 0;
for(int i = 0;i < arrayA.length;i++)
hash |= (1 << (arrayA[i] - 'a')); //|=意思是位或运行,即将hash的二进制与|=后数字进行或运算结果赋值给hash
for(int j = 0;j < arrayB.length;j++){
if((hash & (1 << (arrayB[j] - 'a'))) == 0){ //进行与运算,即当A中不包含arrayB[j]字符时
result = false;
return result;
}
}
return result;
} public static void main(String[] args){
StringContain test = new StringContain();
String A = "abcd";
String B = "aab";
if(test.placeContain(A, B))
System.out.println("使用位运算法得到结果:A字符串包含B字符串");
else
System.out.println("使用位运算法得到结果:A字符串不包含B字符串");
}
}

运行结果:

使用位运算法得到结果:A字符串包含B字符串

算法笔记_024:字符串的包含(Java)的更多相关文章

  1. 算法笔记_134:字符串编辑距离(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 给定一个源串和目标串,能够进行如下操作: 在任意位置上插入一个字符: 替换掉任意字符: 删除任意字符. 写一个程序,实现返回最小操作次数,使得对源串 ...

  2. 算法笔记_023:拓扑排序(Java)

    目录 1 问题描述 2 解决方案 2.1 基于减治法实现 2.2 基于深度优先查找实现 1 问题描述 给定一个有向图,求取此图的拓扑排序序列. 那么,何为拓扑排序? 定义:将有向图中的顶点以线性方式进 ...

  3. 算法之暴力破解和kmp算法 判断A字符串是否包含B字符串

    我们都知道java中有封装好的方法,用来比较A字符串是否包含B字符串 如下代码,contains,用法是 str1.contains(str2), 这个布尔型返回,存在返回true,不存在返回fals ...

  4. 算法笔记_028:字符串转换成整数(Java)

    1 问题描述 输入一个由数字组成的字符串,请把它转换成整数并输出.例如,输入字符串“123”,输出整数123. 请写出一个函数实现该功能,不能使用库函数. 2 解决方案 解答本问题的基本思路:从左至右 ...

  5. 算法笔记_025:字符串的全排列(Java)

    目录 1 问题描述 2 解决方案 2.1 递归实现 2.2 字典序排列实现   1 问题描述 输入一个字符串,打印出该字符串的所有排列.例如,输入字符串”abc”,则输出有字符’a’,’b’,’c’所 ...

  6. 算法笔记_022:字符串的旋转(Java)

    目录 1 问题描述 2 解决方案 2.1 蛮力移位 2.2 三步反转 1 问题描述 给定一个字符串,要求将字符串前面的若干个字符移到字符串的尾部.例如,将字符串“abcdef”的前3个字符‘a’.‘b ...

  7. 算法笔记_228:信用卡号校验(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 当你输入信用卡号码的时候,有没有担心输错了而造成损失呢?其实可以不必这么担心,因为并不是一个随便的信用卡号码都是合法的,它必须通过Luhn算法来验证 ...

  8. 算法笔记_225:数字密码发生器(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 在对银行账户等重要权限设置密码的时候,我们常常遇到这样的烦恼:如果为了好记用生日吧,容易被破解,不安全:如果设置不好记的密码,又担心自己也会忘记:如 ...

  9. 算法笔记_220:猜算式(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 看下面的算式: □□ x □□ = □□ x □□□ 它表示:两个两位数相乘等于一个两位数乘以一个 三位数. 如果没有限定条件,这样的例子很多. 但 ...

随机推荐

  1. android 网络

    韩梦飞沙  韩亚飞  313134555@qq.com  yue31313  han_meng_fei_sha android - async - http 安卓 异步 超文本传输协议 xUtil a ...

  2. android intent 传递 二进制数据

    韩梦飞沙  韩亚飞  313134555@qq.com  yue31313  han_meng_fei_sha 实现序列化.

  3. poj2117 Electricity

      试题描述 求一个图删除一个点之后,联通块最多有多少. 输入 多组数据.第一行两个整数 P,C 表示点数和边数.接下来 C 行每行两个整数 p1,p2,表示 p1 与 p2 有边连接,保证无重边.读 ...

  4. HDU 4619 Warm up 2 最大独立集

    Warm up 2 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=4619 Description Some 1×2 dominoes are pla ...

  5. Codeforces Beta Round #4 (Div. 2 Only) D. Mysterious Present 记忆化搜索

    D. Mysterious Present 题目连接: http://www.codeforces.com/contest/4/problem/D Description Peter decided ...

  6. hdu 3340 Rain in ACStar 线段树区间等差数列更新

    Rain in ACStar Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://acm.hust.edu.cn/problem/show/1385 ...

  7. React-如何在jsx中自动补全标签(vscode)

    痛点:  React库最近的增长趋势很明显, 很多朋友都在选择学习, 很多公司也在选择使用React栈. 但在使用React库写代码的时候, 有一个很让人苦恼的问题, 就是标签在jsx语法中不能自动补 ...

  8. DLL Injection and Hooking

    DLL Injection and Hooking http://securityxploded.com/dll-injection-and-hooking.php Three Ways to Inj ...

  9. 软件包管理 rpm yum apt-get dpkg

    http://blog.csdn.net/ljq1203/article/details/7401616

  10. jdk1.8 foreach

    lambda 表达式效率非常低,测试代码可以看到大概3~5倍的差距 遍历Map的方式有很多,通常场景下我们需要的是遍历Map中的Key和Value,那么推荐使用的: public static voi ...