작성자: naruse (2023-12-25)
번역자: shia
Ruby 3.3.0 릴리스를 알리게 되어 기쁩니다. Ruby 3.3은 Prism이라는 이름의 새 파서를 추가하고, 파서 생성기로 Lrama를 사용하며, RJIT으로 명명된 새로운 순수 Ruby JIT 컴파일러를 추가하고, 특히 YJIT에서 많은 성능 향상이 있습니다.
Prism
- Prism 파서를 기본 gem에 추가했습니다.
- Prism은 Ruby 언어를 위한 이식 가능하고, 에러 내성이 있으며 유지 보수 가능한 재귀 하향 파서입니다.
- Prism은 프로덕션 환경에서 사용할 준비가 되어있으며 활발하게 유지보수되고 있으므로 Ripper 대신 이용할 수 있습니다.
- Prism 사용 방법에 대한 광범위한 문서가 있습니다.
- Prism은 CRuby에서 내부적으로 사용하는 C 라이브러리이자 Ruby 코드를 구문 분석해야 하는 모든 도구에서 사용할 수 있는 Ruby gem입니다.
- Prism API에서 주목할 만한 메서드는 다음과 같습니다.
Prism.parse(source)
는 구분 분석 결과 객체의 일부로 AST를 반환합니다.Prism.parse_comments(source)
는 주석을 반환합니다.Prism.parse_success?(source)
는 에러가 없다면 true를 반환합니다.
- 기여에 관심이 있다면 Prism 저장소에서 직접 풀 리퀘스트나 이슈를 만들 수 있습니다.
ruby --parser=prism
이나RUBYOPT="--parser=prism"
을 사용하여 Prism 컴파일러를 실험해 볼 수 있습니다. 이 플래그는 디버깅 용도임을 잊지 마세요.
Bison을 Lrama로 대체
- Bison 대신 Lrama LALR 파서 생성기를 사용하도록 변경했습니다. [Feature #19637]
- 관심이 있다면 Ruby 파서의 미래 비전을 참고하세요.
- 유지보수성을 위해 Lrama 내부 구문 분석기가 Racc에서 생성한 LR 구문 분석기로 대체되었습니다.
- 매개변수화 규칙
(?, *, +)
를 지원하며, Ruby의 parse.y에서 사용될 예정입니다.
YJIT
- Ruby 3.2 버전 대비 주요 성능 개선 사항
- 스플랫과 나머지 인자 지원이 개선되었습니다.
- 가상 머신의 스택 연산을 위해 레지스터가 할당됩니다.
- 선택적 인수가 포함된 더 많은 호출이 컴파일됩니다. 예외 처리기도 컴파일됩니다.
- 지원되지 않는 호출 타입인 경우, 클래스의 인스턴스 변수가 가지는 객체 형상의 조합이 매우 복잡한 경우에도 컴파일되기 전의 바이트 코드로 전환되지 않습니다.
- Rails의
#blank?
와 특수화한#present?
같은 단순한 메서드는 인라인으로 최적화됩니다. Integer#*
,Integer#!=
,String#!=
,String#getbyte
,Kernel#block_given?
,Kernel#is_a?
,Kernel#instance_of?
,Module#===
은 특별히 최적화되었습니다.- Ruby 3.2보다 컴파일 속도가 약간 개선되었습니다.
- 이제 optcarrot에서 인터프리터보다 3배 이상 빠릅니다!
- Ruby 3.2보다 메모리 사용량이 크게 개선되었습니다.
- 컴파일된 코드의 메타데이터가 훨씬 적은 메모리를 사용합니다.
--yjit-call-threshold
는 애플리케이션의 ISEQ이 40,000개 이상 존재하는 경우, 자동으로 30에서 120으로 증가합니다.- 적게 사용되는 ISEQ의 컴파일을 생략하는
--yjit-cold-threshold
가 추가됩니다. - ARM64에서 더 작은 코드를 생성합니다.
- 코드 GC는 이제 비활성화된 상태로 시작합니다.
--yjit-exec-mem-size
를 새 코드 컴파일을 멈추는 최대 상한으로 취급합니다.- 코드 GC로 인한 급격한 성능 하락이 사라집니다. Pitchfork로 리포크를 사용하는 서버에서 더 나은 쓰기 시 복사 동작을 생성할 수 있습니다.
- 필요하다면
--yjit-code-gc
로 코드 GC를 활성화할 수 있습니다.
- 실행 시간에 YJIT을 활성화할 수 있는
RubyVM::YJIT.enable
이 추가됩니다.- 실행 명령의 인수나 환경 변수의 변경 없이 YJIT을 실행할 수 있습니다. Rails 7.2는 이 메서드를 사용해 기본으로 YJIT을 활성화할 예정입니다.
- 애플리케이션 부팅이 완료된 후에 YJIT를 활성화하는 데에도 사용할 수 있습니다.
부팅 시 YJIT를 비활성화한 상태에서 다른 YJIT 옵션을 사용하려는 경우
--yjit-disable
을 사용할 수 있습니다.
- 기본으로 더 많은 YJIT 통계 정보를 제공합니다.
yjit_alloc_size
와 메타 정보에 관련된 몇몇 통계가 기본으로 이용 가능합니다.--yjit-stats
로부터 생성된ratio_in_yjit
통계는 릴리스 빌드에서도 이용 가능하며, 대부분의 통계에 접근하는 데 특수한 통계나 개발 빌드는 더 이상 필요하지 않습니다.
- 프로파일링 기능 추가
- Linux perf로 프로파일링을 용이하게 할 수 있도록
--yjit-perf
가 추가됩니다. --yjit-trace-exits
는--yjit-trace-exits-sample-rate=N
을 통한 샘플링을 지원합니다.
- Linux perf로 프로파일링을 용이하게 할 수 있도록
- 보다 철저한 테스트와 여러 버그 수정
RJIT
- 순수 Ruby JIT 컴파일러 RJIT을 도입하고 MJIT을 대체했습니다.
- RJIT은 Unix 플랫폼에서 x86-64 아키텍처만 지원합니다.
- MJIT과 달리 런타임에 C 컴파일러가 필요하지 않습니다.
- RJIT은 실험 목적으로 존재합니다.
- 프로덕션 환경에서는 YJIT을 계속 사용해야 합니다.
- Ruby용 JIT 개발에 관심이 있다면 RubyKaigi 2023 3일차에 있었던 k0kubun의 발표를 확인하세요.
M:N 스레드 스케줄러
- M:N 스레드 스케줄러가 추가됩니다. [Feature #19842]
- M개의 Ruby 스레드를 N개의 네이티브 스레드(OS 스레드)로 관리하므로 스레드의 생성, 관리 비용이 절감됩니다.
- C 확장 호환성에 문제가 발생할 가능성이 있으므로, 주 Ractor에서의 기본 설정은 비활성입니다.
RUBY_MN_THREADS=1
환경 변수를 사용해 주 Ractor에서 M:N 스레드를 활성화할 수 있습니다.- M:N 스레드는 주 Ractor가 아닌 Ractor에서 항상 활성화되어 있습니다.
RUBY_MAX_CPU=n
환경 변수는N
(네이티브 스레드의 최대 개수)의 최댓값을 지정합니다. 기본값은 8입니다.- 하나의 Ractor에서는 하나의 Ruby 스레드만 실행 가능하므로 실제로 사용되는 네이티브 스레드의 개수는
RUBY_MAX_CPU
에서 지정한 숫자나 실행 중인 Ractor의 개수보다 작습니다. 그러므로 (대다수를 차지하는) 단일 Ractor 애플리케이션에서는 단 하나의 네이티브 스레드를 사용합니다. - 블로킹 동작을 처리하기 위해,
N
개 이상의 네이티브 스레드가 사용될 수 있습니다.
- 하나의 Ractor에서는 하나의 Ruby 스레드만 실행 가능하므로 실제로 사용되는 네이티브 스레드의 개수는
성능 향상
defined?(@ivar)
가 객체 형상에 맞게 최적화됩니다.Socket.getaddrinfo
와 같은 이름 해결을 (POSIX 스레드가 사용 가능한 환경에서) 중단할 수 있게 됩니다. [Feature #19965]- 가비지 컬렉터의 몇몇 성능 향상
- 오래된 객체가 참조하는 어린 객체는 이제 즉시 오래된 객체로 마킹되지 않습니다. 이 동작은 메이저 GC의 빈도를 상당히 낮춰줍니다. [Feature #19678]
- 메이저 GC를 야기하는 보호받지 않은 객체의 숫자를 제어하는
새로운 변수
REMEMBERED_WB_UNPROTECTED_OBJECTS_LIMIT_RATIO
가 추가되었습니다. 기본값은0.01
(1%)입니다. 이는 메이저 GC의 빈도를 상당히 낮춰줍니다. [Feature #19571] Time
,Enumerator
,MatchData
,Method
,File::Stat
,BigDecimal
등의 주요 타입에 부족했던 쓰기 보호가 구현되었습니다. 이는 마이너 GC에 걸리는 시간과 메이저 GC 빈도를 상당히 줄여줍니다.Hash
,Time
,Thread::Backtrace
,Thread::Backtrace::Location
,File::Stat
,Method
등의 대부분의 주요 클래스는 이제 가변 폭 할당을 사용합니다. 이는 해당 클래스의 할당 및 할당 해제 속도를 빠르게 만들며, 메모리 사용량과 힙 파편화를 줄여줍니다.- 가비지 컬렉터에 약한 참조가 추가됩니다. [Feature #19783]
그 이외의 3.2 이후로 주목할 만한 변경
IRB
IRB에 여러 개선 사항이 추가됩니다. 다음과 같은 내용이 포함됩니다.
- 고급
irb:rdbg
통합 기능은pry-byebug
와 동등한 디버깅 경험을 제공합니다. (문서). ls
와show_source
,show_cmds
와 같은 명령어에서 페이징을 지원합니다.ls
,show_source
명령에서 더 정확하고 유용한 정보를 제공합니다.- 타입 분석을 사용하는 실험적인 자동 완성 기능을 제공합니다. (문서).
- Reline::Face 클래스의 도입으로 자동완성 다이얼로그의 글씨 색과 스타일이 변경 가능합니다. (문서)
또한, IRB는 광범위한 리팩토링을 거쳤으며, 향후 개선이 용이하도록 많은 버그 수정이 이루어졌습니다.
더 자세한 변경은 Ruby 3.3 IRB의 큰 도약을 공개합니다를 참조하세요.
호환성 문제
주의: 기능 버그 수정은 포함되어 있지 않습니다.
-
블록에서 매개변수 없이
it
을 호출하는 기능은 삭제 예정입니다. Ruby 3.4부터it
은 첫 번째 블록 인수를 가리킵니다. [Feature #18980] -
Regexp::new
는 최대 3개의 인수를 받았지만 이제 2개까지 허용합니다. 이는 Ruby 3.2에서 폐기예정이었습니다. [Bug #18797]
삭제된 환경 변수
폐기 예정이었던 환경 변수가 삭제됩니다.
- 환경 변수
RUBY_GC_HEAP_INIT_SLOTS
는 폐기 예정이었으며 아무 효과도 없습니다. 대신 환경 변수RUBY_GC_HEAP_{0,1,2,3,4}_INIT_SLOTS
를 사용해 주세요. [Feature #19785]
Stdlib 호환성 문제
ext/readline
폐기
- 우리는
ext/readline
API와 호환되는 순수 Ruby 구현인reline
을 가지고 있습니다. 앞으로는reline
에 의존할 것입니다.ext/readline
을 사용해야 하는 경우,gem install readline-ext
를 사용하여 rubygems.org를 통해ext/readline
을 설치할 수 있습니다. - 이제 더 이상
libreadline
또는libedit
과 같은 라이브러리를 설치할 필요가 없습니다.
표준 라이브러리 갱신
사용자가 미래의 Ruby 버전에서 내장될 예정의 gem을 Gemfile이나 gemspec에 추가하지 않고 직접 require
할 때 RubyGems와 Bundler가 경고 문구를 출력합니다.
bootsnap gem을 사용하고 있다면, 해당 경고를 생략합니다. 적어도 한 번은 DISABLE_BOOTSNAP=1
환경 변수와 함께 애플리케이션을 실행하길 권장합니다. 이는 이번 버전에만 해당되는 제약입니다.
다음 라이브러리가 대상입니다.
- abbrev
- base64
- bigdecimal
- csv
- drb
- getoptlong
- mutex_m
- nkf
- observer
- racc
- resolv-replace
- rinda
- syslog
다음 기본 gem이 추가되었습니다.
- prism 0.19.0
다음 기본 gem이 갱신되었습니다.
- RubyGems 3.5.3
- abbrev 0.1.2
- base64 0.2.0
- benchmark 0.3.0
- bigdecimal 3.1.5
- bundler 2.5.3
- cgi 0.4.1
- csv 3.2.8
- date 3.3.4
- delegate 0.3.1
- drb 2.2.0
- english 0.8.0
- erb 4.0.3
- error_highlight 0.6.0
- etc 1.4.3
- fcntl 1.1.0
- fiddle 1.1.2
- fileutils 1.7.2
- find 0.2.0
- getoptlong 0.2.1
- io-console 0.7.1
- io-nonblock 0.3.0
- io-wait 0.3.1
- ipaddr 1.2.6
- irb 1.11.0
- json 2.7.1
- logger 1.6.0
- mutex_m 0.2.0
- net-http 0.4.0
- net-protocol 0.2.2
- nkf 0.1.3
- observer 0.1.2
- open-uri 0.4.1
- open3 0.2.1
- openssl 3.2.0
- optparse 0.4.0
- ostruct 0.6.0
- pathname 0.3.0
- pp 0.5.0
- prettyprint 0.2.0
- pstore 0.1.3
- psych 5.1.2
- rdoc 6.6.2
- readline 0.0.4
- reline 0.4.1
- resolv 0.3.0
- rinda 0.2.0
- securerandom 0.3.1
- set 1.1.0
- shellwords 0.2.0
- singleton 0.2.0
- stringio 3.1.0
- strscan 3.0.7
- syntax_suggest 2.0.0
- syslog 0.1.2
- tempfile 0.2.1
- time 0.3.0
- timeout 0.4.1
- tmpdir 0.2.0
- tsort 0.2.0
- un 0.3.0
- uri 0.13.0
- weakref 0.1.3
- win32ole 1.8.10
- yaml 0.3.0
- zlib 3.1.0
다음 기본 gem이 내장됩니다.
- racc 1.7.3
다음 내장 gem이 갱신됩니다.
- minitest 5.20.0
- rake 13.1.0
- test-unit 3.6.1
- rexml 3.2.6
- rss 0.3.0
- net-ftp 0.3.3
- net-imap 0.4.9
- net-smtp 0.4.0
- rbs 3.4.0
- typeprof 0.21.9
- debug 1.9.1
기본 gem 또는 내장 gem에 대한 자세한 내용은 Logger와 같은 GitHub 릴리스 또는 변경 로그에서 확인하세요.
더 자세한 내용은 NEWS나 커밋 로그를 확인해 주세요.
이러한 변경사항에 따라, Ruby 3.2.0 이후로 파일 5532개 수정, 326851줄 추가(+), 185793줄 삭제(-)가 이루어졌습니다!
메리 크리스마스, 해피 홀리데이, 그리고 Ruby 3.3과 함께 프로그래밍을 즐겨보세요!
다운로드
-
https://cache.ruby-lang.org/pub/ruby/3.3/ruby-3.3.0.tar.gz
SIZE: 22065999 SHA1: 1a7e56851bf29bda1183aca99b3b323c58e0187b SHA256: 96518814d9832bece92a85415a819d4893b307db5921ae1f0f751a9a89a56b7d SHA512: 26074009b501fc793d71a74e419f34a6033c9353433919ca74ba2d24a3de432dbb11fd92c2bc285f0e4d951a6d6c74bf5b69a2ab36200c8c26e871746d6e0fc6
-
https://cache.ruby-lang.org/pub/ruby/3.3/ruby-3.3.0.tar.xz
SIZE: 16345456 SHA1: c8f68e1b0a114b90460a0b44165a3b2f540fa5b6 SHA256: 676b65a36e637e90f982b57b059189b3276b9045034dcd186a7e9078847b975b SHA512: 7959c5753bfa0bfc4d6d74060869aabbe9815c1c97930659da11b917ee0803ddbbd80e869e00c48b8694b4ba48709c3b6493fd045568e36e902616c35ababf01
-
https://cache.ruby-lang.org/pub/ruby/3.3/ruby-3.3.0.zip
SIZE: 26935108 SHA1: a433eef1d7f96daeaf3b4cb842d0ed2dd82e7dc1 SHA256: 0e6563f679dd3694732eb3addf9de681c67b584602ac574376b60e7a509d2cd8 SHA512: a94a85937a14b217c1f4b90d24185289ed4aee79239c4f3eecf8034d3fd34e65ee8d66869473857ed153067188adc9b70c0471e4ebe842c9f98ef60c34090450
Ruby는
Ruby는 1993년에 Matz(마츠모토 유키히로) 씨가 처음 개발했고, 현재는 오픈 소스로서 개발되고 있습니다. 여러 플랫폼에서 동작하며, 특히 웹 개발에서 전 세계적으로 이용되고 있습니다.