Skip to main content

differential equations - Using WhenEvent to Change the Sign of a Constant


I am attempting to change the sign of a constant when a certain condition is met during a numerical integration. Here is the code:


μk = 0.3; g = 9.81; R = 1; ti = 0; m = 0.3;

EOM = -μk (R θ'[t]^2 + g Sin[θ[t]]) +
g Cos[θ[t]] == R θ''[t];
IC1 = θ'[0] == 0; IC2 = θ[0] == 0;


sol = NDSolve[{EOM, IC1, IC2,
WhenEvent[θ'[t] == 0, {μk = -μk; tf = t;
"RestartIntegration"}]}, θ[t], {t, ti, ∞}];

θ = θ[t] /. sol;
θd = D[θ[t] /. sol, t];
Plot[{θ, θd}, {t, ti, tf}]

The problem lies in the fact that it is not dissipating as it should. θ should start to approach Pi/2. Instead it oscillates on to Infinity.



Note: I discovered that I cannot simply use Sign[θ'[t]] since the value is crossing zero. A zero value is undesirable in my case.


Slider System


Here is an image of the system being modeled.



Answer



The code has three issues. First, μk cannot be changed during the computation unless it is designated by DiscreteVariables. Second, the list of actions to be taken by WhenEvent needs to be separated by commas, not semicolons. Third, the upper limit of integration cannot be infinity. With these changes, the code becomes


g = 9.81; R = 1; ti = 0; m = 0.3;

EOM = -μk [t] (R θ'[t]^2 + g Sin[θ[t]]) + g Cos[θ[t]] == R θ''[t];
IC1 = θ'[0] == 0; IC2 = θ[0] == 0;


sol = NDSolve[{EOM, IC1, IC2, μk [0] == 0.3,
WhenEvent[θ'[t] == 0, {μk[t] -> -μk[t], tf = t, "RestartIntegration"}]},
{θ[t], θ'[t], μk[t]}, {t, ti, 10}, DiscreteVariables -> μk];

tmax = Flatten[θ[t] /. sol /. t -> "Domain"] // Last;
Plot[Evaluate[{θ[t], θ'[t], μk[t]} /. sol], {t, ti, tmax}, ImageSize -> Large]

enter image description here


Note that the computation terminates with a NDSolve::ndsz error. Basically, the computation has become unstable. Method -> "StiffnessSwitching" does not help. Incidentally, "RestartIntegration" has no substantive effect in this computation and can be omitted.


Improved Solution



Based on additional information provided by the OP in comments below and in the question itself, a better approach is


g = 981/100; R = 1; ti = 0; m = 3/10; μ = 3/10;

EOM = -μ Sign[θ'[t]] (R θ'[t]^2 + g Sin[θ[t]]) + g Cos[θ[t]] == R θ''[t];
IC1 = θ'[0] == 0; IC2 = θ[0] == 0;

sol = NDSolve[{EOM, IC1, IC2}, {θ[t], θ'[t]}, {t, ti, 10}, WorkingPrecision -> 30];

Plot[Evaluate[{θ[t], θ'[t], μ Sign[θ'[t]]} /. sol], {t, ti, tmax}, ImageSize -> Large]


enter image description here


The two solutions are identical until θ'[t] == 0 at t == 2.062520756856687159564857914. In the earlier computation, WhenEvent causes μk to reverse sign, even though θ'[t] does not, thereby introducing negative friction. I have tried numerous WhenEvent options, as well as WorkingPrecision as high as 60 to resolve this issue, but without success. In conclusion, only the second solution is credible physically.


Comments

Popular posts from this blog

functions - Get leading series expansion term?

Given a function f[x] , I would like to have a function leadingSeries that returns just the leading term in the series around x=0 . For example: leadingSeries[(1/x + 2)/(4 + 1/x^2 + x)] x and leadingSeries[(1/x + 2 + (1 - 1/x^3)/4)/(4 + x)] -(1/(16 x^3)) Is there such a function in Mathematica? Or maybe one can implement it efficiently? EDIT I finally went with the following implementation, based on Carl Woll 's answer: lds[ex_,x_]:=( (ex/.x->(x+O[x]^2))/.SeriesData[U_,Z_,L_List,Mi_,Ma_,De_]:>SeriesData[U,Z,{L[[1]]},Mi,Mi+1,De]//Quiet//Normal) The advantage is, that this one also properly works with functions whose leading term is a constant: lds[Exp[x],x] 1 Answer Update 1 Updated to eliminate SeriesData and to not return additional terms Perhaps you could use: leadingSeries[expr_, x_] := Normal[expr /. x->(x+O[x]^2) /. a_List :> Take[a, 1]] Then for your examples: leadingSeries[(1/x + 2)/(4 + 1/x^2 + x), x] leadingSeries[Exp[x], x] leadingSeries[(1/x + 2 + (1 - 1/x...

How to thread a list

I have data in format data = {{a1, a2}, {b1, b2}, {c1, c2}, {d1, d2}} Tableform: I want to thread it to : tdata = {{{a1, b1}, {a2, b2}}, {{a1, c1}, {a2, c2}}, {{a1, d1}, {a2, d2}}} Tableform: And I would like to do better then pseudofunction[n_] := Transpose[{data2[[1]], data2[[n]]}]; SetAttributes[pseudofunction, Listable]; Range[2, 4] // pseudofunction Here is my benchmark data, where data3 is normal sample of real data. data3 = Drop[ExcelWorkBook[[Column1 ;; Column4]], None, 1]; data2 = {a #, b #, c #, d #} & /@ Range[1, 10^5]; data = RandomReal[{0, 1}, {10^6, 4}]; Here is my benchmark code kptnw[list_] := Transpose[{Table[First@#, {Length@# - 1}], Rest@#}, {3, 1, 2}] &@list kptnw2[list_] := Transpose[{ConstantArray[First@#, Length@# - 1], Rest@#}, {3, 1, 2}] &@list OleksandrR[list_] := Flatten[Outer[List, List@First[list], Rest[list], 1], {{2}, {1, 4}}] paradox2[list_] := Partition[Riffle[list[[1]], #], 2] & /@ Drop[list, 1] RM[list_] := FoldList[Transpose[{First@li...

front end - keyboard shortcut to invoke Insert new matrix

I frequently need to type in some matrices, and the menu command Insert > Table/Matrix > New... allows matrices with lines drawn between columns and rows, which is very helpful. I would like to make a keyboard shortcut for it, but cannot find the relevant frontend token command (4209405) for it. Since the FullForm[] and InputForm[] of matrices with lines drawn between rows and columns is the same as those without lines, it's hard to do this via 3rd party system-wide text expanders (e.g. autohotkey or atext on mac). How does one assign a keyboard shortcut for the menu item Insert > Table/Matrix > New... , preferably using only mathematica? Thanks! Answer In the MenuSetup.tr (for linux located in the $InstallationDirectory/SystemFiles/FrontEnd/TextResources/X/ directory), I changed the line MenuItem["&New...", "CreateGridBoxDialog"] to read MenuItem["&New...", "CreateGridBoxDialog", MenuKey["m", Modifiers-...