>Java >Java베이스 >Java의 몇 가지 일반적인 오류 소개

Java의 몇 가지 일반적인 오류 소개

尚
원래의
2019-12-23 15:43:284640검색

Java의 몇 가지 일반적인 오류 소개

일반적인 Java 오류:

1. Null 포인터 오류

Java 배열을 사용할 때 문자열 배열의 요소를 비교해야 하는 경우가 있습니다. 그런 다음 요소가 null이 아닌 경우 프로그램은 정상적으로 실행되지만 비교된 요소가 null이면 프로그램에서 null 포인터 오류가 발생합니다.

해결책: 보호를 추가하고 요소가 null이 아닐 때 판단하세요.

public static void main(Sring[] args){
  String [] sums = "adfafA";
  for(int i=0;i<sums.length;i++){
    if(sums[]!=null && sums[i].equals(sr)){
      System.out.println("找到元素"+sums[i]);
      break;
    }
  }
}

2. 데이터 정밀도 범위 및 유형 변환

Java에는 4가지 기본 데이터 유형이 있습니다.

정수 유형: -1 2 3 4...

부동 소수점 유형: float 1.3f; (지정되지 않은 유형, 기본값은 정밀도가 가장 높음)

문자 유형: 유니코드 인코딩, 2바이트

Boolean 유형: true, false

4가지 기본 데이터 유형의 정확도 범위가 다르며, 단계별로 개선됩니다. 단계.

데이터 유형을 사용할 때 정밀도 범위에 주의하지 않으면 데이터 오버플로가 발생하여 계산 오류가 발생합니다.

네 가지 기본 유형 중 정수, 부동 소수점 및 문자 데이터를 서로 변환할 수 있습니다.

변환 규칙은 다음과 같습니다.

①보이지 않는 변환: 정밀도가 작은 데이터 값을 정밀도가 높은 값에 할당하여 자동으로 변환할 수 있습니다. (하위에서 상위로)

② 강제변환 : 범위가 넓은 데이터값, (상위에서 하위로)

③ 연산자동 승격규칙 : 연산중 자동으로 데이터값으로 변환 고정밀 데이터 유형.

3. 세 가지 유형의 루프 사용

①for 루프: 루프 이전에 루프 횟수가 지정됩니다.

for(表达式;表达式;表达式){
        循环体
 }

break 문: 실행 후 즉시 루프에서 빠져나오고 후속 문을 실행하지 않고 모든 루프를 종료합니다.

continue 문: 실행 후 즉시 현재 단어 루프에서 벗어나 조건이 계속 루프되는지 다시 판단합니다.

②While 루프 문: 횟수를 알 수 없는 루프에 적합합니다.

while(条件表达式){
  语句
 };

③do-While 루프: 횟수를 알 수 없는 루프이기도 합니다. 명령은 판결을 내리기 전에 적어도 한 번 실행되어야 합니다. while이 true이면 루프를 계속하고, 그렇지 않으면 루프를 종료합니다.

do {
 语句
 }while(条件表达式);

4. 문자열을 여러 번 복사하세요

테스트로 찾을 수 없는 오류는 변경할 수 없는 개체의 여러 복사본을 생성하는 것입니다. 불변 객체는 변경할 수 없으므로 복사할 필요가 없습니다. 가장 일반적으로 사용되는 불변 객체는 String입니다.

String 객체의 내용을 변경해야 한다면 StringBuffer를 사용해야 합니다. 다음 코드는 정상적으로 작동합니다.

String s = new String ("Text here");

그러나 이 코드는 성능이 좋지 않고 불필요하게 복잡합니다. 다음과 같은 방법으로 위 코드를 다시 작성할 수도 있습니다.

String temp = "Text here";
String s = new String (temp);

하지만 이 코드에는 완전히 필요하지 않은 추가 문자열이 포함되어 있습니다. 더 나은 코드는

String s = "Text here";

5입니다. 복제에 의해 반환된 객체가 없습니다

캡슐화는 객체 지향 프로그래밍에서 중요한 개념입니다. 안타깝게도 Java를 사용하면 실수로 캡슐화를 깨기 쉽습니다. Java에서는 개인 데이터에 대한 참조를 반환할 수 있습니다. 다음 코드는 이를 나타냅니다.

import java.awt.Dimension;
  /***Example class.The x and y values should never*be negative.*/
  public class Example{
  private Dimension d = new Dimension (0, 0);
  public Example (){ }
  /*** Set height and width. Both height and width must be nonnegative * or an exception is thrown.*/
  public synchronized void setValues (int height,int width) throws IllegalArgumentException{
  if (height <0 || width <0)
  throw new IllegalArgumentException();
  d.height = height;
  d.width = width;
  }
  public synchronized Dimension getValues(){
  // Ooops! Breaks encapsulation
  return d;
  }
  }

예제 클래스는 저장하는 높이 및 너비 값이 항상 음수가 아님을 보장합니다. setValues() 메서드를 사용하여 음수 값을 설정하려고 하면 예외가 발생합니다. 불행하게도 getValues()는 d의 복사본이 아닌 d에 대한 참조를 반환하므로 다음과 같은 파괴적인 코드를 작성할 수 있습니다.

  Example ex = new Example();
  Dimension d = ex.getValues();
  d.height = -5;
  d.width = -10;

이제 예제 객체는 음수 값을 갖습니다. getValues() 호출자가 절대 반환된 Dimension 객체의 너비와 높이 값을 테스트하는 것만으로는 이러한 오류를 감지하는 것이 불가능합니다.

안타깝게도 시간이 지남에 따라 클라이언트 코드가 반환된 Dimension 개체의 값을 변경할 수 있습니다. 이때 특히 멀티 스레드 환경에서는 오류의 원인을 추적하는 것이 지루하고 시간이 많이 걸립니다.

더 좋은 방법은 getValues()가 복사본을 반환하도록 하는 것입니다.

  public synchronized Dimension getValues(){
  return new Dimension (d.x, d.y);
  }

6. 잘못된 데이터 복사

때때로 프로그래머는 복사본을 반환해야 한다는 것을 알지만 실수로 잘못된 데이터를 복사합니다. 데이터 복사 작업의 일부만 수행되었으므로 다음 코드는 프로그래머의 의도에서 벗어났습니다.

import java.awt.Dimension;
  /*** Example class. The height and width values should never * be
  negative. */
  public class Example{
  static final public int TOTAL_VALUES = 10;
  private Dimension[] d = new Dimension[TOTAL_VALUES];
  public Example (){ }
  /*** Set height and width. Both height and width must be nonnegative * or an exception will be thrown. */
  public synchronized void setValues (int index, int height, int width) throws IllegalArgumentException{
  if (height <0 || width <0)
  throw new IllegalArgumentException();
  if (d[index] == null)
  d[index] = new Dimension();
  d[index].height = height;
  d[index].width = width;
  }
  public synchronized Dimension[] getValues()
  throws CloneNotSupportedException{
  return (Dimension[])d.clone();
  }
  }

여기서 문제는 getValues() 메서드가 배열만 복제하고 배열에 포함된 Dimension 객체는 복제하지 않는다는 것입니다. 따라서 호출자는 해당 요소가 다른 Dimension 개체를 가리키도록 내부 배열을 변경할 수 없지만 호출자는 내부 배열 요소(즉, Dimension 개체)의 내용을 변경할 수 있습니다. getValues() 메소드의 더 나은 버전은 다음과 같습니다.

 public synchronized Dimension[] getValues() throws CloneNotSupportedException{
  Dimension[] copy = (Dimension[])d.clone();
  for (int i = 0; i
  // NOTE: Dimension isn’t cloneable.
  if (d != null)
  copy[i] = new Dimension (d[i].height, d[i].width);
  }
  return copy;
  }

원자 유형 데이터의 다차원 배열을 복제할 때 비슷한 실수가 발생합니다. 원자 유형에는 int, float 등이 포함됩니다. 아래와 같이 int 유형의 1차원 배열의 간단한 복제가 정확합니다.

 public void store (int[] data) throws CloneNotSupportedException{
  this.data = (int[])data.clone();
  // OK
  }

int 유형의 2차원 배열을 복사하는 것은 더 복잡합니다. Java에는 int 유형의 2차원 배열이 없으므로 int 유형의 2차원 배열은 실제로 1차원 배열입니다. 해당 유형은 int[]입니다. 단순히 int[][] 유형의 배열을 복제하면 위 예제의 getValues() 메소드의 첫 번째 버전과 동일한 실수가 발생하므로 이를 피해야 합니다. 다음 예는 2차원 int 배열을 복제할 때 잘못된 방법과 올바른 방법을 보여줍니다.

public void wrongStore (int[][] data) throws CloneNotSupportedException{
  this.data = (int[][])data.clone(); // Not OK!
  }
  public void rightStore (int[][] data){
  // OK!
  this.data = (int[][])data.clone();
  for (int i = 0; i
  if (data != null)
  this.data[i] = (int[])data[i].clone();
  }
  }

더 많은 Java 지식을 보려면 java 기본 튜토리얼 열을 주의하세요.

위 내용은 Java의 몇 가지 일반적인 오류 소개의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.