Source code for MDAnalysis.analysis.polymer

# -*- Mode: python; tab-width: 4; indent-tabs-mode:nil; coding:utf-8 -*-
# vim: tabstop=4 expandtab shiftwidth=4 softtabstop=4
#
# MDAnalysis --- https://www.mdanalysis.org
# Copyright (c) 2006-2017 The MDAnalysis Development Team and contributors
# (see the file AUTHORS for the full list of names)
#
# Released under the Lesser GNU Public Licence, v2.1 or any higher version
#
# Please cite your use of MDAnalysis in published work:
#
# R. J. Gowers, M. Linke, J. Barnoud, T. J. E. Reddy, M. N. Melo, S. L. Seyler,
# D. L. Dotson, J. Domanski, S. Buchoux, I. M. Kenney, and O. Beckstein.
# MDAnalysis: A Python package for the rapid analysis of molecular dynamics
# simulations. In S. Benthall and S. Rostrup editors, Proceedings of the 15th
# Python in Science Conference, pages 102-109, Austin, TX, 2016. SciPy.
# doi: 10.25080/majora-629e541a-00e
#
# N. Michaud-Agrawal, E. J. Denning, T. B. Woolf, and O. Beckstein.
# MDAnalysis: A Toolkit for the Analysis of Molecular Dynamics Simulations.
# J. Comput. Chem. 32 (2011), 2319--2327, doi:10.1002/jcc.21787
#

"""
Polymer analysis --- :mod:`MDAnalysis.analysis.polymer`
=======================================================


:Author: Richard J. Gowers
:Year: 2015, 2018
:Copyright: Lesser GNU Public License v2.1+

This module contains various commonly used tools in analysing polymers.
"""
import numpy as np
import scipy.optimize
import warnings
import logging

from .. import NoDataError
from ..core.groups import requires, AtomGroup
from ..lib.distances import calc_bonds
from .base import AnalysisBase

logger = logging.getLogger(__name__)


[docs] @requires('bonds') def sort_backbone(backbone): """Rearrange a linear AtomGroup into backbone order Requires that the backbone has bond information, and that only backbone atoms are provided (ie no side chains or hydrogens). Parameters ---------- backbone : AtomGroup the backbone atoms, not necessarily in order Returns ------- sorted_backbone : AtomGroup backbone in order, so `sorted_backbone[i]` is bonded to `sorted_backbone[i - 1]` and `sorted_backbone[i + 1]` .. versionadded:: 0.20.0 """ if not backbone.n_fragments == 1: raise ValueError("{} fragments found in backbone. " "backbone must be a single contiguous AtomGroup" "".format(backbone.n_fragments)) branches = [at for at in backbone if len(at.bonded_atoms & backbone) > 2] if branches: # find which atom has too many bonds for easier debug raise ValueError( "Backbone is not linear. " "The following atoms have more than two bonds in backbone: {}." "".format(','.join(str(a) for a in branches))) caps = [atom for atom in backbone if len(atom.bonded_atoms & backbone) == 1] if not caps: # cyclical structure raise ValueError("Could not find starting point of backbone, " "is the backbone cyclical?") # arbitrarily choose one of the capping atoms to be the startpoint sorted_backbone = AtomGroup([caps[0]]) # iterate until the sorted chain length matches the backbone size while len(sorted_backbone) < len(backbone): # current end of the chain end_atom = sorted_backbone[-1] # look at all bonded atoms which are also part of the backbone # and subtract any that have already been added next_atom = (end_atom.bonded_atoms & backbone) - sorted_backbone # append this to the sorted backbone sorted_backbone += next_atom return sorted_backbone
[docs] class PersistenceLength(AnalysisBase): r"""Calculate the persistence length for polymer chains The persistence length is the length at which two points on the polymer chain become decorrelated. This is determined by first measuring the autocorrelation (:math:`C(n)`) of two bond vectors (:math:`\mathbf{a}_i, \mathbf{a}_{i + n}`) separated by :math:`n` bonds .. math:: C(n) = \langle \cos\theta_{i, i+n} \rangle = \langle \mathbf{a_i} \cdot \mathbf{a_{i+n}} \rangle An exponential decay is then fitted to this, which yields the persistence length .. math:: C(n) \approx \exp\left( - \frac{n \bar{l_B}}{l_P} \right) where :math:`\bar{l_B}` is the average bond length, and :math:`l_P` is the persistence length which is fitted Parameters ---------- atomgroups : iterable List of AtomGroups. Each should represent a single polymer chain, ordered in the correct order. verbose : bool, optional Show detailed progress of the calculation if set to ``True``. Attributes ---------- results.bond_autocorrelation : numpy.ndarray the measured bond autocorrelation results.lb : float the average bond length .. versionadded:: 2.0.0 lb : float Alias to the :attr:`results.lb`. .. deprecated:: 2.0.0 Will be removed in MDAnalysis 3.0.0. Please use :attr:`results.lb` instead. results.x : numpy.ndarray length of the decorrelation predicted by *lp* .. versionadded:: 2.0.0 results.lp : float calculated persistence length .. versionadded:: 2.0.0 lp : float Alias to the :attr:`results.lp`. .. deprecated:: 2.0.0 Will be removed in MDAnalysis 3.0.0. Please use :attr:`results.lp` instead. results.fit : numpy.ndarray the modelled backbone decorrelation predicted by *lp* .. versionadded:: 2.0.0 fit : float Alias to the :attr:`results.fit`. .. deprecated:: 2.0.0 Will be removed in MDAnalysis 3.0.0. Please use :attr:`results.fit` instead. See Also -------- :func:`sort_backbone` for producing the sorted AtomGroup required for input. Example ------- .. code-block:: python from MDAnalysis.tests.datafiles import TRZ_psf, TRZ import MDAnalysis as mda from MDAnalysis.analysis import polymer u = mda.Universe(TRZ_psf, TRZ) # this system is a pure polymer melt of polyamide, # so we can select the chains by using the .fragments attribute chains = u.atoms.fragments # select only the backbone atoms for each chain backbones = [chain.select_atoms('not name O* H*') for chain in chains] # sort the chains, removing any non-backbone atoms sorted_backbones = [polymer.sort_backbone(bb) for bb in backbones] persistence_length = polymer.PersistenceLength(sorted_backbones) # Run the analysis, this will average over all polymer chains # and all timesteps in trajectory persistence_length = persistence_length.run() print(f'The persistence length is: {persistence_length.results.lp}') # always check the visualisation of this: persistence_length.plot() .. versionadded:: 0.13.0 .. versionchanged:: 0.20.0 The run method now automatically performs the exponential fit .. versionchanged:: 1.0.0 Deprecated :meth:`PersistenceLength.perform_fit` has now been removed. .. versionchanged:: 2.0.0 Former ``results`` are now stored as ``results.bond_autocorrelation``. :attr:`lb`, :attr:`lp`, :attr:`fit` are now stored in a :class:`MDAnalysis.analysis.base.Results` instance. """ def __init__(self, atomgroups, **kwargs): super(PersistenceLength, self).__init__( atomgroups[0].universe.trajectory, **kwargs) self._atomgroups = atomgroups # Check that all chains are the same length lens = [len(ag) for ag in atomgroups] chainlength = len(atomgroups[0]) if not all(l == chainlength for l in lens): raise ValueError("Not all AtomGroups were the same size") self._results = np.zeros(chainlength - 1, dtype=np.float32) def _single_frame(self): # could optimise this by writing a "self dot array" # we're only using the upper triangle of np.inner # function would accept a bunch of coordinates and spit out the # decorrel for that n = len(self._atomgroups[0]) for chain in self._atomgroups: # Vector from each atom to next vecs = chain.positions[1:] - chain.positions[:-1] # Normalized to unit vectors vecs /= np.sqrt((vecs * vecs).sum(axis=1))[:, None] inner_pr = np.inner(vecs, vecs) for i in range(n-1): self._results[:(n-1)-i] += inner_pr[i, i:] @property def lb(self): wmsg = ("The `lb` attribute was deprecated in " "MDAnalysis 2.0.0 and will be removed in MDAnalysis 3.0.0. " "Please use `results.variance` instead.") warnings.warn(wmsg, DeprecationWarning) return self.results.lb @property def lp(self): wmsg = ("The `lp` attribute was deprecated in " "MDAnalysis 2.0.0 and will be removed in MDAnalysis 3.0.0. " "Please use `results.variance` instead.") warnings.warn(wmsg, DeprecationWarning) return self.results.lp @property def fit(self): wmsg = ("The `fit` attribute was deprecated in " "MDAnalysis 2.0.0 and will be removed in MDAnalysis 3.0.0. " "Please use `results.variance` instead.") warnings.warn(wmsg, DeprecationWarning) return self.results.fit def _conclude(self): n = len(self._atomgroups[0]) norm = np.linspace(n - 1, 1, n - 1) norm *= len(self._atomgroups) * self.n_frames self.results.bond_autocorrelation = self._results / norm self._calc_bond_length() self._perform_fit() def _calc_bond_length(self): """calculate average bond length""" bs = [] for ag in self._atomgroups: pos = ag.positions b = calc_bonds(pos[:-1], pos[1:]).mean() bs.append(b) self.results.lb = np.mean(bs) def _perform_fit(self): """Fit the results to an exponential decay""" try: self.results.bond_autocorrelation except AttributeError: raise NoDataError("Use the run method first") from None self.results.x = self.results.lb *\ np.arange(len(self.results.bond_autocorrelation)) self.results.lp = fit_exponential_decay(self.results.x, self.results.bond_autocorrelation) self.results.fit = np.exp(-self.results.x/self.results.lp)
[docs] def plot(self, ax=None): """Visualize the results and fit Parameters ---------- ax : matplotlib.Axes, optional if provided, the graph is plotted on this axis Returns ------- ax : the axis that the graph was plotted on """ import matplotlib.pyplot as plt if ax is None: fig, ax = plt.subplots() ax.plot(self.results.x, self.results.bond_autocorrelation, 'ro', label='Result') ax.plot(self.results.x, self.results.fit, label='Fit') ax.set_xlabel(r'x') ax.set_ylabel(r'$C(x)$') ax.set_xlim(0.0, 40 * self.results.lb) ax.legend(loc='best') return ax
[docs] def fit_exponential_decay(x, y): r"""Fit a function to an exponential decay .. math:: y = \exp\left(- \frac{x}{a}\right) Parameters ---------- x, y : array_like The two arrays of data Returns ------- a : float The coefficient *a* for this decay Notes ----- This function assumes that data starts at 1.0 and decays to 0.0 """ def expfunc(x, a): return np.exp(-x/a) a = scipy.optimize.curve_fit(expfunc, x, y)[0][0] return a