Switching#

flatspin uses a generalized Stoner-Wohlfarth (GSW) switching model to describe the angle-dependent switching threshold (coercive field) of a spin.

A switching astroid is a polar plot of the coercive field at different angles, with \(h_{\perp}\) on the horizontal axis (hard axis) and \(h_{\parallel}\) on the vertical axis (easy axis) (see Magnetic fields).

The coercive field is described by the GSW equation (see Theory):

\[ \left(\frac{h_{\parallel}}{b h_k}\right)^{2/\gamma} + \left(\frac{h_{\perp}}{c h_k}\right)^{2/\beta} = 1 \]

The parameters \(b\), \(c\), \(\beta\) and \(\gamma\) adjust the shape of the switching astroid: \(b\) and \(c\) define the height and width, respectively, while \(\beta\) and \(\gamma\) adjust the curvature of the astroid at the easy and hard axis, respectively. \(h_k\) scales the coercive fields and corresponds to the switching threshold at \(h_\parallel=0\), i.e., when the field is aligned with the hard axis. In the flatspin model, \(h_k\) corresponds to the hc parameter.

Tuning the parameters of the GSW equation allows the model to capture the switching characteristics of magnets with different shapes. For example:

  • Elliptical magnets have a symmetric astroid, described by the regular Stoner-Wohlfarth model: \(b=c=1\) and \(\beta=\gamma=3\).

  • Rectangular magnets have an asymmetric switching astroid: they switch more easily along the parallel axis: \(b < c\).

Switching astroids#

The flatspin.astroid module contains various astroid-related methods. In particular, it provides a database of switching parameters for different magnet shapes (see Astroid database for a full listing). Switching parameters from the database can be obtained with astroid_params():

from flatspin.astroid import astroid_params

sw_params = astroid_params(shape="stadium", width=220, height=80, thickness=20)
print(sw_params)
{'hc': 0.1898, 'sw_b': 0.366, 'sw_c': 1.0, 'sw_beta': 1.68, 'sw_gamma': 2.75}

Switching parameters may be conveniently obtained at model creaton with the astroid_params parameter, which takes a string of the form <shape><width>x<height>x<thickness>:

from flatspin.model import IsingSpinIce

model = IsingSpinIce(astroid_params="stadium220x80x20")
print(model.hc, model.sw_params)
0.1898 (0.366, 1.0, 1.68, 2.75)

The plot below shows switching astroids for a few different magnet shapes and sizes, as obtained from the database.

Hide code cell source
from flatspin.plotting import plot_astroid

def plot_astroid_db(shape, width, height, thickness):
    sw_params = astroid_params(shape=shape, width=width, height=height, thickness=thickness)
    hc = sw_params['hc']
    b = sw_params['sw_b']
    c = sw_params['sw_c']
    beta = sw_params['sw_beta']
    gamma = sw_params['sw_gamma']

    print(f'{shape} {width}x{height}x{thickness}:',
          f'b={b:.2f}, c={c:.2f}, beta={beta:.2f}, gamma={gamma:.2f}')

    return plot_astroid(b, c, beta, gamma, hc,
                        label=f'{shape} {width}x{height}x{thickness}',
                        rotation=90)

plot_astroid_db('ellipse', 220, 80, 20)
plot_astroid_db('stadium', 220, 80, 20)
plot_astroid_db('stadium', 470, 170, 20)
plt.xlabel(r'$h_\perp / h_k$')
plt.ylabel(r'$h_\parallel / h_k$')
plt.legend(loc='upper left', bbox_to_anchor=(1.0, 1.0));
ellipse 220x80x20: b=1.02, c=1.00, beta=2.46, gamma=2.53
stadium 220x80x20: b=0.37, c=1.00, beta=1.68, gamma=2.75
stadium 470x170x20: b=0.12, c=1.00, beta=3.39, gamma=1.61
_images/3ff2e8f37a7433d33d7c2a9295c822396bdc57d507f54eb890562b4784fd7559.svg

Switching a spin#

A spin may flip if the total magnetic field acting on the spin:

  1. is outside of the switching astroid (left hand side of the GSW equation is greater than 1)

  2. is oriented in the opposite direction of the spin magnetization (\(h_\parallel < 0\))

To illustrate the switching model, let us consider a single vertical spin:

from flatspin.model import IsingSpinIce

model = IsingSpinIce(size=(1,1))
model.plot();
_images/01163f3d3ad9527bc9803bd1c1e4e24e8d2b28d06e35ff8624006bde04a3743b.svg

Next, let us set up a cycling external field at some angle to the spin. The animation below shows the switching astroid for the spin (left plot) with the corresponding magnetic field superimposed (arrow inside the astroid). The top right column shows the state of the spin, and the bottom right column shows the external field.

Notice how switching only occurs when the field crosses the negative side of the astroid (\(h_\parallel < 0\)), because of switching condition (2). After switching, the external field is aligned with the magnetization of the spin, and the field arrow jumps to the positive side of the astroid ($h_\parallel > 0).

Because only the negative side of the astroid is relevant for switching, the positive side of the astroid is marked with a dashed line in the astroid plots below.

Hide code cell source
from matplotlib.animation import FuncAnimation
from IPython.display import HTML
from flatspin.encoder import Triangle
from flatspin.plotting import plot_vectors, vector_angle

def animate_switching(model, H=0.15, phi=-100):
    # Reset system back to the polarized state
    model.polarize()
    hk = model.threshold.reshape((-1,1))

    # Set up figure and axes
    fig = plt.figure(facecolor='white')
    ax_astroid = plt.subplot2grid((2,3), (0,0), rowspan=2, colspan=2)
    ax_spin = plt.subplot2grid((2,3), (0,2))
    ax_h_ext = plt.subplot2grid((2,3), (1,2))

    # Set up external field (triangle wave)
    enc = Triangle(timesteps=64, H=H, phi=phi)
    h_ext = enc([1])

    # Plot astroid with field vector
    plt.sca(ax_astroid)
    line, = model.plot_astroid(angle_range=(-np.pi/2, np.pi/2), ls='dashed', rotation=90)
    model.plot_astroid(angle_range=(np.pi/2, 3*np.pi/2), rotation=90, color=line.get_color())
    origin = np.tile([0, 0], (model.spin_count, 1))
    plot_vectors(origin, origin, C=origin[:,0],
                 clim=(-.5,.5), cmap='bwr_r', scale=1, width=.05, pivot='tail', mask_zero=False)
    plt.xlabel(r'$h_\perp / h_k$')
    plt.ylabel(r'$h_\parallel / h_k$')

    # spin axis
    plt.sca(ax_spin)
    plt.axis('off')
    plt.title('spin')

    # h_ext axis
    plt.sca(ax_h_ext)
    plt.axis('off')
    plt.title('h_ext')

    def do_cycle():
        for h in h_ext:
            model.set_h_ext(h)
            model.relax()
            h_tot = model.total_fields()
            yield model.total_fields()

    def do_plot(h_tot):
        h_tot /= hk
        h_tot = np.column_stack([h_tot[:,1], h_tot[:,0]])
        plot_vectors(origin, h_tot, C=np.sign(h_tot[:,1]), ax=ax_astroid, replace=True)

        model.plot(ax=ax_spin, replace=True)

        h_ext = model.h_ext / hk
        plot_vectors(model.pos, h_ext, ax=ax_h_ext, replace=True, scale=.5, width=.1)

    anim = FuncAnimation(fig, do_plot, init_func=lambda: None, frames=do_cycle(),
                         interval=200, blit=False, cache_frame_data=False)
    plt.close() # Only show the animation
    #anim.save("astroid.gif")
    return HTML(anim.to_jshtml())
animate_switching(model)

We may change the astroid shape of the spins in the model by setting the parameters sw_b, sw_c, sw_beta and sw_gamma. Below we change switching parameters to describe an elliptical magnet, and repeat the switching animation. Notice how elliptical magnets are much harder to switch when the field is aligned to the easy axis.

model = IsingSpinIce(size=(1,1), sw_b=1, sw_c=1, sw_beta=3, sw_gamma=3)
animate_switching(model)

Switching many spins#

Finally, we illustrate the switching process for a system of coupled spins in a square spin ice. Inside the astroid, there is one arrow for each spin. There are two main groups of arrows, since spins have two orientations (horizontal and vertical). Notice how, within a group, the arrows do not overlap perfectly. This is because of the dipolar fields from neighboring magnets.

from flatspin.model import SquareSpinIceClosed
model = SquareSpinIceClosed()
animate_switching(model)

The switching process of many spin systems is discussed further in Dynamics.