Nie mówi, że stała jest dynamiczna. Mówi, że zadanie jest dynamiczne.
sepp2k
Odpowiedzi:
141
Twój problem polega na tym, że za każdym razem, gdy uruchamiasz metodę, przypisujesz stałą nową wartość. Jest to niedozwolone, ponieważ powoduje, że stała nie jest stała; mimo że zawartość łańcucha jest taka sama (w każdym razie w tej chwili), sam obiekt string jest inny przy każdym wywołaniu metody. Na przykład:
def foo
p "bar".object_idend
foo #=> 15779172
foo #=> 15779112
Być może gdybyś wyjaśnił swój przypadek użycia - dlaczego chcesz zmienić wartość stałej w metodzie - moglibyśmy pomóc Ci w lepszej implementacji.
Może wolisz mieć zmienną instancji w klasie?
classMyClassclass<<self
attr_accessor :my_constantenddef my_methodself.class.my_constant ="blah"endend
p MyClass.my_constant #=> nilMyClass.new.my_method
p MyClass.my_constant #=> "blah"
Jeśli naprawdę chcesz zmienić wartość stałej w metodzie, a twoją stałą jest String lub Array, możesz „oszukać” i użyć #replacemetody, aby spowodować, że obiekt przybierze nową wartość bez faktycznej zmiany obiektu:
classMyClass
BAR ="blah"def cheat(new_bar)
BAR.replace new_barendend
p MyClass::BAR #=> "blah"MyClass.new.cheat "whee"
p MyClass::BAR #=> "whee"
OP nigdy nie powiedział, że chce zmienić wartość stałej, ale chciał tylko przypisać wartość. Częstym przypadkiem użycia prowadzącym do tego błędu Rubiego jest tworzenie wartości w metodzie z innych zasobów czasu wykonywania (zmienne, argumenty wiersza poleceń, ENV), zwykle w konstruktorze, np def initialize(db,user,password) DB=Sequel.connect("postgres://#{user}:#{password}@localhost/#{db}") end. To jeden z tych przypadków, w których Ruby nie ma prostego sposobu.
Arnaud Meuret
2
@ArnaudMeuret W tym przypadku potrzebujesz zmiennej instancji (np. @variable), A nie stałej. W przeciwnym razie będziesz ponownie przypisywać za DBkażdym razem, gdy utworzysz nową instancję tej klasy.
Ajedi32
2
@ Ajedi32 Ta sytuacja zwykle wynika z zewnętrznych ograniczeń, a nie z wyborów projektowych, takich jak mój przykład z Sequelem. Chodzi mi o to, że przypisywanie wartości stałej jest dozwolone przez Rubiego w niektórych zakresach, a nie w innych. Kiedyś do programisty należał mądry wybór, kiedy wykonać zadanie. Ruby się w tym zmienił. Nie dla każdego jest dobre.
Arnaud Meuret
2
@ArnaudMeuret Przyznaję, że nigdy wcześniej nie korzystałem z Sequela, więc nie mogę tego powiedzieć ze 100% pewnością, ale przeglądając dokumentację Sequela, nie widzę nic, co mówi, że MUSISZ przypisać wynik Sequel.connectdo stałej o nazwie DB . W rzeczywistości dokumentacja wyraźnie mówi, że to tylko zalecenie. Dla mnie to nie brzmi jak zewnętrzne ograniczenie.
Ajedi32,
@ Ajedi32 1) Nigdy tego nie napisałem (nazwa stałej lub nawet to, że musiałeś ją gdzieś przechowywać) to tylko przykład 2) Ograniczenie polega na tym, że twoje oprogramowanie może nie mieć niezbędnych informacji, dopóki zwykle nie jesteś w dynamicznym kontekście .
Arnaud Meuret
69
Ponieważ stałe w Rubim nie powinny być zmieniane, Ruby odradza przypisywanie do nich części kodu, które mogą być wykonywane więcej niż jeden raz, takich jak metody wewnętrzne.
W normalnych okolicznościach należy zdefiniować stałą wewnątrz samej klasy:
Znowu jednak const_setnie jest czymś, do czego naprawdę powinieneś uciekać się w normalnych okolicznościach. Jeśli nie masz pewności, czy naprawdę chcesz przypisywać stałe w ten sposób, możesz rozważyć jedną z następujących alternatyw:
Zmienne klasowe
Zmienne klasowe na wiele sposobów zachowują się jak stałe. Są właściwościami klasy i są dostępne w podklasach klasy, w której zostały zdefiniowane.
Różnica polega na tym, że zmienne klas mają być modyfikowalne i dlatego można je przypisać do metod wewnętrznych bez problemu.
classMyClassdefself.my_class_variable
@@my_class_variableenddef my_method
@@my_class_variable="foo"endendclassSubClass<MyClassendMyClass.my_class_variable
#=> NameError: uninitialized class variable @@my_class_variable in MyClassSubClass.my_class_variable
#=> NameError: uninitialized class variable @@my_class_variable in MyClassMyClass.new.my_method
MyClass.my_class_variable #=> "foo"SubClass.my_class_variable #=> "foo"
Atrybuty klas
Atrybuty klasy to rodzaj „zmiennej instancji w klasie”. Zachowują się trochę jak zmienne klasowe, z tym wyjątkiem, że ich wartości nie są współdzielone z podklasami.
I dla kompletności powinienem prawdopodobnie wspomnieć: jeśli chcesz przypisać wartość, która może być określona tylko po utworzeniu instancji twojej klasy, istnieje duża szansa, że faktycznie szukasz zwykłej starej zmiennej instancji.
Nie możesz nazwać zmiennej wielkimi literami, inaczej Ruby założy, że jest to stała i będzie chciał, aby zachowała stałą wartość. W takim przypadku zmiana jej wartości byłaby błędem i „błędem dynamicznego przypisania stałej”. Z małymi literami powinno być dobrze
Rubiemu nie podoba się to, że przypisujesz stałą wewnątrz metody, ponieważ istnieje ryzyko ponownego przypisania. Kilka odpowiedzi SO przede mną daje alternatywę przypisywania jej poza metodą - ale w klasie, co jest lepszym miejscem do jej przypisania.
Weicome dla SO John. Możesz rozważyć ulepszenie tej odpowiedzi, dodając przykładowy kod tego, co opisujesz.
Cleptus,
0
Wielkie podziękowania dla Doriana i Phrogza za przypomnienie mi o metodzie tablicowej (i haszującej) #replace, która może „zastąpić zawartość tablicy lub skrótu”.
Pogląd, że wartość STAŁA można zmienić, ale z irytującym ostrzeżeniem, jest jednym z niewielu błędnych kroków koncepcyjnych Rubiego - powinny one albo być w pełni niezmienne, albo całkowicie odrzucić stałą ideę. Z punktu widzenia programisty stała jest deklaratywna i zamierzona, jest sygnałem dla innych, że „ta wartość jest naprawdę niezmienna po zadeklarowaniu / przypisaniu”.
Czasami jednak „oczywista deklaracja” faktycznie wyklucza inne, przydatne w przyszłości możliwości. Na przykład...
Tam są przypadki legalnego wykorzystania gdzie wartość danej „Constant za” może naprawdę trzeba być zmieniony, na przykład: ponowne ładowanie ARGV z REPL podobny szybkiej pętli, a następnie ponowne uruchomienie ARGV thru więcej (późniejszej) OptionParser.parse! rozmowy - voila! Daje „argumentom wiersza poleceń” zupełnie nowe dynamiczne narzędzie.
Praktyczny problem jest albo z przypuszczalnego założeniu, że „ARGV musi być ciągle”, lub w oddzielnym metody initialize optparse, która twardych koduje przyporządkowanie ARGV do @default_argv przykład var do dalszej obróbki - że tablica (ARGV) rzeczywiście powinien być parametrem zachęcającym do ponownej analizy i ponownego wykorzystania, w stosownych przypadkach. Właściwa parametryzacja z odpowiednią wartością domyślną (powiedzmy ARGV) pozwoliłaby uniknąć konieczności zmiany „stałej” wartości ARGV. Zaledwie 2 grosze myśli ...
Odpowiedzi:
Twój problem polega na tym, że za każdym razem, gdy uruchamiasz metodę, przypisujesz stałą nową wartość. Jest to niedozwolone, ponieważ powoduje, że stała nie jest stała; mimo że zawartość łańcucha jest taka sama (w każdym razie w tej chwili), sam obiekt string jest inny przy każdym wywołaniu metody. Na przykład:
Być może gdybyś wyjaśnił swój przypadek użycia - dlaczego chcesz zmienić wartość stałej w metodzie - moglibyśmy pomóc Ci w lepszej implementacji.
Może wolisz mieć zmienną instancji w klasie?
Jeśli naprawdę chcesz zmienić wartość stałej w metodzie, a twoją stałą jest String lub Array, możesz „oszukać” i użyć
#replace
metody, aby spowodować, że obiekt przybierze nową wartość bez faktycznej zmiany obiektu:źródło
def initialize(db,user,password) DB=Sequel.connect("postgres://#{user}:#{password}@localhost/#{db}") end
. To jeden z tych przypadków, w których Ruby nie ma prostego sposobu.@variable
), A nie stałej. W przeciwnym razie będziesz ponownie przypisywać zaDB
każdym razem, gdy utworzysz nową instancję tej klasy.Sequel.connect
do stałej o nazwie DB . W rzeczywistości dokumentacja wyraźnie mówi, że to tylko zalecenie. Dla mnie to nie brzmi jak zewnętrzne ograniczenie.Ponieważ stałe w Rubim nie powinny być zmieniane, Ruby odradza przypisywanie do nich części kodu, które mogą być wykonywane więcej niż jeden raz, takich jak metody wewnętrzne.
W normalnych okolicznościach należy zdefiniować stałą wewnątrz samej klasy:
Jeśli z jakiegoś powodu naprawdę potrzebujesz zdefiniować stałą wewnątrz metody (być może dla jakiegoś typu metaprogramowania), możesz użyć
const_set
:Znowu jednak
const_set
nie jest czymś, do czego naprawdę powinieneś uciekać się w normalnych okolicznościach. Jeśli nie masz pewności, czy naprawdę chcesz przypisywać stałe w ten sposób, możesz rozważyć jedną z następujących alternatyw:Zmienne klasowe
Zmienne klasowe na wiele sposobów zachowują się jak stałe. Są właściwościami klasy i są dostępne w podklasach klasy, w której zostały zdefiniowane.
Różnica polega na tym, że zmienne klas mają być modyfikowalne i dlatego można je przypisać do metod wewnętrznych bez problemu.
Atrybuty klas
Atrybuty klasy to rodzaj „zmiennej instancji w klasie”. Zachowują się trochę jak zmienne klasowe, z tym wyjątkiem, że ich wartości nie są współdzielone z podklasami.
Zmienne instancji
I dla kompletności powinienem prawdopodobnie wspomnieć: jeśli chcesz przypisać wartość, która może być określona tylko po utworzeniu instancji twojej klasy, istnieje duża szansa, że faktycznie szukasz zwykłej starej zmiennej instancji.
źródło
W Rubim każda zmienna, której nazwa zaczyna się od dużej litery, jest stałą i możesz ją przypisać tylko raz. Wybierz jedną z tych alternatyw:
źródło
Stałe w ruby nie mogą być definiowane wewnątrz metod. Zobacz na przykład uwagi na dole tej strony
źródło
Nie możesz nazwać zmiennej wielkimi literami, inaczej Ruby założy, że jest to stała i będzie chciał, aby zachowała stałą wartość. W takim przypadku zmiana jej wartości byłaby błędem i „błędem dynamicznego przypisania stałej”. Z małymi literami powinno być dobrze
źródło
Rubiemu nie podoba się to, że przypisujesz stałą wewnątrz metody, ponieważ istnieje ryzyko ponownego przypisania. Kilka odpowiedzi SO przede mną daje alternatywę przypisywania jej poza metodą - ale w klasie, co jest lepszym miejscem do jej przypisania.
źródło
Wielkie podziękowania dla Doriana i Phrogza za przypomnienie mi o metodzie tablicowej (i haszującej) #replace, która może „zastąpić zawartość tablicy lub skrótu”.
Pogląd, że wartość STAŁA można zmienić, ale z irytującym ostrzeżeniem, jest jednym z niewielu błędnych kroków koncepcyjnych Rubiego - powinny one albo być w pełni niezmienne, albo całkowicie odrzucić stałą ideę. Z punktu widzenia programisty stała jest deklaratywna i zamierzona, jest sygnałem dla innych, że „ta wartość jest naprawdę niezmienna po zadeklarowaniu / przypisaniu”.
Czasami jednak „oczywista deklaracja” faktycznie wyklucza inne, przydatne w przyszłości możliwości. Na przykład...
Tam są przypadki legalnego wykorzystania gdzie wartość danej „Constant za” może naprawdę trzeba być zmieniony, na przykład: ponowne ładowanie ARGV z REPL podobny szybkiej pętli, a następnie ponowne uruchomienie ARGV thru więcej (późniejszej) OptionParser.parse! rozmowy - voila! Daje „argumentom wiersza poleceń” zupełnie nowe dynamiczne narzędzie.
Praktyczny problem jest albo z przypuszczalnego założeniu, że „ARGV musi być ciągle”, lub w oddzielnym metody initialize optparse, która twardych koduje przyporządkowanie ARGV do @default_argv przykład var do dalszej obróbki - że tablica (ARGV) rzeczywiście powinien być parametrem zachęcającym do ponownej analizy i ponownego wykorzystania, w stosownych przypadkach. Właściwa parametryzacja z odpowiednią wartością domyślną (powiedzmy ARGV) pozwoliłaby uniknąć konieczności zmiany „stałej” wartości ARGV. Zaledwie 2 grosze myśli ...
źródło