09-02 HDU_Steps2.3 递推,高精度,卡特兰数 HDU1002 HDU103 HDU1018 HDU1130 HDU1131 HDU1133 HDU1134 HDU1267

这几天一直在做题,没有抽出时间来写总结..HDU的Steps做到了第六节了,温习的一些知识点,也学到了不少新东西..加油,争取在省赛前能尽量多学一些!!奋斗


STEPS 2.3的题目基本都是递推或者大数,当然这两者常常是结合在一起的,因为递推尤其像卡特兰数之类的,很容易得到一个非常大的数值,一般简单的只有加法的用CPP就可以了,懒的去启慢吞吞的Eclipse..当然了,复杂的大数运算还是用JAVA比较好一些,毕竟容易写,也不会出错..


2.3.1 HDU1002 A+BⅡ入门大数

主要代码

for(int i=0;i<l1;i++)i1[i]=s1[l1-i-1]-'0';
		for(int i=0;i<l2;i++)i2[i]=s2[l2-i-1]-'0';
		 
		for(int i=0;g!=0||i<l1||i<l2;i++){
			ans[i]=(i1[i]+i2[i]+g)%10;
			g=(i1[i]+i2[i]+g)/10;
			l3++; 
		}


2.3.2 HDU1063 Exponentiation 高精度幂运算

Java水过,不过要注意几个方法的调用,去掉末尾和开头0,并且转成字符串显示,否则会用默认的科学计数法显示

import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.Scanner;
public class Main {
	public static void main(String[] args) {
		new Main();
	}
	public Main(){
		BigDecimal r;
		int n;
		Scanner sc=new Scanner(System.in);
		while(sc.hasNext()){
			r=sc.nextBigDecimal();
			n=sc.nextInt();
                        //去末尾0,转成字符串
			String st=r.pow(n).stripTrailingZeros().toPlainString();
			int ind=0;
			while(true){
				if(st.charAt(ind)!='0')break;
				ind++;
			}
			st=st.substring(ind);
			System.out.println(st);
		}
	}
}


2.3.3 HDU1018 Big Number 求N!位数

一个数的位数可以用Log(N)+1得出(底为10),证明很简单,N可以写成a*10^n,a为小于10的实数,而N的位数就是n+1

f(N!)=Log(N!)+1=Log(N)+log(N-1)+...Log(1)+1

#include <cstdio>
#include <cmath>
using namespace std;
/*
	F(N)=log(N)/log(10)+1 数与位数的关系
	F(N!)=(log(N)+log(N-1)+...+log(1))/log(10)+1 
*/
int solve(int n){
	double r=0;
	for(int i=1;i<=n;i++)r+=log(i);
	return (int)(r/log(10))+1;	
}
int main(){
	int cas,n;
	scanf("%d",&cas);
	while(cas--){
		scanf("%d",&n);
		printf("%d\n",solve(n));
	}
}

2.3.4 HDU1130 How Many Trees

的个数=左树的总数*右树的总数,即d[n]=d[0]d[n-1]+d[1]d[n-2]+....+d[n-1][0] 

卡特兰数,Java水过,可以求出公式.我没有求,直接用递归写的.跟记忆化搜索的思想有些类似

import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.Scanner;

public class Tes {
	public static void main(String[] args) {
		new Tes();
	}
	//d[n]=d[0]d[n-1]+d[1]d[n-2]+....+d[n-1][0]
	BigInteger d[]=new BigInteger[105];
	public Tes(){
		for(int i=0;i<105;i++){
			d[i]=new BigInteger("0");
		}
		Scanner sc=new Scanner(System.in);
		while(sc.hasNext()){
			int n=sc.nextInt();
			System.out.println(dp(n));
		}
	}
	public BigInteger dp(int n){
		if(n==0)return g(1);
		if(d[n].signum()>0)return d[n];
		for(int i=0;i<=n-1;i++){
			d[n]=d[n].add(dp(i).multiply(dp(n-i-1)));
		}
		return d[n];
	}
	
	public BigInteger g(int x){
		return new BigInteger(new Integer(x).toString());
	}
	public BigInteger gs(String s){
		return new BigInteger(s);
	}
}




2.3.5 HDU1133 Buy The Ticket 递推

很经典的一道题,有人说这是卡特兰数,我也不知道是怎么用卡特兰数去解释,总之我是用递推做的

d[m][n]=m*d[m-1][n]+n*d[m][n-1] 其中M表示的是前M+N人中50元的个数,N是100元的个数,显然M+N的状态可以由M+N-1得来,而加的这个人可能是50元的,也可能是100元的.考虑50元,因为不同的排列算不同的结果,所以加的这个人可以跟前面的M-1个人交换位置,序列是不变的,所以要乘上一个M(加上放在第M+N个位置上的情况)

注意边界情况,m<n时d[m][n]=0,n=0时f[m][n]=m!

高精度,我是用Java实现的,附上主要代码


	public BigInteger fac(int x){
		if(x==0)return BigInteger.ONE;
		return fac(x-1).multiply(g(x));
	}
	public BigInteger dp(int m,int n){
		if(n==0)return fac(m);
		if(d[m][n]!=null)return d[m][n];
		
		d[m][n]=new BigInteger("0");
		if(m<n)d[m][n]=BigInteger.ZERO;
		else{
			if(m>n)d[m][n]=dp(m-1,n).multiply(g(m));
			d[m][n]=d[m][n].add(dp(m,n-1).multiply(g(n)));
		}
		return d[m][n];
	}
	public BigInteger g(int x){
		return new BigInteger(new Integer(x).toString());
	}


2.3.6 HDU1131 Count the Trees

跟上面的1130是一样的,只不过要乘上N!


2.3.7 HDU1134 Game of Connections

卡特兰数 F(n)=F(0)F(n-1)+F(1)F(n-2)+..F(n-1)F(0)


2.3.8 HDU1267 下沙的沙子有几粒

和上面的1133差不多,只不过不需要考虑排列的情况

所以d[i][j]=d[i-1][j]+d[i][j-1]; d[i][j]=0(i<j) 其中i表示H个数,j表示D个数



你可能感兴趣的:(java,String,null,Integer,Class,import)