material template
Here we show the build-in material template.
/**
* @class MaterialExample
* @brief A MaterialExample material class.
* @author tlc
* @date 06/07/2018
* @version 0.2.0
* @file MaterialExample.h
* @addtogroup Material-1D
* @ingroup Material
* @{
*/
#ifndef MATERIALEXAMPLE_H
#define MATERIALEXAMPLE_H
#include <Material/Material.h>
/**
* \brief It is recommended to store data, especially constant data, in a simple
* structure. The motivation is to obtain a clear interface so that store and recover
* of objects will be easier.
*/
struct MaterialExampleData {
const double elastic_modulus; // elastic modulus
const double yield_stress; // initial yield stress
const double hardening_ratio; // hardening ratio
const double beta; // isotropic/kinematic hardening factor
const double plastic_modulus; // plastic modulus
};
class MaterialExample final : MaterialExampleData, public Material {
double current_back_stress = 0.;
double current_plastic_strain = 0.;
double trial_back_stress = 0.;
double trial_plastic_strain = 0.;
public:
explicit MaterialExample(unsigned = 0, // tag
double = 2E5, // elastic modulus
double = 400., // initial yield stress
double = .05, // hardening ratio
double = 0., // isotropic/kinematic hardening factor
double = 0. // density
);
void initialize(const shared_ptr<DomainBase>&) override;
unique_ptr<Material> get_copy() override;
int update_trial_status(const vec&) override;
int clear_status() override;
int commit_status() override;
int reset_status() override;
void print() override;
};
#endif
#include "MaterialExample.h"
/**
* \brief The constructor of material model depends on the specific model to be
* implemented. But for the base `Material` class, it takes only three input arguments:
*
* - Unique Material Tag
* - Material Type
* - Density
*
* Here we are implementing a uniaxial bilinear hardening model, hence `MaterialType::D1`
* is passed to the base. The material type will be used to validate with associated
* elements/sections to ensure the consistency of the sizes of data passed between objects.
*
* \param T Unique Material Tag
* \param E Elastic Modulus
* \param Y Yield Stress
* \param H Hardening Ratio
* \param B Beta
* \param R Density
*/
MaterialExample::MaterialExample(const unsigned T, const double E, const double Y, const double H, const double B, const double R)
: MaterialExampleData{E, Y, H, B, E * H / (1. - H)}
, Material(T, MaterialType::D1, R) {}
/**
* \brief Unless the target material model involves other material models to compute
* responses, in general, it is not necessary to get additional information from
* other parts of the model.
*
* In general cases, history variables shall be initialised and initial stiffness
* (and initial damping if appropriate) shall be set to proper value.
*
* To enable initial values for history variables, build-in `trial_history` and
* `current_history` shall be used, developers can initialise them via method
* `initialise_history()` so that initial values set by `initial` command will not be
* overwritten.
*
* In this example, instead of using build-in history variables, we manage history
* variables, namely back stress and plastic strain, by ourselves.
*
*/
void MaterialExample::initialize(const shared_ptr<DomainBase>&) {
current_back_stress = trial_back_stress = 0.;
current_plastic_strain = trial_plastic_strain = 0.;
trial_stiffness = current_stiffness = initial_stiffness = elastic_modulus;
}
/**
* \brief The `get_copy()` method should always be implemented with `make_unique`.
* In case the model defines other memory management, developers may need to further
* provide a copy ctor to make `make_unique` work.
*
* **!!!NEVER DO A MANUAL COPY OF DATA IN THIS METHOD!!!**
*
* \return a copy of material model
*/
unique_ptr<Material> MaterialExample::get_copy() { return make_unique<MaterialExample>(*this); }
/**
* \brief There are two states we are managing at any time point.
* The current state is the converged state from the last time substep. Since it is
* converged, all data shall be valid and accurate.
* The trial state stores the response computed based on converged state and new trial
* strain. It may be discarded, committed or overwritten with new trial values.
*
* **WE ALWAYS COMPUTE TRIAL STATE BASED ON CURRENT STATE AND NEW TRIAL STRAIN**
* **NEVER COMPUTE RESPONSE BASED ON ANY INFORMATION FROM UNCONVERGED STATE**
*
* Developers who are not familiar with classic plasticity theory may consult textbooks
* for details.
*
* \param t_strain trial strain
* \return error flag
*/
int MaterialExample::update_trial_status(const vec& t_strain) {
trial_strain = t_strain;
incre_strain = trial_strain - current_strain;
if(fabs(incre_strain(0)) <= tolerance) return 0;
trial_back_stress = current_back_stress;
trial_plastic_strain = current_plastic_strain;
trial_stiffness = initial_stiffness;
trial_stress = current_stress + elastic_modulus * incre_strain;
const auto shifted_stress = trial_stress(0) - current_back_stress;
const auto yield_func = fabs(shifted_stress) - yield_stress - (1. - beta) * plastic_modulus * current_plastic_strain;
if(yield_func > 0.) {
const auto incre_plastic_strain = yield_func / (elastic_modulus + plastic_modulus);
trial_stress -= suanpan::sign(shifted_stress) * elastic_modulus * incre_plastic_strain;
trial_stiffness *= hardening_ratio;
trial_back_stress += suanpan::sign(shifted_stress) * beta * plastic_modulus * incre_plastic_strain;
trial_plastic_strain += incre_plastic_strain;
}
return 0;
}
/**
* \brief Operations are required to achieve the following objective.
*
* current state -> 0
* trial state -> 0
*
* \return error flag
*/
int MaterialExample::clear_status() {
current_strain.zeros();
current_stress.zeros();
current_stiffness = initial_stiffness;
current_back_stress = 0.;
current_plastic_strain = 0.;
return reset_status();
}
/**
* \brief Operations are required to achieve the following objective.
*
* current state <- trial state
*
* \return error flag
*/
int MaterialExample::commit_status() {
current_strain = trial_strain;
current_stress = trial_stress;
current_stiffness = trial_stiffness;
current_back_stress = trial_back_stress;
current_plastic_strain = trial_plastic_strain;
return 0;
}
/**
* \brief Operations are required to achieve the following objective.
*
* current state -> trial state
*
* \return error flag
*/
int MaterialExample::reset_status() {
trial_strain = current_strain;
trial_stress = current_stress;
trial_stiffness = current_stiffness;
trial_back_stress = current_back_stress;
trial_plastic_strain = current_plastic_strain;
return 0;
}
void MaterialExample::print() {
suanpan_info("A material example based on uniaxial J2 bilinear mixed hardening model.\n");
suanpan_info("current strain: %.5E\tcurrent stress: %.5E.\n", current_strain.at(0), current_stress.at(0));
}
Last updated