Ruby 初心者スレッド Part 67
プログラミング言語 Rubyについての、初心者向けスレです。質問・要望・雑談などをどうぞ 質問するときは、OSやRubyのバージョン、エラーメッセージを書いて下さい。 Ruby on Rails については、WEBプログラミング板で 前スレ Ruby 初心者スレッド Part 66 https://mevius.5ch.net/test/read.cgi/tech/1578068134/ るりまサーチ (リファレンス検索) http://rurema.clear-code.com/ Rubyist Magazine - るびま http://jp.rubyist.net/magazine/ 逆引きRuby http://www.namaraii.com/rubytips/ Ruby コミュニティ公式 https://www.ruby-lang.org/ VIPQ2_EXTDAT: checked:vvvvv:1000:512:: EXT was configured プログラミング自体初心者です ファイバーを全く理解していなかったので勉強してみているのですが class Test def initialize @fiber = Fiber.new {test} end def test p "これがファイバーのブロックで与えた処理だ" Fiber.yield p "理解したかな???" end def fiber_test(count = 60) @fiber.resume p "帰ってきたぞ" @fiber.resume p "再び帰ってきたぞ" count.times do {Fiber.yield} end end このコードを組んで別の場所でTest.new.fiber_testとインスタンスを作成して呼び出してみたのですが @fiber.resumeでtestメソッドへ移動し Fiber_yieldでfiber_testメソッドへ戻ることはわかったのですが testメソッドで@fiber.resumeを実行すると親子関係がおかしくなるからかFiberErrorを起こすのに対し fiber_testメソッド側でFiber_yieldを実行すると一瞬画面が止まるだけでコンソールにも何も表示されません これは何が起きているのでしょうか? 引数のcountを増やすほど長く画面が固まります >>20 >>19 を読み間違えていたので訂正 CSV.parse '"0123",1', converters: ->(f, i){i.index.zero? ? f : f.to_i} # => [["0123, 1]] Fiber は、coroutine みたいなものか 初心者がやるものじゃない。 10年以上やってから、やるもの f = Fiber.new do n = 0 loop do puts "子の前 : #{ n }" Fiber.yield( n ) # 親へ puts "子の後 : #{ n }" n += 1 end end 2.times do |idx| puts "親の前 : #{ idx }" puts "親の値 : #{ f.resume }" # 子へ puts "親の後 : #{ idx }" end 出力 親の前 : 0 子の前 : 0 親の値 : 0 親の後 : 0 親の前 : 1 子の後 : 0 # 要注意! 子の前 : 1 親の値 : 1 親の後 : 1 >>22 そのコードはfiber_testメソッドのFiber_yieldしてるところで can't yield from root fiberのエラーになるよ エラーにならないならrubyのバージョンと再現可能なコードを書いて Fiber.yield と @var.resume で処理の投げあいをしてるだけ で、resumeは変数のメソッドだから変数が参照できればどこからでも呼べるが、 Fiber.yieldっていうのは、Fiber.new{ ここ }の中でしか呼び出せない なお初心者がFiberを使ったコードを書いても普通にloopで書いたほうがいいもんを煩雑にさせるだけだから今覚えてもすぐ明日から使うぞーってなる機能ではない Ruby使った事ない勢です 知人から恐らくruby(のgem ocra)でコンパイルされたexeファイルを修正してほしいと頼まれたのですが これ逆アセンブルする方法無いですか? >>29 無知で申し訳ないです サクラエディタでexeとか怪しいtmp開いてみたけど見られないんですよね 解凍というのはどう言うことでしょうか?ocraにそういったコマンドがあるのでしょうか? それとも実はzipでコンパイルされてるとか? >>30 自己解凍形式 実行すれば一時ディレクトリにソースも解凍されてる >>20 これにcoverters : :numeric付けてみたけどカンマを含むデータがうまく処理できず >>21 いや、それが知りたいんだが >>23 こうすりゃいいのか。ありがとう 0 で始まる数値は8進数だから、:numeric ではバグル 012 は、10 になる。 一方、09 は8進数では存在しないから、9 となる require "csv" text = <<'TEXT' "09",1 "012",2 TEXT csv = CSV.new( text ) csv.convert( :numeric ) p csv.read #=> [ [9.0, 1], [10, 2] ] csv=<<-TXT "01.23",1 "09",1 "012",2 TXT a=csv.each_line.map{|n| eval"[#{n}]" } p a >>35 そうそう、これもどうにかしたかった >>36 nullがあるとダメじゃね? >>19 の例が単純すぎたのでもうちょっとマシな例 "0123456789","5CH CO.,LTD","2021/01/01",12345.67 "0123456790","2CH.COM",,0.00 つーか、あとは自分でできるやろ。 やりたいようにやれ。 うむ。>>23 で解決した あとはお題として書いてみた ruby初心者です。 調べたけどわからないので教えて下さい neriでexeファイルを作ろうとすると、簡単なもので試したのですが、 internal:C:/Ruby30-x64/lib/ruby/3.0.0/rubygems/core_ext/kernel_require.rb>:85:in `require': cannot load such file -- Win32API (LoadError) のエラーが出ます。 Windows 10 ruby 3.0 x64 Bat_To_Exe_Converter v3.2 >>42 gem list にWin32APIはあるのか? >>43 gem listでwin32-api(1.9.2 universal-mingw32) はありました。 「ruby `require': cannot load such file -- Win32API」で検索! Win32APIはruby3.0から廃止 当然Win32APIを使ってたgemsはRuby3.0では作者がメンテナンスするまで動かないからneri作者の尻ひっぱたくか前のバージョン使え >>47 いや、さすがにWin32APIはしゃあない。 むしろ、とっくに避けていなければならんかったレベル。 >>46 なるほど。ありがとうございます。 わからず、ずっと悩んでました。 環境変数の設定を一切せず rubyの起動オプションですべて明示するというやり方って 現実的になにかトラブルが発生したりしますか? コマンドラインの長さとか? 最近ならもうだいじょうぶかな? そんなに長い指定はないので今の所問題く動いているっぽいです ありがとうございます 教えていただきたいのですが、rubyのclassでそのclassに存在するプロパティ一覧を取得する方法ありますか? 以下の様なクラスを作ってます。 class Data def self.csv_header return 'name,id,data1,data2' end attr_reader :name, :id, :data1, :data2 def initialize(name:, id:, data1:, data2) @name = name @id = id @data1 = data1 @data2 = data2 end def to_csv return "#{@name},#{@id},#{@data1},#{@data2}" end end 要するに、何か複数のデータを持つクラスがあり、それをcsvとして出力したいです。 そのヘッダーを出力する機能をクラスメソッドとして持たせたいのですが、このクラスメソッド(csv_header)の中身を手で書くのではなく、javaで言う所のreflectionか何かを使って出力する方法はありますか? よろしくお願いいたします >>54 hoge = Data.new('name', id: 0, data1: 1, data2: 2) ivs = hoge.instance_variables ivs.each{|iv| puts "%s => %p" % [iv, hoge.instance_variable_get(iv)]} # => name => "name" id => 0 data1 => 1 data2 => 2 黒魔術・メタプログラミングで、よく見る インスタンス変数用のinstance_variables、 クラス/モジュールには、クラス変数(@@these_things)用のclass_variables >>54 そういう用途はStruct使うといいんでない? Foo = Struct.new("Foo", :name, :id, :data1, :data2) foo = Foo.new("foo", 1, "123", "456") foo.to_h.keys #=> [:name, :id, :data1, :data2] foo.to_h.keys.to_csv #=> "name,id,data1,data2¥n" Structは初期化時の引数の数が一致してなくてもエラーにならずにnilがデフォルト値になる それがいやな場合は継承してコンストラクタを上書き class Bar < Struct.new("Bar", :name, :id, :data1, :data2) def initialize(name, id, data1=nil, data2=nil) super end end 質問させてください。 以下のようなデータがあるとして。 data = { "id": "price_xxxxxxx", "object": "price", "active": true, "billing_scheme": "per_unit", "created": 1234567, "currency": "usd", "metadata": { "commission": "2" } } このデータを p data.metadata.commission とすると4が表示されるようにしたいです。 どうやって変換すればいいでしょうか? 普通にブラケットでアクセスするかdigしたほうがいい気がするが どうしてもドットでアクセスしたいなら再帰的にOpenStructに変換する data = JSON.parse(data.to_json, object_class: OpenStruct) data.metadata.commission #=> "2" ハッシュだから、hash.x.xyz は無理だろ Effective Ruby の項目10 に書いてある、 構造物はハッシュじゃなくて、クラス・Struct を使おうというやつだろ ただ、その都度、構造が変わるものは、ハッシュの方が良いし、 構造がネストしている場合も難しい require "json" json = <<'JSON' { "a": "abc", "x": { "xyz": "2" }} JSON p hash = JSON.parse( json ) #=> {"a"=>"abc", "x"=>{"xyz"=>"2"}} p hash[ "x" ][ "xyz" ] #=> 2 >>61 , >>62 ありがとうございます。 背景を書いてませんでしたが、rspec で必要なデータなのです。。 後出しですいません。 なので data.metadata.commission で確認できることが必要なので、Struct(もしくはOpenStruct)で、変換させてみます。 >>59 def hoge!(obj) return unless Hash === obj def obj.method_missing(name, *_) = self[name] obj.each_value{Hash === _1 && hoge!(_1)} end hoge!(data) p data.metadata.commission # => "2" # 冗談だけど >>54 です。 アドバイスありがとうございます。 私も色々と調べたのですが、オブジェクトを1回作らないと難しいかもしれませんね。 仮にデータが全くなかった場合、csvのヘッダーのみ記述されたcsvファイルを出力したいと考えています。 しかし、データが全くないと、オブジェクトが無いため、オブジェクトからプロパティを取得する方法が使えません。 もちろん、ダミーのオブジェクトを作ればOKですが、少々不自然かと思いました。 Structは知らなかったので、勉強させていただきます。 ありがとうございました。 https://twitter.com/yukihiro_matz/status/1358048924141973504?s=20 > 世界一周クルーズの広告が表示され、妻と「コロナ収まったら行きたいね」 世界一周は最低500万、上は数千万にもなる。 うっかり金持ち自慢をしてしまう… https://twitter.com/5chan_nel (5ch newer account) インストールしなくていい機能縮小版rubyを作って、awkを代替えできるようにする というのはどうでしょう >>68 そんな用途なら、互換性が期待できないRubyよりも、信頼できるPerlのほうがええやろ。 すまん唐突だけど質問させて eql口頭でってなんで読んだら良いの? ミスった eqlって口頭では何て発音したら良いの? です イコールって読んでる equal?とeql?が入り乱れて使われてるところはしらんが。 まぁrubyしてるならequalとeql両方知ってるはずだし短いイコールの方、とかでも伝わるんじゃ >>72 rubyやり始めたところで、eqlがイコールって気づかなかった。短い方のイコールって良いですね!ありがとうございます。 >>73 自分もイーキューエルだて頭の中で読んでた >>74 なんでやーーー!! すみません、質問です hash生成時に、 str={"key1" = "val1"} ではエラーになる理由を教えてください。 p hash = { "a" => "b" } #=> {"a"=>"b"} 全角の{}を使うな。 文字列以外は半角英数字のみ >>77 すみません、携帯からで全角と半角を間違えてしまいました。半角として見てくださいm(_ _)m p h_1 = { "a" => "b" } #=> {"a"=>"b"} p h_2 = { "あ": "い" } #=> {:あ=>"い"} p h_3 = { one: 1, "two": "2" } #=> {:one=>1, :two=>"2"} 新しめの書き方では、2, 3 みたいに書ける : を使うと、文字列のキーがシンボルになる "two": が、:two になる one: はシンボルのまま 書いていただいた方法でないと生成できないのですね 丁寧な回答ありがとうございました AtCoderの解答見てると、rubyで巨大配列を扱うような時に激遅になりがちな気がする 配列アクセスの遅さなのかループの遅さなのかわからんが JIT があるから、1秒間で100万回ループすると、ネイティブコードにコンパイルされて、 1秒間で1,000万回ループできるようになる irbにおいて、 def test; p test; end とだけ打ち込みました。 すると、Enumerable.methods.grep(/test/) => [:test] が得られます。 何故、testメソッドが勝手にEnumerableのメソッドになるのか教えていただけないでしょうか。 環境は ruby 2.7.1p83 (2020-03-31 revision a0c7c23c9c) [x86_64-darwin18] です。 https://docs.ruby-lang.org/ja/latest/class/main.html >トップレベルで定義したメソッドは Object の private インスタンスメソッドとして定義されます。 って思ったけどこっちの環境でもprivateにならないな はて、何故でしょう ruby 2.6.6 [x86_64-linux] なら、 def test; p test; end p Enumerable.methods.grep(/test/) #=> [ ] p test が文法エラーになるのでは? test という識別子が存在しないから 0084です。 def test; p test; end ではなく、 def test; p “test”; end でした。 大変申し訳ありませんでした。 ほかのメソッドでも同じような動きを確認しています。今日は立て込んでしまっているので、追って別のメソッドの例も提示します。 >>0085さん、確認ありがとうございます。 私は初心者なので、以下の認識で正しいか、何か例外があるのか不安でお尋ねしました。認識は正しかったようですので、もう少し調べてみます。 >トップレベルで定義したメソッドは Object の private インスタンスメソッドとして定義されます。 privateかどうかをどうやって確認したの? 要再現コード ruby 2.6 => [ ] ruby 2.7 => [:test] ruby 3.0 => [ ] irb(main):001:0> def foo; p "Foo!" end => :foo irb(main):002:0> "abc".foo "Foo!" => "Foo!" irb(main):003:0> 3.14.foo "Foo!" => "Foo!" irb(main):004:0> [].foo "Foo!" => "Foo!" 90です。 パソコンを再起動して、.irb_historyをtouchで初期化したのち、 以下を実行しても同じ結果でした。 irb(main):001:0> def test; p "test"; end => :test irb(main):002:0> Enumerable.methods.grep(/test/) => [:test] irb(main):010:0> exit $ rbenv -v rbenv 1.1.2 $ rbenv versions system * 2.7.1 (set by /Users/someone/.rbenv/version) 別のメソッドも作ってみましたが、結果は同じでした。 irb(main):001:1* def mul(a,b) irb(main):002:1* a*b irb(main):003:0> end => :mul irb(main):004:0> Enumerable.methods.grep(/mul/) => [:mul] irb(main):006:0> exit $ irb irb(main):001:0> Enumerable.methods.grep(/mul/) => [] 90です。 92さんがおっしゃる通り、環境を作り直すこととします。 私の疑問を受けて、再現性について試していただいた皆様、 ご協力ありがとうございました。 irb(とpry)でトップレベルで定義するとなんかおかしい $ cat test.rb def hoge; end puts 'public: %p' % [methods.grep(/hoge/)] puts 'private: %p' % [private_methods.grep(/hoge/)] $ ruby -v test.rb ruby 2.6.1p33 (2019-01-30 revision 66950) [x64-mingw32] public: [] private: [:hoge] $ irb irb(main):001:0> load("test.rb") public: [] private: [:hoge] => true irb(main):002:0> def fuga;end => :fuga irb(main):003:0> puts 'public: %p' % [methods.grep(/fuga/)] public: [:fuga] => nil irb(main):004:0> puts 'private: %p' % [private_methods.grep(/fuga/)] private: [] => nil irb, pry は特殊なのかな? 漏れは、VSCode のCode Runner で、 選択したコード片、またはファイル全体を、右クリックメニューから実行してる これが楽 WEB+DB vol.121 Ruby 3 特集、30 ページ分 Ruby on Rails 6 の本を書いている、2人の新刊 Go言語 ハンズオン、掌田津耶乃、2021/3/6 Elixir実践ガイド、黒田努、2021/2/5 Ruby からGoか、Elixirか、どっちへ進むべきか? mruby, Rust もあるけど Ruby勉強始めたんですが ↓この部分が毎回書きづらいです コード補完とかってないんですか? a.map{|x|x+1}←の|x|の部分 >>101 Ruby 2.7 以降は、番号指定パラメータが導入されたので a.map{_1+1} プログラミング言語Crystal、初のメジャーリリースとなるバージョン1.0を公開 最近は全言語で、VSCode の拡張機能・Remote Container で、Docker を使う Windows 10 Home でも、WSL2 が出来るようになった Elixir でも、無名関数をキャプチャー演算子・& を使って、短く書ける fn a, b -> a * b end &(&1 * &2) & &1 * &2 rubyでExcelへデータを出力しようとしているんですが、 どうしてもできないことがあり質問します。 ruby -v 3.0.0p0 rubyでExcelのデータを検索するときはFindを使うと思うのですが、 日付項目(シリアル値)を検索するときはどうすればいいのでしょうか。 以下をやってみましたが、検索できませんでした。 find('3/30') ・・・Excelの表示を指定 find('4321') ・・・シリアル値を指定 >>110 使ってるライブラリが分からないと答えようがない >>111 失礼しました。 ライブラリはWin32OLEです >>110 上はできた。下はダメだった ちなみに表示が3月30日の場合、find('3/30') はいけたが find('3月30日') はダメ >>113 え、いけました?? 家に帰ったらもう一回やってみます。 教えて頂いてありがとうございます。 >>17 on ruby rails と出力されると思いきや、ruby “on ¥n rails”だって。 なんでだ? Here Documentの仕様ってどうなってんの? >>115 終端行までなんだから、その前にある改行文字は含まれてあたりまえやろ。 ヒアドキュメントの評価値とは別に #{print "ruby "} で"ruby "とprintされるクソみたいなコードだから まあなぞなぞみたいなもんでしょ クソコードとかいったら身も蓋もない Loggerで出力する内容を奪い取って、他のメソッド呼び出しにするってどう言う実装したらいいですか? 例えば全てをTest::any::log(msg)を内部的に呼び出して終わるみたいなLoggerを作るにはどうしたらいいですか? Logger.new(STDERR)だと全て標準エラーに出力ですが、これを全てTest::any::log(msg)の呼び出しで標準エラーには何も出さないloggerを作りたいのです C:/Ruby25-x64/lib/ruby/2.5.0/fileutils.rb では、 インスタンス変数に、標準エラーを代入してる @fileutils_output ||= $stderr @fileutils_output = $stderr 漏れは、それを真似て、 DryRun の時だけ、noop: true, verbose: true を付けて、 標準出力から標準エラーに切り替えるような、モジュールを作っている module MyFileUtils @fileutils_output = $stdout extend self def f( *args, **options ) # 配列・ハッシュ p @fileutils_output @fileutils_output.puts( args, options ) end module DryRun include MyFileUtils @fileutils_output = $stderr extend self def f( *args, **options ) super( *args, **options, noop: true, verbose: true ) end end end MyFileUtils.f( 1, 2, a: 8, b: 9 ) MyFileUtils::DryRun.f( 1, 2, a: 8, b: 9 ) Ruby 逆引きハンドブックには、こう書いてある Logger.new の引数には、ログファイルパスか、 write, close が定義された、IO などのオブジェクトを指定する read.cgi ver 07.5.5 2024/06/08 Walang Kapalit ★ | Donguri System Team 5ちゃんねる