此文已经同步至个人站点博客,点击下方链接可以体验更加阅读模式:
《java题库》
1.构造器(constructor)是否可被重写(override)?
constructor
(构造器)不能被继承,所以不能被override
(重写),但是可以被overloading
(重载)。
2.方法可以同时即是 static 又是 synchronized 的吗?
synchronized
是对类的当前实例进行加锁,防止其他线程同时访问该类的该实例的所有synchronized
块,注意这里是“类的当前实例”, 类的两个不同实例就没有这种约束了。那么static synchronized
恰好就是要控制类的所有实例的访问了,static synchronized
是限制线程同时访问jvm
中该类的所有实例同时访问对应的代码快。实际上,在类中某方法或某代码块中有synchronized
,那么在生成一个该类实例后,改类也就有一个监视快,放置线程并发访问改实例synchronized
保护块,而static synchronized
则是所有该类的实例公用一个监视快了,也也就是两个的区别了,也就是synchronized
相当于 this.synchronized
,而static synchronized
相当于Something.synchronized
。
结论就是:两个关键字可以共存,其区别如下:
-
synchronized static
是某个类的范围,synchronized static cSync{ }
防止多个线程同时访问这个 类中的synchronized static
方法。它可以对类的所有对象实例起作用。 -
synchronized
是某实例的范围,synchronized isSync(){ }
防止多个线程同时访问这个实例中的synchronized
方法。
3.abstract 的 method是否可同时是 static,是否可同时是 native,是否可同时是synchronized?
1、abstract与static
(what)
abstract
:用来声明抽象方法,抽象方法没有方法体,不能被直接调用,必须在子类overriding后才能使用
static
:用来声明静态方法,静态方法可以被类及其对象调用
(how)
static
与abstract
不能同时使用
(why)
用static
声明方法表明这个方法在不生成类的实例时可直接被类调用,而abstract
方法不能被调用,两者矛盾。
2、abstract与native
(what)
native
:用来声明本地方法,该方法的实现由非java语言实现,比如C。一般用于java与外环境交互,或与操作系统交互
(how)
native
可以与所有其它的java
标识符连用,但是abstract
除外。
(why)
因为native
暗示这些方法是有实现体的,只不过这些实现体是非java
的,但是abstract
却显然的指明这些方法无实现体。
3、abstract与synchronized
(what)
synchronized
:用于防止多个线程同时调用一个对象的该方法,与static
连用可防止多个线程同时调用一个类的该方法
(how)
abstract
与synchronized
不能同时使用
(why)
从synchronized
的功能也可以看出,用synchronized
的前提是该方法可以被直接调用,显然和abstract
连用冲突。
4.Java支持哪种参数传递类型
传值!传值!可能会得到两种答案之一:(1)Java传递原始类型数据时使用的是传值方式;传递对象时则使用传引用方式;String类型的数据采用的是传值方式,因为字符串是不可变的。(2)Java传递所有参数都使用传值方式。很有争议,理解到了就行。
为什么是第二个答案是正确的。理解传值和传引用的区别的关键是要记住,当你向一个方法传递一个对象时,Java没有把对象放入堆栈,它只是拷贝对象的引用然后将这个引用的拷贝放入堆栈。也就是说,根据定义,Java使用的是传值方式。
值传递:方法调用时,实际参数把它的值传递给对应的形式参数,方法执行中形式参数值的改变不影响实际参 数的值。
引用传递:也称为传地址。方法调用时,实际参数的引用(地址,而不是参数的值)被传递给方法中相对应的形式参数,在方法执行中,对形式参数的操作实际上就是对实际参数的操作,方法执行中形式参数值的改变将会影响实际参数的值。
下面举例说明:
package w3ang.javabasic;
public class PassValue
{
static void exchange(int a, int b)
{// 静态方法,交换a,b的值
int temp;
temp = a;
a = b;
b = temp;
}
public static void main(String[] args)
{
int i = 10;
int j = 100;
System.out.println("before call: " + "i=" + i + ", " + "j = " + j);// 调用前
exchange(i, j); // 值传递,main方法只能调用静态方法
System.out.println("after call: " + "i=" + i + ", " + "j = " + j);// 调用后
}
}
/*
这是基本类型,传值---传递基本数据类型参数
before call: i=10, j = 100
after call: i=10, j = 100
*/
说明:调用exchange(i, j)
时,实际参数i,j
分别把值传递给相应的形式参数a,b
,在执行方法exchange()
时,形式参数a,b
的值的改变不影响实际参数i、j
的值,i、j
的值在调用前后并没改变。
引用传递---对象作为参数
如果在方法中把对象(或数组)作为参数,方法调用时,参数传递的是对象的引用(地址),即在方法调用时,实际参数把对对象的引用(地址)传递给形式参数。这是实际参数与形式参数指向同一个地址,即同一个对象(数组),方法执行时,对形式参数的改变实际上就是对实际参数的改变,这个结果在调用结束后被保留了下来。
package w3ang.javabasic;
class Book
{
String name;
private float price;
Book(String n,float p)
{//构造方法
name = n;
price = p;
}
static void change(Book a_book, String n, float p)
{ // 静态方法,对象作为参数
a_book.name = n;
a_book.price = p;
}
public void output()
{ // 实例方法,输出对象信息
System.out.println("name: " + name + ", " + "price: " + price);
}
}
public class PassAddr
{
public static void main(String[] args)
{
Book b = new Book("java2", 32.5f);
System.out.print("before call: "); // 调用前
b.output();
//b.change(b, "c++", 45.5f);//这样也Ok,但是提示:The static method change
//(Book, String, float)from the type Book should be accessed in a static way
Book.change(b, "c++", 45.5f); // 引用传递,传递对象b的引用,修改对象b的值
System.out.print("after call: "); // 调用后
b.output();
}
}
/*
这是对象,引用传递---对象作为参数
before call: name: java2, price: 32.5
after call: name: c++, price: 45.5
*/
说明:调用 change(b,"c++",45.5f)
时,对象b
作为实际参数,把引用传递给相应的形式参数a_book
,实际上a_book
也指向同一个对象,即该对象有两个引用名:b
和a_book
。在执行方法change()
时,对形式参数a_book
操作就是对实际参数b
的操作。
5.一个对象被当作参数传递到一个方法,是值传递还是引用传递
6.当一个对象被当作参数传递到一个方法后,此方法可改变这个对象的属性,并可返回变化后的结果,那么这里到底是值传递还是引用传递?
public void add(int a) { int b = a; }
这个可以看作是值传递,a
是基本数据类型,他把他的值传给了b
public void add(Object obj) { Object objTest = obj; }
这个可以看作是址传递,obj
是引用数据类型,是把他栈中指向堆中的对象的地址值赋值给了objTest
. 这时候就同时有两个引用指向了堆中的某个Object
对象 其实这样看来,java应该只有值传递的。如果是基本数据类型,传递的就是实际的值. 如果是引用数据类型,传递的就是该引用的地址值。
7.我们能否重载main()方法?
可以,我们可以重载main()
方法。一个Java类可以有任意数量的main()
方法。为了运行java类,类的main()
方法应该有例如“public static void main(String[] args)
”的声明。如果你对此声明做任何修改,编译也是可以成功的。但是,运行不了Java程序。你会得到运行时错误,因为找不到main方法,就是说JVM之认可public static void main(String[] args)
声明的mian()
方法。
8.如果main方法被声明为private会怎样?
运行错误:
Exception in thread "main" java.lang.NoSuchMethodError: main
结论:虚拟机只运行"public static void main(String[] args)"
形式的main函数,且只可有一个此种形式的main函数。