java基础知识和语法
1.类的信息加载完成后,会自动调用静态代码块,可以完成静态属性的初始化功能
对象准备创建时,也会自动调用代码块,但不是静态的;
public class java_3_24
{
public static void main(String[] args) {
new User08();
}
}
class User08{
static {
System.out.println("静态代码块执行1");
}
static {
System.out.println("静态代码块执行2");
}
static void test(){
System.out.println("test...");
}
{
System.out.println("代码块执行");
}
static {
System.out.println("静态代码块执行3");
}
}
注意:先调用静态代码块,再调用了代码块,这是准备创建的情况,既会调用代码块,也会调用静态代码块;
public class java_3_24
{
public static void main(String[] args) {
User08.test(); //因为test是静态方法,所以可以通过类名直接调用;
}
}
class User08{
static {
System.out.println("静态代码块执行1");
}
static {
System.out.println("静态代码块执行2");
}
static void test(){
System.out.println("test...");
}
{
System.out.println("代码块执行");
}
static {
System.out.println("静态代码块执行3");
}
}
这是信息加载的情况,只会调用静态代码块;
2.包:package;
在包之中容纳类,调用包中的类:
java.lang.Object;
主要目的是分类管理;类可以没有包;
包的声明语句只能在一个源码文件中使用一次;
包名一般都是小写;
3.一般情况下,使用类时,都会使用类的全名:包名+类,但有例外:
(1)使用的类就在本包中,无需全名;
(2)java.lang中的类,无需全名;
(3)事先import一个类,也无需全名;
例如:
import java.util.Date;
public class java_3_24
{
public static void main(String[] args) {
Date d = new Date();
}
}
但import在使用时有需要注意的地方:
(1)import语句只能用于package后,class前;
(2)import可以多次使用,导入多个类;
如果要导入多个类,可以使用通配符*来操作:
import java.util.*;
public class java_3_24
{
public static void main(String[] args) {
Date d = new Date();
}
}
4.构建对象:
构造方法:在一个类中如果没有自己设置创建方法,那么JVM会自动添加一个公共的,无参的构造方法,方便使用;
(1)构造方法也是方法,但没void关键字;
(2)方法名和类名完全相同;
(3)构造方法可以传递参数,但是一般传递参数的目的是用于对象属性的赋值;
例如:
public class java_3_24
{
public static void main(String[] args) {
System.out.println("before");
User11 user = new User11();
System.out.println("after");
user.test();
}
}
class User11{
User11(){
System.out.println("user...");
}
void test()
{
System.out.println("test...");
}
}
在定义构造方法的同时,打印一个user;
例如:
public class java_3_24
{
public static void main(String[] args) {
User11 user = new User11("zhangsan");
System.out.println(user.username);
}
}
class User11{
String username;
User11(String name){
username = name;
}
}
打印结果为zhangsan;
5.面向对象编程中有3个非常重要的特征:继承,封装,多态;
下面介绍继承:
(1)类存在父子关系:子类可以直接获取到父类的成员属性和成员方法。
(2)类的继承只能单继承一个父类;
(3)一个父类可以有多个子类;
例如:
import java.util.concurrent.Callable;
public class java_3_24
{
public static void main(String[] args) {
Child c = new Child();
System.out.println(c.name);
c.test();
};
}
class Parent{
String name = "zhangsan";
void test(){
System.out.println("test...");
}
}
class Child extends Parent{
}
6.如果父类和子类含有相同的属性,那么可以采用特殊的关键字来区分:
super & this. 例如:
import java.util.concurrent.Callable;
public class java_3_24
{
public static void main(String[] args) {
Child c = new Child();
c.test();
};
}
class Parent{
String name = "zhangsan";
}
class Child extends Parent{
String name = "lisi";
void test(){
System.out.println(super.name);
System.out.println(this.name);
}
}
父子类中的构造方法:
父类对象是在子类对象创建前创建完成,创建子类对象前,会调用父类的构造方法完成父类的创建
默认情况下,子类对象创建时,会默认调用父类的构造方法完成父类对象的创建,使用的是super方法;
例如:
import java.util.concurrent.Callable;
public class java_3_24
{
public static void main(String[] args) {
Child c = new Child();
};
}
class Parent{
Parent(String name){
System.out.println("parent...");
}
}
class Child extends Parent{
Child() {
super("zhangsan");
System.out.println("child...");
}
}
父类要求传参才能构建,所以这时省略super方法就行不通了,所以要手动的进行传参,注意super方法时在子类的构建方法中调用的;
7.多态:
所谓的多态,其实就是一个对象在不同场景下表现出来的不同状态和形态;
多态语法其实就是对对象的使用场景进行了约束;
import java.util.concurrent.Callable;
public class java_3_24
{
public static void main(String[] args) {
Person p = new Person();
p.testperson();
Person p1 = new Boy();
p1.testperson();
p1.testboy(); //会报错
Person p2 = new Girl();
p2.testgirl(); //会报错
};
}
class Person{
void testperson(){
System.out.println("testperson");
}
}
class Boy extends Person{
void testboy(){
System.out.println("testboy");
}
}
class Girl extends Person{
void testgirl(){
System.out.println("testgirl");
}
}
一个对象可以使用的功能取决于引用变量的类型;上面的p1和p2都是person类型,所以可以引用person的功能;
8.一个类中,不能重复声明相同的方法(方法名、参数列表,但是和返回值类型无关),也不能声明相同的属性;
如果方法名相同,但参数列表不同,会被认为是不同的方法,只不过名称一样,这个操作在java中称之为方法的重载;
例如:
import java.util.concurrent.Callable;
public class java_3_24
{
public static void main(String[] args) {
User14 user = new User14();
user.login(12321);
user.login("nima", "cnm");
user.login("123123");
}
}
class User14{
void login(String account, String password){
System.out.println("账号密码登录");
}
void login(int tel){
System.out.println("手机号登录");
}
void login(String vx){
System.out.println("weixin登录");
}
}
三个方法名字相同,执行不同功能;
9.方法的重载:
import java.util.concurrent.Callable;
public class java_3_24
{
public static void main(String[] args) {
User12 user = new User12();
}
}
class User12{
User12(){
this("zhangsan");
}
User12(String name){
this(name, "男");
}
User12(String name, String sex){
System.out.println(name + "," + sex);
}
}
当调用无参构造方法时,用this来调用自身的构造方法,并传入参数,再调再传,然后执行打印操作;
10.匹配方法时的精度扩大:
import java.util.concurrent.Callable;
public class java_3_24
{
public static void main(String[] args) {
byte b = 10;
test(b);
}
static void test(byte b){
System.out.println("bbb");
}
static void test(short s){
System.out.println("sss");
}
static void test(char c){
System.out.println("ccc");
}
static void rest(int i){
System.out.println("iii");
}
}
此时打印bbb没毛病,但当我去掉test(byte b)这个方法时,就会打印sss,这是因为byte类型扩大精度优先转换为容易转换的类型;
注意:byte类型无法和char类型转换,因为byte类型有负数,而char类型没有负数;
11.重载和多态的综合:
import java.util.concurrent.Callable;
public class java_3_24
{
public static void main(String[] args) {
AAA aaa = new AAA();
BBB bbb = new BBB();
test(bbb);
}
static void test(AAA aaa){
System.out.println("aaa");
}
}
class AAA{
}
class BBB extends AAA{
}
打印结果为aaa,因为当前test方法应该接受AAA类型,但传入BBB类型,就会发生类似于基本类型的精度提升,只不过这里是从子向父提升;
12.方法的重写:父类对象的方法其实主要体现通用性,无法在特殊的场合下使用;
如果子类对象需要在特殊场合使用,那么需要重写方法的逻辑,这个操作在java中称为方法的重写;(并不影响父类的方法,使用super类型还是可以访问);
重写方法:子类的方法和父类的方法,方法名相同,返回值类型相同,参数列表要相同;
总结:(1)一个对象使用什么方法,取决于引用变量的类型;
(2)一个对象能使用什么属性,取决于引用变量的类型;
(3)一个对象的方法具体的使用是需要看具体的对象的;
(4)一个对象的属性具体的使用是不需要看具体的对象的,属性在哪里声明就在哪里使用;
例如:
import java.util.concurrent.Callable;
public class java_3_24
{
public static void main(String[] args) {
CCC ddd = new DDD();
System.out.println(ddd.sum());
}
}
class CCC{
int i = 10;
int sum(){
return i + 10;
}
}
class DDD extends CCC{
int i = 20;
int sum(){
return i + 20;
}
}
ddd是CCC类型,所以应该看CCC中有没有sum方法,有,那么能使用,但具体的实现过程是依靠DDD中的sum方法,所以输出结果为40;
import java.util.concurrent.Callable;
public class java_3_24
{
public static void main(String[] args) {
CCC ddd = new DDD();
System.out.println(ddd.sum());
}
}
class CCC{
int i = 10;
int sum(){
return i + 10;
}
}
class DDD extends CCC{
int i = 20;
}
当把子类中的sum类型去掉后,方法具体的实现在子类中没有,那么去父类中寻找,父类的sum实现时,默认使用的是this属性,也就是父类中的属性,所以结果为20;
再看:
import java.util.concurrent.Callable;
public class java_3_24
{
public static void main(String[] args) {
CCC ddd = new DDD();
System.out.println(ddd.sum());
}
}
class CCC {
int i = 10;
int geti() {
return i;
}
int sum() {
return geti() + 10;
}
}
class DDD extends CCC {
int i = 20;
int geti() {
return i;
}
}
这串代码中,首先运行sum,在子类中没办法实现,最终要落回父类,再执行geti方法,这个需要子类中的方法进行实现,所以最终打印30;
13.访问权限:
public:公共的,访问权限修饰符,java源码中,公共类只能有一个,而且必须和源码文件名相同;
(1)private:私有的,同一个类中可以使用;
(2)(default):默认权限,不设定时默认设定,同类同包可用;
(3)protected:受保护的:同包,同类,子类可用;
(4)public:公共的,任意使用;
14.Java中不允许外部类使用private,protected修饰;
所谓外部类,就是在源码中直接声明的类;
所谓的内部类,就是类中声明的类;
注意:内部类就当成外部类的属性使用即可,因为内部类可以看做内部类的属性,所以需要构造外部类对象才可以使用;
例如:内部类的创建过程:
import java.util.concurrent.Callable;
public class java_3_24
{
public static void main(String[] args) {
OuterClass outer = new OuterClass();
OuterClass.InnerClass innerClass = outer.new InnerClass();
}
}
class OuterClass{
public class InnerClass{
}
}
15.类的单一创建:单例模式:
由于类的创建过程复杂,如果在外部直接创建类的对象,可能会丢失一些结构,使对象使用出现问题;
还有就是类占用内存较大,我们尽可能少创建类的对象,所以我们可以把类的构建方法进行private修饰,那么就只能在类中构建了;同时再加上if语句做判断,那么就保证了一个类只会产生一个对象;
例如:
import java.util.concurrent.Callable;
public class java_3_24
{
public static void main(String[] args) {
User19 instance = User19.getInstance();
}
}
class User19{
private static User19 user19 = null;
private User19(){
}
public static User19 getInstance(){
if(user19 == null){
user19 = new User19();
}
return user19;
}
}
在这串代码中,创建了一个user19作为属性,然后设计getinstance方法,实现了在类的内部创建对象;
其中,由于静态方法只能访问静态属性,非静态方法既可以访问静态也可以访问非静态属性,那么就需要给属性前加上static;
16.Java中提供了一种语法,可以在数据初始化后不被修改,使用关键字final;
final可以修饰变量;变量的值一旦初始化后无法修改。
例如:
import java.util.concurrent.Callable;
public class java_3_24
{
public static void main(String[] args) {
final String name = "zhangsan";
name = "lisi";
}
}
会发生报错;
(1)final可以修饰变量:变量的值一旦初始化后无法修改;
(2)final可以修饰属性:那么JvM无法自动进行初始化,需要自己进行初始化,属性值不能发生变化;
(3)final可以修饰方法,这个方法不能被子类重写;
(4)final可以修饰类,这样的类就没有子类了;
(5)final不可以修饰构造方法
(6)final可以修饰方法的参数,一旦修饰,参数就无法修改了;
