文章目录
封装
封装的概念
将类的某些信息隐藏在类内部,不允许外部程序直接访问,而是通过该类提供的方法来实现对隐藏信息的操作和访问
把尽可能多的东西藏起来,对外提供便捷的接口
如何实现封装
1、private关键字来限制访问权限
2、保证属性安全-主要使用setter实现封装的安全性
3、提供便捷访问的方式-使用getter来保障访问调用
public class Penguin {
private String name;
private int health;
private int love;
private String sex;
public Penguin() {
}
public void setName(String name){//setter 赋值
this.name=name;
}
public String getName(){//取值
return this.name;
}
public void setHealth(int health){
if(health< 0||health>100) {
System.out.println("输入的健康值不合法,默认值为:50");
health=50;
}
this.health=health;
}
public int getHealth(){
return this.health;
}
public void setLove(int love){
if(love<0 || love>100){
love=20;
}
this.love=love;
}
public int getLove(){
return this.love;
}
public void setSex(String sex){
//公母合法默认值是公
if(sex==null ||!sex.equals("母")&&!sex.equals("公")){
System.out.println("输入不合法默认值是公");
sex="公";
}
this.sex=sex;
}
public String getSex(){
return this.sex;
}
public void sayHello(){
System.out.println("我叫"+name+",我是一只"+sex+"企鹅,我的健康值是:"+
health+",我和主人的亲密度是:"+love);
}
Penguin p = new Penguin();
p.setName("qq公仔");
p.setHealth(-100);
p.setLove(101);
p.setSex("男");
p.sayHello();
(2)如果有构造方法,在构造器中调用setter保证封装的安全性
public Penguin(String name, int health, int love, String sex) {
//构造方法的封装
setSex(sex);
setLove(love);
setHealth(health);
setName(name);
}
Penguin p2 = new Penguin("zz",1003,-1,"hhh");
p2.sayHello();
封装的好处
便于使用者正确使用系统,防止错误修改属性
有助于系统之间的松耦合,提高系统独立性
提高软件的可重用性
降低了构建大型系统的风险
包
包的作用
防止命名冲突,区分名字相同的类
有助于实施访问权限控制
允许类组成较小的单元(类似文件夹),易于找到和使用相应的文件
在项目中能有清晰的组织架构
创建包
作为Java源代码第一条语句
用package声明包,以分号结尾(package 包名;)
jdk提供基本包
java.lang:虚拟机自动引入
java.util:提供一些实用类
java.io:输入、输出
利用idea创建包
1、分别创建包和类
2、创建类的过程中创建类所在的包
包命名规范
包名由小写字母组成,不能以圆点开头或结尾
package mypackage;
包名之前最好加上唯一的前缀,通常使用组织倒置的网络域名
package net.javagroup.mypackage;
包名后续部分依不同机构内部的规范不同而不同
package net.javagroup.research.powerproject;–部门名.项目名
导入包
import 包名. 类名;导入指定包中指定类
import java.util.*; 导入java.util包中所有类
使用包的注意事项
一个类同时引用了两个来自不同包的同名类
必须通过完整类名来区分
public class Penguin {
static int a;
public static void main(String[] args) {
//如果一个类中出现了多个同名的类名
//其中一个必须使用全类名来确定具体的类路径
cn.kgc.kb11.Penguin p = new cn.kgc.kb11.Penguin();
Penguin.a=1;
}
}
每个包都是独立的,顶层包不会包含子包的类
package和import的顺序是固定的
package必须位于第一行(忽略注释行)
只允许有一个package语句
其次是import
接着是类的声明
访问权限
public 公共的 范围是全工程
protected 包私有+子类 范围比默认的多一个子类可见
默认 包级私有 范围是同包可见
private 类私有 范围是类内部可见
static
static可以用来修饰:
成员变量
静态变量,可以直接通过类名访问
成员方法
静态方法,可以直接通过类名访问
代码块
静态代码块,当Java虚拟机加载类时,就会执行该代码块(会在第一次访问到该类时执行一次)
public class TestStatic {
public static int a;
public int b;
public void test(){
//普通方法可以调用静态方法和静态属性
//普通方法可以直接调用普通方法和普通属性
a++;
b++;
test2();
}
public static void test2(){
//静态方法可以调用静态方法和静态属性
//不可以直接调用普通方法和普通属性,需要用对象.方式使用
a++;
// b++;
// test();
}
static {//引入配置文件
System.out.println("会在第一次访问到该类时执行一次");
}
public static void main(String[] args) {
TestStatic.a++;
TestStatic t = new TestStatic();
}
}
static代码块
JVM加载类时,加载静态代码块
如果有多个静态块,按顺序加载
每个静态代码块只会被执行一次
public class StaticTest {
static int num=100;
static{
num+=100;
System.out.println(num);
}
static{
num+=100;
System.out.println(num);
}
public static void main(String[] args) {
StaticTest st1=new StaticTest();
StaticTest st2=new StaticTest();
System.out.println(StaticTest.num);
}
}
结果 200
300
300
static变量
类的成员变量包括
类变量(静态变量)
被static修饰的变量
在内存中只有一个拷贝
类内部,可在任何方法内直接访问静态变量
其他类中,可以直接通过类名访问
实例变量
没有被static修饰的变量
每创建一个实例,就会为实例变量分配一次内存,实例变量可以在内存中有多个拷贝,互不影响
public class StaticTest {
static int num;
public static void main(String[] args) {
num++;
StaticTest st1=new StaticTest();
st1.num++;
StaticTest st2=new StaticTest();
st2.num++;
StaticTest.num++;
System.out.println(num); //4
}
}
static变量的作用:
(1)能被类的所有实例共享,可作为实例之间进行交流的共享数据
(2)如果类的所有实例都包含一个相同的常量属性,可把这个属性定义为静态常量类型,从而节省内存空间
static方法
静态方法:可直接通过类名访问
静态方法中不能使用this和super
不能直接访问所属类的实例变量和实例方法
可直接访问类的静态变量和静态方法
**实例方法:**通过实例访问
可直接访问所属类的静态变量、静态方法、实例变量和实例方法
静态方法必须被实现 main()就是最常用的静态方法
public void play(int n) {
static int localv=5; //在实例方法里不可以定义static变量
health = health - n;
System.out.println(name+" "+localv+" "+health+" "+love);
}
继承
使用继承
子类与父类符合is-a关系
继承是代码重用的一种方式,将子类共有的属性和行为放到父类中
[访问修饰符] class Pet {
//公共的属性和方法
}
继承关键字-extends
[访问修饰符] class Dog extends Pet {
//子类特有的属性和方法
}
继承是Java中实现代码重用的重要手段之一。Java中只支持单根继承,即一个类只能有一个直接父类
例如:public class Dog extends Pet,Penguin 编译报错
super关键字
作用:用来区分父类和子类的同名属性和方法
子类访问父类成员
1、访问父类构造方法:
super();
super(属性名);
2、访问父类属性
super.属性名
3、访问父类方法
super.方法名();
super关键字来访问父类的成员
super只能出现在子类的方法和构造方法中
super调用构造方法时,只能是第一句
super不能访问父类的private成员
理解继承
子类继承父类的什么?
继承public和protected修饰的属性和方法,不管子类和父类是否在同一个包里
继承默认权限修饰符修饰的属性和方法,但子类和父类必须在同一个包里
不能被继承的父类成员:
private成员
构造方法
子类与父类不在同包,使用默认访问权限的成员
继承条件下的构造方法
private String name;
private int health;
private int love;
public Pet(String name, int health, int love) {
this.name = name;
this.health = health;
this.love = love;
System.out.println("父类带三个参数的构造方法");
}
public Pet() {
System.out.println("父类无参构造方法");
}
public Dog(){
System.out.println("子类dog无参构造方法");
}
public Dog(String name, int health, int love, String strain) {
super(name, health, love);
this.strain = strain;
System.out.println("子类dog带参构造方法");
}
一、子类构造方法(不管是带参还是无参)没有通过super显式调用父类的有参构造方法,系统默认先调用父类的无参构造方法,然后再调用子类相应的有参或无参构造
1、Dog dog=new Dog();
运行:父类无参构造方法
子类dog无参构造方法
2、Dog dog=new Dog("ZZZ’,100,100,“中华田园犬”);
前提:super(name, health, love);这句话我注释掉
二、子类构造方法通过super显式调用父类的有参构造方法,执行父类相应构造方法
三、子类构造方法通过this显式调用自身的其他构造方法,在相应构造方法中应用以上两条规则
我们假设dog 又有了新的特性 private String a;