目录

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. KMP的小结

    http://www.cnblogs.com/kuangbin/archive/2012/08/14/2638803.html 如果有哪一天不记得模板了就去看看大神的 .  非常清晰易懂.

  2. 【插头DP】BZOJ1814-Formula

    [题目大意] 给出一个m*n的矩阵里面有一些格子为障碍物,求经过所有非障碍格子的哈密顿回路个数. [思路] 最典型的插头DP.分为三种情况: (1)当前格子既没有上插头也没有左插头. 如果下边和右边都 ...

  3. LinkCutTree 总结

    最近学习了LinkCutTree,总结一下. LinkCutTree是一种数据结构(是Tree Decomposition中的一种),她维护的一般是无向图(一个森林),支持连边.删边.链修改.链查询( ...

  4. git提交时”warning: LF will be replaced by CRLF“提示

    今天把项目做完之后,然后用Git准备提交代码的时候,遇到warning: LF will be replaced by CRLF警告. 当时在网上查了资料,发现很多的解决办法都是:修改git全局配置, ...

  5. Codeforces Beta Round #8 B. Obsession with Robots 暴力

    B. Obsession with Robots 题目连接: http://www.codeforces.com/contest/8/problem/B Description The whole w ...

  6. uoj 48 核聚变反应强度 次小公因数

    [UR #3]核聚变反应强度 Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://uoj.ac/problem/48 Description 著名核 ...

  7. Codeforces Round #299 (Div. 2) B. Tavas and SaDDas 水题

    B. Tavas and SaDDas Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/535/p ...

  8. JVM堆内存的分代

    虚拟机的堆内存共划分为三个代:年轻代(Young Generation).年老代(Old Generation)和持久代(PermanentGeneration).其中持久代主要存放的是Java类的类 ...

  9. [置顶] getopt_long函数基本用法-linux

    一.感性认识: [c-sharp]  view plain copy   #include <stdio.h> #include <getopt.h> char * l_opt ...

  10. bosondata/chrome-prerender: Render JavaScript-rendered page as HTML/PDF/mhtml/png/jpeg using headless Chrome

    bosondata/chrome-prerender: Render JavaScript-rendered page as HTML/PDF/mhtml/png/jpeg using headles ...