JAVA程序设计:扰乱字符串(LeetCode:87)

给定一个字符串 s1,我们可以把它递归地分割成两个非空子字符串,从而将其表示为二叉树。

下图是字符串 s1 = "great" 的一种可能的表示形式。

    great
   /    \
  gr    eat
 / \    /  \
g   r  e   at
           / \
          a   t
在扰乱这个字符串的过程中,我们可以挑选任何一个非叶节点,然后交换它的两个子节点。

例如,如果我们挑选非叶节点 "gr" ,交换它的两个子节点,将会产生扰乱字符串 "rgeat" 。

    rgeat
   /    \
  rg    eat
 / \    /  \
r   g  e   at
           / \
          a   t
我们将 "rgeat” 称作 "great" 的一个扰乱字符串。

同样地,如果我们继续交换节点 "eat" 和 "at" 的子节点,将会产生另一个新的扰乱字符串 "rgtae" 。

    rgtae
   /    \
  rg    tae
 / \    /  \
r   g  ta  e
       / \
      t   a
我们将 "rgtae” 称作 "great" 的一个扰乱字符串。

给出两个长度相等的字符串 s1 和 s2,判断 s2 是否是 s1 的扰乱字符串。

示例 1:

输入: s1 = "great", s2 = "rgeat"
输出: true
示例 2:

输入: s1 = "abcde", s2 = "caebd"
输出: false

思路:

方法1:递归搞搞

class Solution {
    public boolean isScramble(String s1, String s2) {        
    	HashMap<String,Integer> memoization=new HashMap<>();
    	return isCrambleRecursion(s1,s2,memoization);
    }
    public boolean isCrambleRecursion(String s1,String s2,HashMap<String,Integer>memoization)
    {
    	int ret=memoization.getOrDefault(s1+"#"+s2, -1);
    	if(ret==1)
    		return true;
    	else if(ret==0)
    		return false;
    	if(s1.length()!=s2.length())
    	{
    		memoization.put(s1+"#"+s2, 0);
    		return false;
    	}
    	if(s1.equals(s2))
    	{
    		memoization.put(s1+"#"+s2, 1);
    		return true;
    	}
    	
    	int[] letters=new int[26];
    	for(int i=0;i<s1.length();i++)
    	{
    		letters[s1.charAt(i)-'a']++;
    		letters[s2.charAt(i)-'a']--;
    	}
    	for(int i=0;i<26;i++)
    		if(letters[i]!=0)
    		{
    			memoization.put(s1+"#"+s2, 0);
    			return false;
    		}
    	for(int i=1;i<s1.length();i++)
    	{
    		if(isScramble(s1.substring(0, i),s2.substring(0, i)) && isScramble(s1.substring(i),s2.substring(i)))
    		{
    			memoization.put(s1+"#"+s2, 1);
    			return true;
    		}
    		if(isScramble(s1.substring(0, i),s2.substring(s2.length()-i)) && isScramble(s1.substring(i),s2.substring(0, s2.length()-i)))
    		{
    			memoization.put(s1+"#"+s2, 1);
    			return true;
    		}
    	}
    	memoization.put(s1+"#"+s2, 0);
    	return false;
    }
}

方法二:动态规划

class Solution {
    public boolean isScramble(String s1, String s2) {  
    	
    	if(s1.length()!=s2.length())
    		return false;
    	if(s1.equals(s2))
    		return true;
    	
    	int[] letters=new int[26];
    	for(int i=0;i<s1.length();i++)
    	{
    		letters[s1.charAt(i)-'a']++;
    		letters[s2.charAt(i)-'a']--;
    	}
    	for(int i=0;i<26;i++)
    		if(letters[i]!=0)
    			return false;
    	int length=s1.length();
    	boolean[][][] dp=new boolean[length+1][length][length];
    	for(int len=1;len<=length;len++)
    		for(int i=0;i+len<=length;i++)
    			for(int j=0;j+len<=length;j++)
    			{
    				if(len==1)
    					dp[len][i][j]=s1.charAt(i)==s2.charAt(j);
    				else
    				{
    					for(int q=1;q<=len;q++)
    					{
    						dp[len][i][j]=dp[q][i][j] && dp[len-q][i+q][j+q] || dp[q][i][j+len-q] && dp[len-q][i+q][j];
    						if(dp[len][i][j])
    							break;
    					}
    				}
    			}
    	return dp[length][0][0];
    }
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
简要介绍<br>JAVA CLASS文件加密工具是一款专门为保护您的JAVA源代码而设计的软件。传统的JAVA代码保护方式通常是扰乱生成的CLASS文件,从而降低反编译生成的源代码的可读性;有的保护工具甚至能生成部分废代码(垃圾代码),使得反编译后的源码在重新编译时出现编译错误,不能直接生成CLASS文件。这种方法具有一定的效果,但是不能彻底保护您的JAVA源代码。 <br><br>JAVA CLASS文件加密工具对CLASS文件进行加密保护,加密密钥高达256位(bit,即:32字节),并采用多重加密的算法,既安全又高效。加密后的CLASS文件不可能被破解;反编译工具对加密后的CLASS文件无能为力,根本就不能处理! <br><br>运行方式<br>运行时,加密后的CLASS文件要能正常加载,需要使用我们提供的动态库hidea.dll。执行java时带上参数<br>-agentlib:<动态库文件所在路径>\hidea<br>注意:不要加文件后缀.dll,直接使用文件的名字部分(classloader)!<br><br>举例说明:例如,本加密工具安装在c:\hideasoft\java_protect,执行加密后的CLASS文件的命令行如下:<br>java -agentlib:c:\hideasoft\java_protect\hidea <您的CLASS类及参数><br>应用场合<br>独立的应用程序(Application,自定义main方法),运行java时,带上参数-agentlib:<所在路径>\hidea <br>Tomcat等JAVA Web Server,修改启动脚本,把执行java的命令行加上参数-agentlib:<所在路径>\hidea <br>JBOSS等JAVA Application Server(应用服务器),修改启动脚本,把执行java的命令行加上参数-agentlib:<所在路径>\hidea <br>适用环境操作系统:Windows 98/2000/XP 等Windows系统 <br>JDK:1.5.0及以上版本

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值