Cracking the coding interview--Q1.8

原文:

Assume you have a method isSubstring which checks if one word is a substring of another. Given two strings, s1 and s2, write code to check if s2 is a rotation of s1 using only one call to isSubstring ( i.e., “waterbottle” is a rotation of “erbottlewat”).

译文:

假设你有一个isSubstring函数,可以检测一个字符串是否是另一个字符串的子串。 给出字符串s1和s2,只使用一次isSubstring就能判断s2是否是s1的旋转字符串, 请写出代码。旋转字符串:"waterbottle"是"erbottlewat"的旋转字符串。


一开始看到这道题目的时候,想到最笨的方法每次s1旋转一个位置,再判断是否是子串,不是则继续旋转,这样效率很低而且不符合题目只调用一次isSubstring的要求.......

想不到好的方法解决本题,一下是参考大神的,思路很巧妙。


解题思路

题目说我们使用一次isSubstring函数就可以判断s2是否是s1的旋转字符串, 如果从原始字符串s1和s2直接入手肯定不行,因为它们根本不存在子串关系。 如果不断地旋转字符,然后调用isSubstring,又需要调用多次的isSubstring。 而且通过旋转字符再判断,可以直接用等号判断,根本用不上isSubstring。

既然如此,我们就要考虑去改变原始字符串。要判断a串是否是b串的子串, 一般情况下都会有b串长度大于a串,长度相等的话就直接判断它们是不是相等的串了。 我们可以考虑把串s1变长,然后调用一次isSubstring判断s2是否是s1变长后的子串, 如果是,就得出s2是s1的旋转字符串。s1怎么变长呢?无非就是s1+s1或是s1+s2, s2一定是s1+s2的子串,因此这样做没有任何意义。而s1+s1呢? 我们就上面的例子进行讨论:s1=waterbottle,s2=erbottlewat. 则:

s1 + s1 = waterbottlewaterbottle

很容易可以发现,s1+s1其实是把s1中每个字符都旋转了一遍,而同时保持原字符不动。 比如waterbottle向右旋转2个字条应该是:terbottlewa,但如果同时保持原字符不动, 我们得到的就是waterbottlewa,而terbottlewa一定是waterbottlewa的子串, 因为waterbottlewa只是在terbottlewa的基础上再加上一条原字符不动的限制。 因此s1+s1将包含s1的所有旋转字符串,如果s2是s1+s1的子串,自然也就是s1 的旋转字符串了。


package chapter_1_arraysandstring;

import java.util.Scanner;

/**
 * 假设你有一个isSubstring函数,可以检测一个字符串是否是另一个字符串的子串。 
 * 给出字符串s1和s2,只使用一次isSubstring就能判断s2是否是s1的旋转字符串, 请写出代码。
 * 旋转字符串:"waterbottle"是"erbottlewat"的旋转字符串。
 * 
 * @author LiangGe
 *
 */
public class Question_1_8 {

	/**
	 * 判断str2是否是str1的子串
	 * 
	 * string indexOf(str) 返回str字符或者子串在string的第一次出现位置
	 * 如果不存在则返回-1,存在则返回出现位置
	 */
	public static boolean isSubString(String str1, String str2) {
		if(str1.indexOf(str2) != -1) {
			return true;
		} else {
			return false;
		}
	}
	
	/**
	 * 判断str2是否是str1的旋转子串
	 * 由于只能使用一次判断isSubString, str1+str1 就拼接成每个字符都旋转
	 * 此时判断str2是否是 str1+str1的子串就能判断出是否是旋转子串
	 */
	public static boolean isRotation(String str1, String str2) {
		String strMerge = str1 + str1;
		return isSubString(strMerge, str2);
	}
	
	public static void main(String args[]) {
		Scanner scanner = new Scanner(System.in);
		while(scanner.hasNext()) {
			String curStrLine = scanner.nextLine();
			String str[] = curStrLine.split(" ");
			if(str.length > 2) {
				System.out.println("输入合适错误");
				break;
			}
			String str1 = str[0];
			String str2 = str[1];
			
			if(isRotation(str1, str2)) {
				System.out.format("%s 是  %s 的旋转字符串\n", str2, str1);
			} else {
				System.out.format("%s 不是  %s 的旋转字符串\n", str2, str1);
			}
		}
	}
}


参考:http://hawstein.com/posts/1.8.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值