资源描述
第5章 实验 – 数组
5.1实验目标
(1) 理解在Java中如何声明、分配和使用(多维)数组
(2) 学习数组排序和查找的基本算法
(3) 了解JAVA Reflection API 和其他常用类
5.2实验说明
本章实验教程将介绍数组(array)的使用,学习Java数组相关的类Array和Arrays。简单介绍Java反射(Reflection)的用法。
以下的实验包括4种类型, 每种类型都用括号里面的字母表示:
D - 例程, 表示这是一个例子, 要求练习者阅读指南和代码;
I - 交互式练习, 练习者完成实验指定的简单任务,如修改部分代码, 观察程序运行时的变化等;
W - 热身练习, 练习者的编程工作量逐渐加大。
P - 完整编程 ,要求练习者根据要求,完成完整的JAVA程序。
5.3实验准备
从本实验教程光盘中拷贝Lab05文件目录到本地磁盘, 如D: 盘。Lab05文件目录中将包含本次实验所需的所有资料。 Lab05的相关资料也可以从本实验教程的网站下载:http://javaLab/lab05.zip
将Lab04中的MathUtility.java(程序清单4-6)程序拷贝到Lab05目录下。
5.4实验任务
实验5.4.1: (D)数组的基本概念
Java变量(variable)可以存储程序的数据,每个变量有标识、类型和作用域。使用同种类型和作用域的多个数据时,我们通常把这些数据存放在一个数据结构而不是单个的变量中。最常见的数据结构是数组(array)。一个数组中能够存储的数据个数是固定的,并且存放的数据类型相同。尽管数组是一个对象,但是数组的用法与基本数据类型相似。
图5.1可以帮助我们理解数组的结构。
图5.1 长度为8的数组
这是一个有8个元素的数组,数组中的每个元素有各自的值。图中每个元素上方的数字表明了元素的索引(index)。数组中的元素总是使用索引引用的。第一个元素的索引总是为0。重复,第一个元素的索引总是为0,而最后一个元素的索引总是“数组长度-1”。上图的数组中,最后一个元素的索引是 8-1 = 7。
我们可以创建任何数据类型的数组。下面来看一个整型数组的例子。
编译并运行ExamScoresl.java,输入0到100之间的一组整数为命令行参数。假设输入的这组整数就是考试成绩,这个程序将显示原始成绩和调整后的成绩。成绩调整的规则是:将最高成绩上调到100分,计算调整的幅度,再根据幅度相应调整其他成绩。
程序清单5-1:ExamScores1.java
// Given a sequence of exam scores as
// command-line arguments, displays table
// of exam scores and curved scores.
public class ExamScores1 {
public static void main(String[] commandLineArgs)
{
if ( commandLineArgs.length == 0 ) {
System.out.print("Given a sequence of exam scores ");
System.out.println("entered as command line arguments,");
System.out.print("this program will curve them up ");
System.out.println("by raising the highest to 100 and ");
System.out.println("raising all others by the same amount.");
System.exit(0);
} // if no command-line arguments
// Named constant for width of displayed columns:
final int columnWidth = 5;
// Named constant for the number of scores:
final int numberOfScores = commandLineArgs.length;
// Declare reference to array
// which will store exam scores:
int[] scores;
// Instantiate (create) the array itself,
// i.e. allocate space to store the exam scores:
scores = new int[numberOfScores];
// Read scores into array and determine maximum:
int maximumScore = 0;
for ( int i = 0; i < scores.length; i++ ) {
scores[i] = Integer.parseInt(commandLineArgs[i]);
maximumScore = Math.max(maximumScore, scores[i]);
} // for i
// Determine how many points to curve up all scores:
int curveUp = 100 - maximumScore;
// Output table of original scores and curved scores
for ( int i = 0; i < scores.length; i++ ) {
// Print original score in a right-justified column:
String scoreText = Integer.toString(scores[i]);
for ( int j = scoreText.length(); j < columnWidth; j++ )
System.out.print(" ");
System.out.print(scoreText);
// Calculate curved score;
int curvedScore = scores[i] + curveUp;
// Print curved score in a right-justified column;
String curvedScoreText = Integer.toString(curvedScore);
for ( int k = curvedScoreText.length(); k < columnWidth; k++ )
System.out.print(" ");
System.out.println(curvedScoreText);
} // for i
} // method main(String[])
} // class ExamScores1
大家可能已经发现,ExamScoresl.java源代码中main方法的命令行参数数组的名字与惯常不同,这里使用了commandLineArgs。其实使用什么名字作为命令行参数的数组名并没有关系,只要这个名字在整个main方法中保持一致就行。在一个具体的程序中,我们通常把命令行参数数组取名为args,把命令行参数个数的变量取名为argv,命令行参数数组是一个字符串数组。
ExamScoresl.java程序中需要跟踪所有的成绩,因此将考试成绩存放在一个int型数组中。创建数组的三个步骤是:(1)声明一个数组引用变量;(2)实例化(创建)数组本身,也就是为数组分配内存空间;(3)初始化数组中的每个变量。
1. 数组的声明
数组变量的声明和其他Java变量的声明类似,声明数组的语法模板:
DataType ArrayName [ConstIntExpression];
或
DataType [ConstIntExpression] ArrayName;
DataType是数据类型,即数组中元素的数据类型;ArrayName是合法的变量;[ ]符号中的ConsIntExpression指明数组的大小,即数组中元素的个数。
程序ExamScoresl.java中,声明int数组变量:
int [] scores;
另外一种声明数组的方式:
int scores[];
这两种数组声明方式在语法上都是正确的,都能被编译器接受。编者个人喜欢第一种(int[] scores)声明方式,因为它更清晰地说明了数组变量的名字是scores,而不是scores[];scores被声明为一个指向整型数组的指针。然而,更普遍使用的是第二种方式,尤其在Java官方文档中更为明显,因为这种方式与C++中的声明方式比较接近。
2. 实例化数组
实例化scores数组的语句如下:
scores = new int[numberOfScores];
实例化数组,即为数组元素分配内存空间,是通过new操作符实现的。new操作符后是数据类型以及元素个数。元素个数放在[]运算符中。下面是一些例子:
counts = new int[5];
names = new String[100];
3. 数组元素的初始化
将scores数组中的元素在循环中初始化的语句如下:
for ( int i = 0; i < scores.length; i++ ) {
scores[i] = Integer.parseInt(commandLineArgs[i]);
scores.length是数组元素的个数,就象commandLineArgs.length是数组commandLineArgs的元素个数一样。数组元素的范围从0到length - 1。
4. 数组声明、实例化和初始化的更多解释
声明和实例化数组的常见形式如下例所示:
int[] scores;
scores = new int[numberOfScores];
声明和实例化数组可以合并成一个语句:
int[] scores = new int[numberOfScores];
还可以定义初始化列表(Initializer List)来初始化数组元素:
int[] numbers = {4, 3, 2, 1};
String[] name = {"Zhang", "Wang", "Li"};
数组的长度是由{}中的元素个数隐含定义的,numbers数组的长度为4,name数组的长度为3。当数组的声明和创建分开时,仍然可以使用初始化列表,只是这种方式不太简练:
int[] numbers;
numbers = new int[]{4, 3, 2, 1};
再次强调一下,数组变量是一个内存位置的名字,这个内存位置存储的不是数组本身,而是数组在内存中存放的地址。因此,数组变量存放了一个指向数组的指针。当数组变量第一次声明时,它并没有真正指向一个数组。数组声明只是开辟一个存放指针的空间。之后,当使用关键字new实例化数组并将其指定给一个数组变量时,该数组变量就存放了一个指向数组的指针。
范例5-1程序的顶端有这样一个声明:
final int columnWidth = 5;
这个声明看上去象一个变量的声明,不同的是在最前面有一个关键字final。关键字final意味着columnWidth被声明为一个常量。常量的值在第一次初始化之后就不能改变。
为什么要费心去定义常量,而不在代码中直接使用常量的值呢?原因在于方便程序员修改程序,程序员只需查看代码顶端并修改常量的值,而不需要修改该常量的值出现的每行代码。
实验5.4.2: (I)访问数组元素
引用数组中的元素,使用 [ ] 操作符。范例5-1中,访问scores数组元素:
int score = scores[0]; // 得到第一个元素
int score = scores[3]; // 得到第四个元素
[ ]中的索引可以是整数,或者整型变量。索引的取值范围在0到数组元素个数-1之间。数组变量与数组中索引变量的区别在于,数组变量代表整个数组,数组中的索引变量代表了数组中的元素在数组中的位置。
通常我们会使用for循环语句来操作数组元素,见范例5-2。
程序清单5-2:AccessArrayWithForLoop.java
public class AccessArrayWithForLoop {
public static void main(String[] args) {
String[] months = { "Jan", "Feb", "Mar", "Apr", "May", "Jun",
"July", "Aug", "Sep", "Oct", "Nov", "Dec"};
int[] sizes = {31,28,31,30,31,30,31,31,30,31,30,31};
for(int i = 0; i < months.length; i++ ) {
System.out.println(months[i] + " has " + sizes[i] + " days.");
}
}
}
注意for循环中的变量i,i的取值在0到months.length-1之间。试着把源代码
for(int i = 0; i < months.length; i++ )
改成:
for(int i = 0; i < = months.length; i++ )
重新编译、运行程序。回答下面的问题:
代码可以正常编译吗?为什么?
___________________________________________________________
程序可以正常执行吗?为什么?
___________________________________________________________
实验5.4.3: (I)查找最大数
GreatestNumber.java(范例5-3)将对话框输入的3个数字保存在数组中,计算出其中的最大值。
程序清单5-3:GreatestNumber.java
import javax.swing.JOptionPane;
public class GreatestNumber {
public static void main(String[] args) {
int[] num = new int[10];
int counter;
int max = 0;
int totalnumber = 3;
// Prompt a user to enter numbers
for(counter = 0; counter < totalnumber; counter++) {
num[counter] = Integer.parseInt ( JOptionPane.showInputDialog (
"Enter numbers until " + totalnumber + " numbers are entered"));
// Compute the greatest number up to this point
if((counter == 0)||(num[counter] > max))
max = num[counter];
}
// Display the greatest number.
JOptionPane.showMessageDialog(null,
"The number with the greatest value is " + max);
}
}
现在,参照GreatestNumber.java写一个新的程序SmallestNumber.java, 找出3个数中的最小值。
实验5.4.4: (W)练习使用数组
写一个类似于ExamScores1.java的程序ExamScores2.java,从命令行参数输入学生成绩,没有输入命令行参数时,程序会显示提示信息。要求程序输出成绩列表,以及成绩的最高分和最低分。程序中不使用Math类的max和min方法。
实验5.4.5: (P)用数组实现斐波纳契函数
编写程序FibonacciArray.java,用数组实现斐波纳契函数。由键盘读入n值(参见实验1.4.15)。程序将在控制台输出此斐波纳契序列f(0),f(1),…… f(n)。
斐波纳契序列的定义如下:
如果n = 0 或者 n = 1 , 则 f(n) = 1
如果n是一个整数并且n > 1 , 则f(n)= f(n-1) + f(n-2)
实验5.4.6: (I)字符串和字符数组
在Java库文件中查看String类的toCharArray方法。我们提到过String包含一串字符,更准确地讲,包含了一个字符数组以及操作这些字符的大量方法。
String类的toCharArray方法将字符串转换为Character数组。toCharArray是一个实例方法还是类方法,查看Java文档,写下结果:
_______________________________________________________
范例5-3使用toCharArray方法实现字符串反转。
程序清单5-3:StringReverse.java
public class StringReverse {
public static void main(String[] args) {
System.out.println(reverse(args[0]));
}
public static String reverse( String inputString) {
if (inputString == null)
return null;
else {
char[] charArray = inputString.toCharArray(); //string to char array
for (int i = 0; i < charArray.length / 2; i++) {
char temp = charArray [i];
charArray [i] = charArray [charArray.length - i - 1];
charArray [charArray.length - i - 1] = temp;
}
String re = new String(charArray); //char array to string
return re;
}
} // method reverse
}
以上程序的算法如下图所示,程序进行1/2数组长度次的循环,每次调换头尾两个位置中的元素。在第一次循环中,先将位置0上的元素存放在temp变量,然后将位置9上的元素赋值到位置0,最后将temp中元素存入位置9。在第二次循环时,先将位置1上的元素存放在temp变量,然后将位置8上的元素赋值到位置1,最后将temp中元素存入位置8。依次类推,最后一次循环,将位置4和位置5中的元素交换。
图5.2 字符串反转算法描述
实验5.4.7: (I)传递数组参数
用简单类型数据做参数时,Java遵循值传递的原则。实质上,就是将参数值先拷贝一份,然后将拷贝传递给方法。因此,在方法内部修改参数对原来的变量没有影响(因为只是对拷贝的修改),而且,方法返回时,方法调用的空间回收,拷贝的参数空间也被收回。
Java中,不能把对象传递给方法,只能传递对象的引用(Reference,即对象的内存地址)。引用本身也是采用值调用。当一个方法接受对象的引用后,它就可以直接操纵对象。
尽管数组在语法上有别于其他一般意义上的对象,但数组被Java看作是对象。数组以引用调用的方式传递给方法,数组的引用变量包含一个指针,指向数组在存储器中的实际存储位置。
编译并运行程序ValueVsReference.java。
程序清单5-4:ValueVsReference.java
public class ValueVsReference {
public static void main(String[] args) {
short[] a = { 0, 2, 4, 6, 8 };
System.out.println("The values originally stored in the array are:");
for ( int i = 0; i < a.length; i++ )
System.out.print(" " + a[i]);
System.out.println();
System.out.println("Let's see how the values change.");
modifyPrimitive(a[0]);
for ( int i = 0; i < a.length; i++ )
System.out.print(" " + a[i]);
System.out.println(" Method tried to modify primitive element.");
modifyArray(a);
for ( int i = 0; i < a.length; i++ )
System.out.print(" " + a[i]);
System.out.println(" Method modified array.");
modifyArrayReference(a);
for ( int i = 0; i < a.length; i++ )
System.out.print(" " + a[i]);
System.out.println(" Method modified its own reference to array.");
} // method main
public static void modifyPrimitive(short p) {
p = 20;
} // method methodA
public static void modifyArray(short[] b) {
for ( int j = 0; j < b.length; j++ )
b[j] = (short) (j*(-3));
} // method modifyArray
public static void modifyArrayReference(short[] b) {
b = new short[5];
for ( int j = 0; j < b.length; j++ )
b[j] = (short) (j*3);
} // method modifyArrayReference
} // class ValueVsReference
范例5-4定义了main和其他三个方法modifyPrimitive,、modifyArray和modifyArrayReference。
(1) modifyPrimitive 方法修改参数p的值,p是short原始数据类型。因此,此方法的参数转递为值传递。main方法中调用modifyPrimitive方法,传递的实际参数是数组的一个元素a[0],此元素的类型也是short原始数据类型。
(2) modifyArray方法的参数为数组引用变量b,而不是单独的某个数组元素。方法的头部定义如下:
public static void modifyArray(short[] b)
通过引用变量b,可以操纵b指向的数组对象。main方法中调用modifyArray方法,传递的实际参数是数组引用变量a,此时a和b的实际值都是一个指向同一数组对象的指针。modifyArray方法中,对b引用的数组对象的修改,就是对a引用的数组对象的修改。方法调用完成后,变量b的空间被回收,a引用的数组对象被修改。
(3) modifyArrayReference 方法内,数组引用变量b 被重新赋值,指向一个新创建的数组:
b = new short[5];
main方法中调用modifyArrayReference方法,传递的实际参数是数组引用变量a。但由于上面的这个赋值语句,形式参数b指向了一个新的数组,而不再指向a引用的数组。对变量b引用的数组对象的任何修改都不会改变变量a引用的数组对象。
现在,请写下程序执行后的结果,并认真思考为什么是这样的结果。
___________________________________________________________________________________
___________________________________________________________________________________
___________________________________________________________________________________
___________________________________________________________________________________
实验5.4.8: (W)写一个带有数组参数的方法
在MathUtility.java(见程序清单4-6)中,写一个静态方法average,此方法用一个双精度浮点数(double)数组作为参数,并返回此数组中元素的平均值。如果数组长度为0,方法返回值为0,并打印一个标准错误输出流消息。回顾如何定义一个指向double类型数组的指针,并在方法的参数列表中声明。
以如下方式定义average方法的方法头:
public static double average (double[] doubleArray)
运行AverageTest.java程序测试average方法。
程序清单5-5:AverageTest
public class AverageTest {
public static void main(String[] args) {
if (args.length == 0) {
System.out.println("This program will average any "
+ "number of precision floating-point ");
System.out.println("numbers entered as command-line "
+ "arguments. Result should be zero");
System.out.println("for no command-line arguments.");
} // if
double[] numbers = new double[args.length];
for ( int i = 0; i < args.length; i++ )
numbers[i] = new Double(args[i]).doubleValue();
System.out.println("The average is " + MathUtility.average(numbers) + ".");
} // method main
} // class AverageTest
实验5.4.9: (P)写一个返回值为数组的方法
一个方法如果没有返回值,则在该方法的前面用void关键字来修饰;如果返回值的类型为基本数据类型,只需在声明方法的前面加上相应的数据类型即可。同理,若方法需返回一个数组,则必须在该方法的前面加上数组类型的修饰符。例如,方法返回一个整形数组,则在该方法前加上int[ ]。
在MathUtility.java(见程序清单4-6)中,写一个power方法,计算一组数的指数。计算结果以数组的方式返回,这个方法应该有如下的方法头:
public static double[] power(float[] base, int exponent)
要求作为参数传递进来的数组不被改变。提示:在power方法内部,创建了一个新的数组并把base的内容拷贝到其中。然后对新数组进行操作,而不是修改旧的数组,旧的数组被完整地保留下来。
执行程序PowerTest4.java来测试该方法。
程序清单5-6:PowerTest4.java
public class PowerTest4 {
public static void main(String[] args) {
System.out.print("This program will raise the float ");
System.out.println("numbers in an array to an integer power.");
float[] basesToTest = { 1, 2, 3, 4};
testPower(basesToTest, 2);
testPower(basesToTest, 3);
testPower(basesToTest, -2);
testPower(basesToTest, 0);
} // method main(String[])
public static void testPower(float[] base, int exponent)
{
System.out.println();
System.out.print("Raising the numbers in ");
printArray(base);
System.out.println(" to the " + exponent + " power:");
double[] result = MathUtility.power(base, exponent);
System.out.print(" The numbers in ");
printArray(base);
System.out.println(" to the " + exponent + " power are:");
System.out.print(" ");
printArray(result);
} // method testPower
public static void printArray(double[] numbers) {
System.out.print("{");
if ( numbers.length > 0 )
System.out.print(" " + numbers[0]);
for ( int i = 1; i < numbers.length; i++ )
System.out.print(", " + numbers[i]);
System.out.print(" }");
} // method printArray(double[])
} // class PowerTest4
实验5.4.10: (P)简单的数组中的查找
在数组中查找某个元素,一种简单的方法是顺序查找法(Linear Search),即,从数组的第一个记录开始,逐个进行数组中元素和给定值的比较。
写一个LinearSearch.java程序查找一个字符串中包含多少个指定字符,这个字符串和指定字符由键盘输入。要求输出此字符串中指定字符的位置,及字符总数。
实验5.4.11: (D)多维数组
1. Java语言中,多维数组(Multi-dimensional arrays)被看作是数组的数组。
二维数组可以被想象成一个二维的表,表里的所有元素有统一的数据类型。
一个int类型的二维数组声明如下:
int a[ ][ ];
int[ ][ ] a;
二维数组的初始化有两种方式:静态初始化和动态初始化。
(1) 静态初始化
a[ ][ ]={{1,2},{3,4},{5,6,7}};
图5.3 静态初始化一个二维数组a
Java语言中,不要求多维数组每一维的大小相同。
(2
展开阅读全文