Java java지도 시간 Java 정렬 보고서: 비교 방법이 일반 계약 예외 솔루션을 위반합니다.

Java 정렬 보고서: 비교 방법이 일반 계약 예외 솔루션을 위반합니다.

Jun 30, 2017 am 10:35 AM
method

이 기사에서는 주로 Java의 정렬 예외에 대한 솔루션을 소개합니다. 비교 방법은 일반 계약을 위반합니다. 기사의 소개는 매우 상세하며 필요한 모든 친구가 살펴볼 수 있는 특정 참고 자료와 학습 가치가 있습니다. 아래.

머리말

비교 방법이 일반 계약을 위반합니다가 지난 주에 온라인에서 정렬된 Java 코드 조각에 나타났습니다. 이 문제를 해결하는 방법에 대한 지식을 공유하겠습니다. 여기. . Comparison method violates its general contract,在解决这个问题的途中学到了一些知识这里总结分享一下。

异常原因

这个排序导致的异常将会在java7以上的版本出现,所以如果你的JDK从6升级到了7或者8,那一定要小心此异常。

在java7的兼容列表中,就有对此排序不兼容的说明:

Area: API: Utilities
Synopsis: Updated sort behavior for Arrays and Collections may throw an IllegalArgumentException
Description: The sorting algorithm used by java.util.Arrays.sort and (indirectly) by java.util.Collections.sort has been replaced. The new sort implementation may throw an IllegalArgumentException if it detects a Comparable that violates the Comparable contract. The previous implementation silently ignored such a situation.
If the previous behavior is desired, you can use the new system property, java.util.Arrays.useLegacyMergeSort, to restore previous mergesort behavior.
Nature of Incompatibility: behavioral
RFE: 6804124
로그인 후 복사

我从资料中查阅到java7开始引入了Timsort的排序算法。我之前一直以为大部分标准库的内置排序算法都是快速排序。现在才得知很多语言内部都使用Timsort排序。随后我在wiki百科上找到了这样一句话:

t was implemented by Tim Peters in 2002 for use in the Python programming language.

所以这个排序自然是以他命名的。

随后我又在网上找到了这样一张图排序比较的图:

可以发现,Timsort在表现上比QuickSort还要好。

这篇博客不去详细讨论Timsort的实现(看上去这个算法还挺复杂的),我可能会写另一篇博客单独讨论Timsort,简单来说Timsort结合了归并排序和插入排序。这个算法在实现过程中明确需要:严格的单调递增或者递减来保证算法的稳定性。

  • sgn(compare(x, y)) == -sgn(compare(y, x))

  • ((compare(x, y)>0) && (compare(y, z)>0)) implies compare(x, z)>0

  • compare(x, y)==0 implies that sgn(compare(x, z))==sgn(compare(y, z)) for all z

看上去很像离散数学课中学习的集合的对称性,传递性的关系。

所以异常的原因是因为排序算法不够严谨导致的,实际上业务上的代码经常不如纯技术上的严谨。比如对于这样一个算法:

选出航班中的最低价

那如果两个相等低价同时存在,按照寻找最低价的逻辑如果这么写:

if (thisPrice < lowPrice){
 lowPrice = thisPrice;
}
로그인 후 복사

那低价这个位置就是“先到先得”了。

但如果这么实现:

if(thisPrice <= lowPrice){
 lowPrice = thisPrice;
}
로그인 후 복사

那后面的低价就会覆盖前面的,变成了“后来者居上”。编程中经常遇到先到先得和后来者居上这两个问题。

所以对于上面那个需要提供严谨的判断大小比较函数实现。所以如果是这样的:

return x > y ? 1 : -1;
로그인 후 복사

那么就不符合此条件。

不过我们逻辑要比这个复杂,其实是这样一个排序条件。按照:

  • 价格进行排序,如果价格相等则起飞时间靠前的先排。

  • 如果起飞时间也相等,就会按照:

  • 非共享非经停>非经停>非共享>经停的属性进行优先级选择,如果这些属性都全部相等,才只能算是相等了。

所以这个判断函数的问题是:

public compareFlightPrice(flightPrice o1, flightPrice o2){
 // 非经停非共享
 if (o1.getStopNumber() == 0 && !o1.isShare()) {
 return -1;
 } else if (o2.getStopNumber() == 0 && !o2.isShare()) {
 return 1;
 } else {
 if (o1.getStopNumber() == 0) {
  return -1;
 } else if (o2.getStopNumber() == 0) {
  return 1;
 } else {
  if (!o1.isShare()) {
  return -1;
  } else if (!o2.isShare()) {
  return 1;
  } else {
  if (o1.getStopNumber() > 0) {
   return -1;
  } else if (o2.getStopNumber() > 0) {
   return 1;
  } else {
   return 0;
  }
  }
 }
 }
}
로그인 후 복사

这个函数有明显的先到先得的问题,比如对于compareFlightPrice(a, b) ,如果ab都是非共享非经停,那么这个就会把a排到前面,但如果调用compareFlightPrice(b, a)

예외 원인


🎜🎜이 정렬로 인한 예외는 java7 이상 버전에서 나타나므로 JDK를 6에서 7, 8로 업그레이드하는 경우에는 이 예외에 주의해야 합니다.
🎜🎜java7의 호환성 목록에는 이런 종류의 비호환성에 대한 설명이 있습니다. 🎜
-Djava.util.Arrays.useLegacyMergeSort=true
로그인 후 복사
🎜java7이 Timsort 정렬 알고리즘을 도입하기 시작했다는 정보에서 찾았습니다. 나는 항상 표준 라이브러리에 내장된 대부분의 정렬 알고리즘이 빠른 정렬이라고 생각했습니다. 이제 저는 많은 다국어가 내부적으로 Timsort를 사용한다는 것을 알게 되었습니다. 그러다가 Wikipedia에서 다음 문장을 발견했습니다. 🎜🎜t는 2002년 Tim Peters가 Python에서 사용하기 위해 구현했습니다. 프로그래밍 언어입니다.
🎜🎜따라서 이 정렬은 자연스럽게 그의 이름을 따서 명명되었습니다. 🎜🎜그런 다음 인터넷에서 다음과 같은 사진 정렬 비교를 찾았습니다. 🎜

🎜🎜Timsort가 QuickSort보다 성능이 더 좋은 것을 확인할 수 있습니다. 🎜🎜이 블로그에서는 Timsort의 구현에 대해 자세히 설명하지 않습니다(이 알고리즘은 상당히 복잡한 것 같습니다). Timsort를 별도로 설명하기 위해 다른 블로그를 작성할 수도 있습니다. 간단히 말해서 Timsort는 병합 정렬과 삽입 정렬을 결합합니다. 이 알고리즘은 알고리즘의 안정성을 보장하기 위해 구현 중에 엄격한 단조 증가 또는 감소를 요구합니다. 🎜

🎜

  • 🎜sgn(비교(x, y)) == -sgn(비교(y, x))🎜
  • 🎜((compare(x, y)>0) && (compare(y, z)>0))은 비교(x, z)>0을 의미합니다.🎜
  • 🎜compare(x, y)==0은 모든 z에 대해 sgn(compare(x, z))==sgn(compare(y, z))를 의미함
    🎜
🎜이산 수학 수업에서 배운 집합의 대칭 및 전이 관계와 매우 유사해 보입니다. 🎜🎜그래서 예외가 발생하는 이유는 정렬 알고리즘이 충분히 엄격하지 않기 때문입니다. 실제로 비즈니스 코드는 순수 기술만큼 엄격하지 않은 경우가 많습니다. 예를 들어 다음과 같은 알고리즘의 경우: 🎜🎜항공편 중 최저 가격 선택
🎜🎜두 개의 동일한 저가 가격이 동시에 존재하는 경우 최저가 찾기 논리에 따라 다음과 같이 작성하면 : 🎜rrreee🎜그럼 최저가의 입장은 "선착순" 입니다. 🎜🎜하지만 이것이 달성된다면: 🎜rrreee🎜그러면 뒤쪽의 낮은 가격이 앞쪽의 가격을 덮게 되고 "후발자 우선"이 됩니다. 프로그래밍은 종종 선착순과 후순위라는 두 가지 문제에 직면합니다. 🎜🎜그래서 위의 경우에는 엄격한 판단과 크기 비교 기능 구현이 필요합니다. 따라서 다음과 같이 보인다면: 🎜rrreee🎜 이는 이 기준을 충족하지 않는 것입니다. 🎜🎜하지만 우리의 논리는 이보다 더 복잡합니다. 실제로는 이러한 정렬 조건입니다. 정렬 기준: 🎜
  • 🎜가격이 동일할 경우 출발 시간이 빠른 것이 우선 순위가 됩니다. 🎜
  • 🎜출발 시간도 동일할 경우 다음 속성에 따라 수행됩니다. 🎜
  • 🎜Non-shared non-stop>Non-stop>Non- 공유>중지우선순위 선택, 이러한 속성이 모두 동일하면 동일한 것으로 간주될 수 있습니다. . 🎜
🎜이 판단 함수의 문제점은 다음과 같습니다. 🎜rrreee🎜이 함수에는 명백한 선착순 문제가 있습니다. 예를 들어 compareFlightPrice(a, b) , ab와 ab가 모두 비공유이고 논스톱이면 a가 첫 번째 순위가 되지만, <code>compareFlightPrice(b, a)가 호출되면 b가 첫 번째 순위가 되므로 a가 비공유, 논스톱인지 판단해야 합니다. b가 비공유 논스톱이 아닌 경우에만 a가 1순위가 될 수 있습니다. 🎜🎜물론, 비교 기능을 변경하는 것 외에도 또 다른 해결 방법이 있습니다. 즉, jvm에 시작 매개변수를 추가하는 것입니다. 🎜rrreee🎜또한 집합에 동일한 요소가 있다는 의미는 아니며 비교 함수가 위의 엄격한 정의를 충족하지 않는다는 점에 유의해야 합니다. 실제로 이 예외는 확실히 안정적으로 나타날 가능성이 있습니다. 이는 프로덕션 환경에서 발생하는 예외입니다. 결국 Java는 전체 배열을 먼저 확인할 만큼 어리석지 않습니다. 실제로 정렬 과정에서 이 조건을 충족하지 못하는 것을 발견합니다. 따라서 일종의 추심 명령을 통해 이러한 판단을 우회할 수 있는 가능성이 있습니다. 🎜

위 내용은 Java 정렬 보고서: 비교 방법이 일반 계약 예외 솔루션을 위반합니다.의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

인기 기사

R.E.P.O. 에너지 결정과 그들이하는 일 (노란색 크리스탈)
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 최고의 그래픽 설정
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 아무도들을 수없는 경우 오디오를 수정하는 방법
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25 : Myrise에서 모든 것을 잠금 해제하는 방법
4 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

Java의 클래스로드 메커니즘은 다른 클래스 로더 및 대표 모델을 포함하여 어떻게 작동합니까? Java의 클래스로드 메커니즘은 다른 클래스 로더 및 대표 모델을 포함하여 어떻게 작동합니까? Mar 17, 2025 pm 05:35 PM

Java의 클래스 로딩에는 부트 스트랩, 확장 및 응용 프로그램 클래스 로더가있는 계층 적 시스템을 사용하여 클래스로드, 링크 및 초기화 클래스가 포함됩니다. 학부모 위임 모델은 핵심 클래스가 먼저로드되어 사용자 정의 클래스 LOA에 영향을 미치도록합니다.

카페인 또는 구아바 캐시와 같은 라이브러리를 사용하여 자바 애플리케이션에서 다단계 캐싱을 구현하려면 어떻게해야합니까? 카페인 또는 구아바 캐시와 같은 라이브러리를 사용하여 자바 애플리케이션에서 다단계 캐싱을 구현하려면 어떻게해야합니까? Mar 17, 2025 pm 05:44 PM

이 기사는 카페인 및 구아바 캐시를 사용하여 자바에서 다단계 캐싱을 구현하여 응용 프로그램 성능을 향상시키는 것에 대해 설명합니다. 구성 및 퇴거 정책 관리 Best Pra와 함께 설정, 통합 및 성능 이점을 다룹니다.

캐싱 및 게으른 하중과 같은 고급 기능을 사용하여 객체 관계 매핑에 JPA (Java Persistence API)를 어떻게 사용하려면 어떻게해야합니까? 캐싱 및 게으른 하중과 같은 고급 기능을 사용하여 객체 관계 매핑에 JPA (Java Persistence API)를 어떻게 사용하려면 어떻게해야합니까? Mar 17, 2025 pm 05:43 PM

이 기사는 캐싱 및 게으른 하중과 같은 고급 기능을 사용하여 객체 관계 매핑에 JPA를 사용하는 것에 대해 설명합니다. 잠재적 인 함정을 강조하면서 성능을 최적화하기위한 설정, 엔티티 매핑 및 모범 사례를 다룹니다. [159 문자]

고급 Java 프로젝트 관리, 구축 자동화 및 종속성 해상도에 Maven 또는 Gradle을 어떻게 사용합니까? 고급 Java 프로젝트 관리, 구축 자동화 및 종속성 해상도에 Maven 또는 Gradle을 어떻게 사용합니까? Mar 17, 2025 pm 05:46 PM

이 기사에서는 Java 프로젝트 관리, 구축 자동화 및 종속성 해상도에 Maven 및 Gradle을 사용하여 접근 방식과 최적화 전략을 비교합니다.

적절한 버전 및 종속성 관리로 Custom Java 라이브러리 (JAR Files)를 작성하고 사용하려면 어떻게해야합니까? 적절한 버전 및 종속성 관리로 Custom Java 라이브러리 (JAR Files)를 작성하고 사용하려면 어떻게해야합니까? Mar 17, 2025 pm 05:45 PM

이 기사에서는 Maven 및 Gradle과 같은 도구를 사용하여 적절한 버전 및 종속성 관리로 사용자 정의 Java 라이브러리 (JAR Files)를 작성하고 사용하는 것에 대해 설명합니다.

See all articles