今天主要是了解数组和方法,下面主要是展示一下方法的概念和调用以及技巧,数组的引用调用命名等规则
方法的概念
多次反复编写明显是不合适的,而结构化编程中的模块化在Java中的体现就是自定义方法。
需求:输出1-100之间所有素数
public class Test1 {
// 特殊的方法,用于表示程序的执行起始点,也就意味着不能写错
public static void main(String[] args) {
for(int i=2;i<101;i++){
if(abc(i)){ //调用处传入的数据是实参
System.out.println(i);
}
}
System.out.println(abc(10,0));//语法报错,因为abc方法在定义时,只说明一个参数,但是调用处是两个参数,不对应。
}
//定义方法,命名规则为要求见名知意,一般建议首字母小写,大写字母分词。否则必须添加注释说明
//判断kk是否为素数
//boolean表示这个方法的返回值必须为boolean,也就是只能true/false,其它类型则出错
//方法名称:标识符的命名规则
//()中的内容为方法参数,就是调用方法时必须传递的数据
//int表示传入的参数必须为int类型,否则报错
//至于参数的名称可以理解为占位符,称之为形参,调用时会被传递的数据所替代
public static boolean abc(int kk){//自定义方法,用于封装一个具体的处理过程,当需要使用这个处理过程时只需要通过名称就可以直接调用。实际参数的传递是通过实参和形参一一对应实现的(位置对应)。
boolean res=true;
for(int i=2;i<=kk/2;i++){
if(kk%i==0){
res=false;
break;
}
}
return res;//return表示立即终止当前程序的运行,并返回调用处
}
}
方法可以理解为一个命名的代码块,通过名称就可以重复使用这段代码,而不需要反复书写,可以达到
代码重用的目的
方法可以有参数,也可以没有参数;方法可以有返回值,也可以没有返回值[必须声明返回值为void]
方法定义的具体位置没有关系,可以先写调用【会有报错】,然后定义方法也可以;先写定义后调用也
可以。
调用方式常见的有三种
单独调用。这种方式无法使用方法的返回值。格式:方法名称(参数值);
调用是show(1,5),这个show方法没有返回值,所以不能接收返回值,否则语法报错【int
k=show(1,5);】;如果有返回值,可以不接受返回值
public static int pp(){...}
int k=pp(); //语法正确
pp(); //语法正确
打印调用。这种方式可以将方法的返回值直接打印。格式:System.out.println(方法名称(参数值)); 注意
使用sysout方法调用需要有返回值
赋值调用。这种方式可以将方法的返回值赋值给一个变量,注意变量的数据类型必须和方法的返回值类
型对应。格式:数据类型 变量名称= 方法名称(参数值)
重名问题
变量的名称是否可以与方法名称重名?可以。
public class Test03 {
public static void main(String[] args) {
pp(); //方法名称允许相同
pp(11);
}
public static void pp() {
System.out.println("void...pp()");
}
public static void pp(int k) { //两个同名的方法参数不同才可以重名定义
System.out.println("void...ppp()");
}
//方法参数不同有3种情况:类型不同、数量不同、顺序不同[不是方法参数名称]
}
public class Test03 {
public static void main(String[] args) {
pp();
pp(11);
}
public static void pp() {
System.out.println("void...pp()");
}
public static void pp(int k) {
System.out.println("void...pp()");
}
public static void pp(int k1,String s1) {
System.out.println("void...ppp()");
}
public static void pp(int s1,int k1) {
System.out.println("void...ppp0()");
}
}
两个不同的方法中,能否各自有一个重名的变量?可以,而且各个方法中的临时变量之间没有任何关
系。
- main方法中有2个临时变量begin和end,add方法中也有两个临时变量begin和end,这个方法中
的临时变量没有任何关系,各自生存在不同的方法中
参数传递
- 形式参数:在定义方法的时候,写在小括号之内的变量,就叫形式参数。实际上在方法定义中起到占位符的作用,会在方法调用时被传递过来的实际值所替代
- 实际参数:在调用方法的时候,真正传入方法里的数据,叫做实际参数。
圆括号中的实参列表为调用方法时实际传入的实际参数,称为实参列表。声明方法时圆括号中的参数称
为形式参数,形式参数和实际参数在数据类型和个数上一定要匹配
注意:调用方法时形式参数和实际参数的个数和顺序必须一致,数据类型也必须相同。
两条规则
对于基本类型来说,形式参数的操作【不会】影响实际参数。值是单向传递
对于引用类型来说,形式参数的操作【会】影响实际参数。
对任何语言来说,方法或者函数解决了需要重复使用的代码的次数
递归调用
递归调用指在方法执行过程中允许出现直接或者间接的该方法本身的调用
例如;斐波那契数列Fibonacci sequence,又称黄金分割数列,以兔子繁殖为例子而引入,故又称为“兔子数列”,指的是这样一个数列:1、1、2、3、5、8、13、21、34…(以递推的方法定义:F(1)=1,F(2)=1, F(n)=F(n-1)+F(n-2)(n>=3,n∈N*))
public class Test05 {
public static void main(String[] args) {
int months = 0;
Scanner sc = new Scanner(System.in);
while (true) {
System.out.println("月份:");
String ss = sc.nextLine();
try {
months = Integer.parseInt(ss);
if (months > 0) break;
System.out.println("请重新输入月份数!");
} catch (Exception e) {
System.out.println("请输入合法的月份数!");
}
}
int num = tongji(months);
System.out.println(months + "月后的兔子数为:" + num);
}
public static int tongji(int months) {
if (months > 0) {
if (months == 1 || months == 2) return 1;
return tongji(months - 1) + tongji(months - 2);
}
return 0;
}
}
递归调用比较符合正常人的思维方式,但是相当的浪费内存,所以如果能使用其他方式解决就不要使用递归。
循环和递归对比:
- 递归:易于理解、速度慢、存储空间大
- 循环:不易于理解、速度快、存储空间小
汉诺塔问题:
- 梵天创造世界的时候做了三根金刚石柱子,在一根柱子上从下往上按照大小顺序摞着64片黄金盘。大梵天命令婆罗门把圆盘从下面开始按大小顺序重新摆放在另一根柱子上。并且规定,在小圆盘上不能放大圆盘,在三根柱子之间一次只能移动一个圆盘移动次数是f(n).显然f(1)=1,f(2)=3,f(3)=7,且f(k+1)=2*f(k)+1
public class 汉诺塔问题 {
public static void main(String[] args) {
System.out.println(move(3));
}
public static long move(int n) {
if (n == 1)
return 1;
return 2 * move(n - 1) + 1;
}
}
递归调用的特征
Java语言支持方法的递归调用
- 使用递归调用时必须可以逐渐接近结束点,不能发散
- 递归调用比较符合正常人的思维方式,但是相当的浪费内存,所以如果能使用其他方式解决就
不要使用递归
数组
数组是表示多个相同类型变量的集合(在一个数组中所存放的所有元素的类型必须一致),可以使用共同的名字引用它
- 属于复杂数据类型
由类型相同的元素组成的有顺序的数据集合
Java数组是固定的不能扩展[长度一旦声明,不能修改]
可以存储基本数据类型或对象
数组可以定义为任意数据类型,并且可分为一维数组或多维数组
一维数组
- 一维数组实质上是相同类型变量的列表。要创建一个数组必须首先定义数组变量所需的类型。通的一
- 维数组的声明格式是:type var-name[ ];
int[] arr1 = new int[10]; // int[]或arr2[]都是用于声明是数组类型,int用于声明每个
元素都是int类型
int arr2[] = new int[20]; //要求使用数组之前必须先声明所需空间大小,即存储的元素个
数,一旦声明则不能修改
int[] brr;
brr[0]=123; //语法报错
- 数组中的元素必须类型相同,不能存储不同类型的元素,除非使用Object[]
- 数组中的元素必须类型相同,不能存储不同类型的元素,除非使用Object[]
使用数组之前,必须先定义后使用,定义的方式: int[] arr或者int arr[]
- 声明数组变量后,必须进行初始化操作,也就是定义数组的长度 arr=new int[5],这里表示开启可
以存放5个int类型数据的数组,这些元素时连续存放的 - 简单类型的数组初始化长度后,每个元素都有默认值
- 创建数组后,每个元素都有一个默认值,如果针对的是引用类型,则默认值为null;如果是简单类
型中byte short int long float double,则默认值为0,如果char类型默认值为\0,如果boolean类
型,则默认false
在Java中允许直接赋值初始
- int[] arr={1,2,3,4,5,6,7}; 注意在于int[]中不能写具体的对应长度
- 也可以写成new int[]{1,2,3,4,5,6,7},但是int[]中不能有长度值
int[] arr= {1,2,3,4,5,6};
System.out.println(arr[3]);
int[] brr=new int[] {1,2,3,4,5,67};//注意不能修改int[]为int[6],否则语法报错
System.out.println(brr[3]);
声明一个int型的一维数组:int number[];
[ ]在字节左边,对后面所有变量都有影响char[] s, t
int arr[],brr[]; //等价于int[] arr,brr;
arr=new int[5];
brr=new int[3];
System.out.println();
数组虽然声明了变量类型。但不存在实际的数值,它的值为null。为了使数组number 成为实际的、物理上存在的整型数组,你必须用运算符new 来为其分配地址并且把它赋给number
int[] arr;//声明整型数组变量arr
System.out.println(arr); //语法错误,必须先声明后使用,先赋初值后使用
运算符new 是专门用来分配内存的运算符,格式为:array-var = new type[size]; 开启出来的空间是连续,可以通过下标进行快速定位
例如上面的代码: int[] kk=new int[10];
length属性
数组对象中有个属性length表示的是数组的长度
int[] arr=new int[] {1,2,3,4,5,6};
System.out.println(arr.length); //获取数组中的元素个数
使用运算符new来分配数组,必须指定数组元素的类型和数组元素的个数。用运算符new分配数组后,数组中的元素将会被自动初始化为一个值,这个值具体是什么和类型相关。如果是数值类型【整数、浮点数、char类型】其中自动初始值为0,如果boolean类型则自动初始值为false
比如s = newchar[26];程序,生成一个包含26个字符值的数组,并将每个数组元素的初始值设为‘\u0000’
数组元素的访问
针对数组中的元素可以通过下标进行访问,访问的语法为: 数据变量的名称[标]
数组中的所有元素将被初始化为零。接下来可以给数组中每个元素赋值
kk[0]=10;
kk[1]+=13;
......
这样给数组赋值显得不够现实
声明的同时初始化
在数组定义时指定元素的初始值,即称为数组的初始化
int a[]={1,2,3,4,5};
声明一个数组并初始数组内容
int[] score = {90, 85, 55, 94, 77}; //也可以写作new int[]{90, 85, 55, 94, 77},但是不允许写成new
int[5]{90, 85, 55, 94, 77},语法报错
int[] arr=new int[] {2,5,3,1,9,0};
for(int i=0;i<arr.length;i++)
System.out.println("数组arr的第"+i+"个元素值为:"+arr[i]);
格式化输出
int[] score = {90, 85, 55, 94, 77}; //也可以写作new int[]{90, 85, 55, 94, 77}
for(int i = 0; i < score.length; i++)
System.out.printf("score[%d] = %d\n", i, score[i]);
//前面的格式符号%x的个数应该和后面参数一致,如果出现了参数不足时则会报错
int[] arr = new int[] { 2, 5, 3, 1, 9, 0 };
for (int i = 0; i <= arr.length; i++) //注意=的问题
System.out.printf("%s[%d] = %d\n", "arr", i, arr[i]); //没有语法错误,但
是执行代码会有报错
错误分析:在控制台上可以看到报错信息ArrayIndexOutOfBoundsException示数组越界,就是访问一个不存在的元素
当取出数组元素的时,数组下标索引不能超出数组范围[0到length-1],如果超出数组范围会发生
ArrayIndexOutOfBoundsException。例如声明一个数组的元素个数为5,而去取了它的第6个,那么程序就会提示下标越界的错误
可以使用动态的方式来宣告数组长度,而不用在程序中事先决定数组大小
public class Test08 {
public static void main(String[] args) {
Scanner sc=new Scanner(System.in);
int len=sc.nextInt();
int[] arr=initArr(len);//可以根据输入的参数创建不同长度的数组
System.out.println(arr.length);
}
public static int[] initArr(int length) { //if(length<1)
//return new int[5];
return new int[length];//初始化数组长度时参数不同小于0,但是允许等于0
}
}
循环打印出小写字母a~z
char c='a';
for(int i=0;i<26;i++)
System.out.println((char)(c+i));
char[] arr=new char[26];
for(int i=0;i<26;i++)
arr[i]=(char)(c+i);
//输出数组中的数据
for(int i=0;i<arr.length;i++)
System.out.println(arr[i]);
//foreach结构,是用于遍历集合写法的语法糖,是for循环的简化写法
for(char temp:arr)
System.out.println(temp);
将同一个对象指定给两个参考名称
运行后则brr变量和arr变量都指向同一个数组,就是前面创建的数组,其中存放了26个字符的数组
char[] arr=new char[26];
for(int i=0;i<arr.length;i++)
arr[i]=(char)('a'+i);
char[] brr=arr;//运行结果是变量brr和变量arr指代同一个位置
brr[brr.length-1]='\u9e00'; //修改brr中的内容实际上也修改了arr的内容
for(char temp:arr)
System.out.println(temp);