Java 基础之修饰符关键词整理

 更新时间:2017年02月20日 17:17:12   投稿:lqh  
这篇文章主要介绍了Java 基础之修饰符关键词整理的相关资料,需要的朋友可以参考下

Java 基础之修饰符关键词整理

我成为一个Java程序员距今已有一段时日。最近,有人问我关于Java修饰符关键字的一个问题,但我根本不知道那是什么。所以我觉得除了实际编程和算法,我也有必要学习这些内容。

通过谷歌搜索,我只得到一些琐碎的要点,并不完整。所以我以此主题写了这篇文章。这也是一个可用于测试你的计算机科学知识的面试问题。

Java修饰符是你添加到变量、类和方法以改变其含义的关键词。它们可分为两组:

  1. 访问控制修饰符
  2. 非访问修饰符

让我们先来看看访问控制修饰符,以及如何使用它们的一些代码示例。

修饰符 说明
public 公共可见
private 类可见
protected 包和所有的子类可见

那么如何使用这三种访问控制修饰符呢?请看下面两个类。请忽略此处代码的低效,因为这是教程。

创建一个名为project/mypackage/Person.java文件,并添加以下代码:

package mypackage;
class Person {
 private String firstname;
 private String lastname;
 protected void setFirstname(String firstname) {
  this.firstname = firstname;
 }
 protected void setLastname(String lastname) {
  this.lastname = lastname;
 }
 protected String getFirstname() {
  return this.firstname;
 }
 protected String getLastname() {
  return this.lastname;
 }
}

上面的Person类有private变量和protected方法。这意味着这些变量将只能从类访问,方法将只能从mypackage包访问。

接下来创建一个名为project/mypackage/Company.java的文件,并添加以下代码:

package mypackage;
import java.util.*;
public class Company {
 private ArrayList<Person> people;
 public Company() {
  this.people = new ArrayList<Person>();
 }
 public void addPerson(String firstname, String lastname) {
  Person p = new Person();
  p.setFirstname(firstname);
  p.setLastname(lastname);
  this.people.add(p);
 }
 public void printPeople() {
  for(int i = 0; i < this.people.size(); i++) {
   System.out.println(this.people.get(i).getFirstname() + " " + this.people.get(i).getLastname());
  }
 }
}

上面的类是公共的,因此它可以从包内部和外部的任何类进行访问。它有一个只能在类内访问的私有变量,以及一堆的公共方法。由于Person类和Company类共享相同的包,所以Company类可以访问Person类以及所有它的方法。

为了完成访问控制修饰符的示范,让我们在一个新的project/MainDriver.java文件中创建一个驱动程序类:

import mypackage.*;
public class MainDriver {
 public static void main(String[] args) {
  Company c = new Company();
  c.addPerson("Nic", "Raboy");
  c.printPeople();
  Person p = new Person();
  p.setFirstname("Maria");
  p.setLastname("Campos");
 }
}

请记住,由于Company类是公共的,所以我们在添加和打印人的时候没有问题。然而,由于Person类是受保护的,所以我们会得到一个编译时错误,因为MainDriver不是mypackage包的一部分。

现在,让我们来看看现有的非访问修饰符,以及如何使用它们的一些示例代码。

修饰符 说明
static 用于创建类、方法和变量
final 用于最终确定类、变量和方法的实施方式
abstract 用于创建抽象方法和类
synchronized 用于多线程的同步机制对资源进行加锁,使得在同一个时间,只有一个线程可以进行操作
Volatile 一个变量声明为volatile,就意味着这个变量是随时会被其他线程修改的,因此不能将它cache在线程memory中。

那么如何使用这五个非访问修饰符呢?

Java中static修饰符的一个很好的例子就是:

int max = Integer.MAX_VALUE
int numeric = Integer.parseInt("1234");

在上面的例子中,请注意我们利用了Integer类中变量和方法,而不是先实例化。这是因为那些特定的方法和变量都是静态的。

abstract修饰符则略有不同。你可以创建一个带方法的类,但它们基本只能定义。你不能对它们添加逻辑。例如:

abstract class Shape {
 abstract int getArea(int width, int height);
}

然后在子类里,你才可以增加例如下面这样的代码:

class Rectangle extends Shape {
 int getArea(int width, int height) {
  return width * height;
 }
}

下面要讲讲synchronized和volatile修饰符。

先来看一个线程的例子,在这个例子里我们将从两个不同的线程去访问相同的方法:

import java.lang.*;
public class ThreadExample {
 public static void main(String[] args) {
  Thread thread1 = new Thread(new Runnable() {
   public void run() {
    print("THREAD 1");
   }
  });
  Thread thread2 = new Thread(new Runnable() {
   public void run() {
    print("THREAD 2");
   }
  });
  thread1.start();
  thread2.start();
 }
 public static void print(String s) {
  for(int i = 0; i < 5; i++) {
   System.out.println(s + ": " + i);
  }
 }
}

运行上述代码将输出打印一个随机的顺序。可能是连续的,也可能不连续,取决于CPU。然而,如果我们使用synchronized修饰符,那么第一个线程必须在第二个线程开始打印之前完成。print(String s)方法可以是这样的:

public static synchronized void print(String s) {
 for(int i = 0; i < 5; i++) {
  System.out.println(s + ": " + i);
 }
}

接下来,让我们看看使用volatile 修饰符的例子:

import java.lang.*;
public class ThreadExample {
 public static volatile boolean isActive;
 public static void main(String[] args) {
  isActive = true;
  Thread thread1 = new Thread(new Runnable() {
   public void run() {
    while(true) {
     if(isActive) {
      System.out.println("THREAD 1");
      isActive = false;
     }
    }
   }
  });
  Thread thread2 = new Thread(new Runnable() {
   public void run() {
    while(true) {
     if(!isActive) {
      System.out.println("THREAD 2");
      try {
       Thread.sleep(100);
      } catch (Exception e) {
      }
      isActive = true;
     }
    }
   }
  });
  thread1.start();
  thread2.start();
 }
}

由于volatile变量是一种状态标志,所以运行上面的代码会打印线程数,并在它们之间交替。这是因为该标志被存储在主存储器中。如果我们去掉volatile关键字,该线程将只交替一次,因为只使用一个本地参考,两个线程基本上彼此隐身。

结论

Java修饰符理解起来会有一点棘手,而且实际上很多程序员并不怎么熟悉它们。这是一个很好的面试问题,可以用于测试你的书本知识。最后,如果我有什么遗漏或解释错误的地方,欢迎各位不吝指出。

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

相关文章

  • Spring Boot中使用Spring MVC的示例解析

    Spring Boot中使用Spring MVC的示例解析

    MVC 是一种常见的软件设计模式,用于分离应用程序的不同部分以实现松散耦合和高内聚性,这篇文章主要介绍了如何在Spring Boot中使用Spring MVC,需要的朋友可以参考下
    2023-04-04
  • java动态目录树的实现示例

    java动态目录树的实现示例

    在开发过程中,常常需要对目录结构进行操作和展示,本文主要介绍了java动态目录树的实现示例,具有一定的参考价值,感兴趣的可以了解一下
    2024-03-03
  • java中获取hashmap中的所有key方式

    java中获取hashmap中的所有key方式

    这篇文章主要介绍了java中获取hashmap中的所有key方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-03-03
  • springboot+vue前后端分离项目中使用jwt实现登录认证

    springboot+vue前后端分离项目中使用jwt实现登录认证

    本文介绍了如何在SpringBoot+Vue前后端分离的项目中使用JWT实现登录认证,内容包括后端的响应工具类、JWT工具类、登录用户实体类、登录接口、测试接口、过滤器、启动类以及前端的登录页面实现,感兴趣的可以了解一下
    2024-10-10
  • 深入理解Java虚拟机之经典垃圾收集器

    深入理解Java虚拟机之经典垃圾收集器

    这篇文章主要介绍了深入理解Java虚拟机之经典垃圾收集器的相关知识,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-11-11
  • Java并发编程示例(六):等待线程执行终止

    Java并发编程示例(六):等待线程执行终止

    这篇文章主要介绍了Java并发编程示例(六):等待线程执行终止,在本节,示例程序演示等待初始化方法完成后,再去执行其他任务,需要的朋友可以参考下
    2014-12-12
  • Java中的异步回调问题

    Java中的异步回调问题

    这篇文章主要介绍了Java中的异步回调问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-12-12
  • java用字节数组解决FileInputStream读取汉字出现乱码问题

    java用字节数组解决FileInputStream读取汉字出现乱码问题

    这篇文章主要介绍了java用字节数组解决FileInputStream读取汉字出现乱码问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-05-05
  • 初学者Android studio安装图文详解

    初学者Android studio安装图文详解

    本文给大家分享android studio 安装图文详解包括下载安装及遇到的问题,非常不错,具有参考借鉴价值,需要的朋友参考下吧
    2017-09-09
  • mybatis-plus(insertBatchSomeColumn批量添加方式)

    mybatis-plus(insertBatchSomeColumn批量添加方式)

    这篇文章主要介绍了mybatis-plus(insertBatchSomeColumn批量添加方式),具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-03-03

最新评论