All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.xmlcml.cml.tools.DisorderTool Maven / Gradle / Ivy

/**
 *    Copyright 2011 Peter Murray-Rust et. al.
 *
 *    Licensed under the Apache License, Version 2.0 (the "License");
 *    you may not use this file except in compliance with the License.
 *    You may obtain a copy of the License at
 *
 *        http://www.apache.org/licenses/LICENSE-2.0
 *
 *    Unless required by applicable law or agreed to in writing, software
 *    distributed under the License is distributed on an "AS IS" BASIS,
 *    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 *    See the License for the specific language governing permissions and
 *    limitations under the License.
 */

package org.xmlcml.cml.tools;

import java.math.BigInteger;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;

import nu.xom.Node;

import org.apache.log4j.Logger;
import org.xmlcml.cml.base.AbstractTool;
import org.xmlcml.cml.base.CMLAttribute;
import org.xmlcml.cml.base.CMLConstants;
import org.xmlcml.cml.base.CMLElement;
import org.xmlcml.cml.base.CMLUtil;
import org.xmlcml.cml.element.CMLAtom;
import org.xmlcml.cml.element.CMLMolecule;
import org.xmlcml.cml.element.CMLScalar;
import org.xmlcml.cml.tools.DisorderToolControls.ProcessControl;
import org.xmlcml.cml.tools.DisorderToolControls.RemoveControl;
import org.xmlcml.euclid.Util;
import org.xmlcml.molutil.ChemicalElement.AS;

/**
 *
 * @author pm286
 *
 */
public class DisorderTool extends AbstractTool {
	private static final String UNRESOLVED_DISORDER = "unresolvedDisorder";
	private static final String RESOLVED_DISORDER = "resolvedDisorder";

	@SuppressWarnings("unused")
	private static Logger LOG = Logger.getLogger(DisorderTool.class);

	private CMLMolecule molecule;
	private boolean disorderProcessed;
	private boolean containsDisorder;

	private DisorderToolControls disorderOptions;
    /** dewisott */
	public static final String RESOLVED_DISORDER_DICTREF = "cif:"+RESOLVED_DISORDER;
    /** dewisott */
	public static final String UNRESOLVED_DISORDER_DICTREF = "cif:"+UNRESOLVED_DISORDER;
	private String disorderString;

	/**
	 * constructor
	 *
	 * @param molecule
	 */
//	private DisorderTool() {
//		;
//	}

	/** constructor.
	 *
	 * @param molecule
	 */
	public DisorderTool(CMLMolecule molecule) {
		this(molecule, new DisorderToolControls());
	}

	/** constructor.
	 *
	 * @param mol
	 * @param disorderMan
	 */
	public DisorderTool(CMLMolecule mol, DisorderToolControls disorderMan) {
		if (mol == null) {
			throw new IllegalArgumentException("Molecule cannot be null");
		}
		if (disorderMan == null) {
			throw new IllegalArgumentException(
			"Disorder manager cannot be null");
		}
		this.molecule = mol;
		this.disorderOptions = disorderMan;
	}

	/**
	 * gets the cml molecule.
	 *
	 * @return molecule
	 */
	public CMLMolecule getMolecule() {
		return molecule;
	}
	
	/**
	 * 
	 * @return any dictRef with 'unresolvedDisorder'
	 */
	public boolean isDisordered() {
		return molecule.query(
				".//*[local-name()='scalar' and contains(@dictRef, '"+UNRESOLVED_DISORDER+"')]").size() > 0;
	}


	/**
	 * Resolves crystallographic disorder.
	 *
	 * At present the only RemoveDisorderControl is to remove minor disorder.
	 * These are components with occupancies <= 0.5.
	 *
	 * Can provide two different kinds of ProcessDisorderControl: 1. STRICT -
	 * this tries to process the disorder as set out in the CIF specification
	 * (http://www.iucr.org/iucr-top/cif/#spec) If it comes across disorder
	 * which does not comply with the spec then throws an error.
	 *
	 * 2. LOOSE - this tries to process the disorder as with STRICT, but does
	 * not throw an error if it comes across disorder not valid with respect to
	 * the CIF spec. Instead it tags the provided molecule with metadata stating
	 * that the disorder cannot currently be processed and does nothing more.
	 *
	 * NOTE: if the disorder is successfully processed then the molecule is
	 * tagged with metadata stating that the molecule did contain disorder, but
	 * has now been removed.
	 *
	 * @exception RuntimeException
	 */
	public void resolveDisorder() {
		if (molecule == null) {
			throw new IllegalStateException(
			"Molecule property is null, no operations possible");
		}
		List disorderAssemblyList = checkDisorder();
		processDisorder(disorderAssemblyList);
	}

	/**
	 *
	 * @param disorderAssemblyList
	 */
	private void processDisorder(List disorderAssemblyList) {
		if (disorderProcessed) {
			if (RemoveControl.REMOVE_MINOR_DISORDER.equals(disorderOptions
					.getRemoveControl())) {
				for (DisorderAssembly assembly : disorderAssemblyList) {
					assembly.removeMinorDisorder();
				}
			}
		}
		// remove atom disorder information after processing IF disorder
		// has been successfully resolved.
		if (disorderProcessed || !containsDisorder) {
			removeAtomDisorderInformation();
		}
	}

	/**
	 * removes all disorder elements and attributes from each disordered atom.
	 *
	 */
	private void removeAtomDisorderInformation() {
		List nodes = CMLUtil.getQueryNodes(molecule, ".//"+CMLAtom.NS+"/@occupancy | "
				+".//"+CMLAtom.NS+"/"+CMLScalar.NS+"[contains(@dictRef, '"+CrystalTool.DISORDER_ASSEMBLY+"') or " +
				"contains(@dictRef, '"+CrystalTool.DISORDER_GROUP+"')" +
				"]", CMLConstants.CML_XPATH);
		for (Node node : nodes) {
			node.detach();
		}
	}

	/**
	 * checks that the disorder complies with the CIF specification.
	 * http://www.iucr.org/iucr-top/cif/#spec
	 *
	 * called only from processDisorder().
	 *
	 * @param disorderAssemblyList
	 * @param pControl
	 * @return list of disorder assemblies in the given molecule
	 */
	private List checkDisorder() {
		List disorderAssemblyList = getDisorderAssemblyList();
		List failedAssemblyList = new ArrayList();
		List finishedAssemblyList = new ArrayList();
		if (disorderAssemblyList.size() > 0) {
			containsDisorder = true;
			ProcessControl pControl = disorderOptions.getProcessControl();
			boolean isMetadataSet = false;
			for (DisorderAssembly da : disorderAssemblyList) {
				List disorderGroupList = da.getDisorderGroupList();
				if (disorderGroupList.size() < 2) {
					if (ProcessControl.LOOSE.equals(pControl)) {
						//System.err.println("Group list less than 2");
						failedAssemblyList.add(da);
						continue;
					} else if (ProcessControl.STRICT.equals(pControl)) {
						throw new RuntimeException(
								"Disorder assembly should contain at least 2 disorder groups: "
								+ da.toString());
					}
				}
				List commonAtoms = da.getCommonAtoms();
				for (CMLAtom commonAtom : commonAtoms) {
					if (!CrystalTool.hasUnitOccupancy(commonAtom)) {
						if (ProcessControl.LOOSE.equals(pControl)) {
							//System.err.println("Common atom doesn't have unit occupancy.");
							failedAssemblyList.add(da);
							continue;
						} else if (ProcessControl.STRICT.equals(pControl)) {
							throw new RuntimeException(
									"Common atoms require unit occupancy: "
									+ commonAtom.getId()
									+ ", in disorder assembly, "
									+ da.getAssemblyCode());
						}
					}
				}
				for (DisorderGroup dg : disorderGroupList) {
					List atomList = dg.getAtomList();
					for (CMLAtom atom : atomList) {
						if (CrystalTool.hasUnitOccupancy(atom)) {
							if (ProcessControl.LOOSE.equals(pControl)) {
								//System.err.println("Atom in disorder group has unit occupancy.");
								failedAssemblyList.add(da);
								continue;
							} else if (ProcessControl.STRICT.equals(pControl)) {
								throw new RuntimeException("Atom, "
										+ atom.getId() + ", in disorder group, "
										+ dg.getGroupCode()
										+ ", has unit occupancy");
							}
						}
						if (Math.abs(atom.getOccupancy() - dg.getOccupancy()) > CrystalTool.OCCUPANCY_EPS) {
							if (ProcessControl.LOOSE.equals(pControl)) {
								//System.err.println("Inconsistent occupancy.");
								failedAssemblyList.add(da);
								continue;
							} else if (ProcessControl.STRICT.equals(pControl)) {
								throw new RuntimeException(
										"Atom "
										+ atom.getId()
										+ ", in disorder group, "
										+ dg.getGroupCode()
										+ ", has inconsistent occupancy ("
										+ atom.getOccupancy()
										+ ") with that of the disorder group ("
										+ dg.getOccupancy()+").");
							}
						}
					}
				}
				finishedAssemblyList.add(da);
			}			
			// attempt to reconcile errors in disorder assemblies
			boolean fixed = false;
			if (failedAssemblyList.size() > 0) {
				try {
					fixed = fixFailedAssemblies(failedAssemblyList);
				} catch (RuntimeException e) {
					disorderString = e.getMessage();
					fixed = false;
				}
				if (fixed) {
					// if process reaches this point then failed assemblies have been
					// fixed - add them to finished list
					DisorderTool newDt = new DisorderTool(molecule);
					List newDA = newDt.getDisorderAssemblyList();

					// if the process reaches this point without an error being thrown then
					// the disorder can be processed. Add metadata to say so!
					addDisorderMetadata(true);
					isMetadataSet = true;
					return newDA;
				} else {
					if (!isMetadataSet) {
						addDisorderMetadata(false);
						isMetadataSet = true;
					}
					throw new RuntimeException ("Could not resolve invalid disorder: "+disorderString);
				}
			} else {
				if (containsDisorder) {
					// disorder has already been processed
					addDisorderMetadata(true);
					isMetadataSet = true;
				}
			}
		}
		return finishedAssemblyList;
	}

	/**
	 * takes a list of disorder assemblies that are invalid with respect to the CIF
	 * specification and tries to create a set of assemblies that are valid so that
	 * they can be parsed.
	 *
	 * @param failedAssemblyList
	 * @return whether a set of valid assemblies has been found.
	 */
	@SuppressWarnings("all")
	private boolean fixFailedAssemblies(List failedAssemblyList) {
		// aggregate all disordered atoms from the failed assemblies
		List disorderedAtoms = new ArrayList();
		for (DisorderAssembly failedAssembly : failedAssemblyList) {
			disorderedAtoms.addAll(failedAssembly.getCommonAtoms());
			for (DisorderGroup group : failedAssembly.getDisorderGroupList()) {
				disorderedAtoms.addAll(group.getAtomList());
			}
		}
		// attempt to resolve failed assemblies, by first splitting the atoms into
		// lists of similar occupancy.  Then test to see if there are 2 or more that
		// add up to unity (fail if not so or if any lists left over)
		List unityAtoms = new ArrayList();
		Map> atomMap = new LinkedHashMap>();
		for (CMLAtom disorderedAtom : disorderedAtoms) {
			boolean added = false;
			double occupancy = disorderedAtom.getOccupancy();
			// test to see if any of the lists has occupancy of 0.5.  If so then throw
			// exception as we cannot be confident of getting the proper structure if there
			// are more than one assemblies with this occupancy
			if (occupancy == 0.5) {
				throw new RuntimeException("Cannot fix invalid disorder - found atom"
						+" with occupancy of 0.5");
			}
			if (occupancy == Double.NaN) {
				throw new RuntimeException("Atom "+disorderedAtom+" has no occupancy set.  Cannot resolve disorder.");
			}
			//LOG.debug("occupancy is: "+occupancy);
			// if atom has unit occupancy then we can ignore it for the rest of
			// the method
			if (Math.abs(1.0 - occupancy) < CrystalTool.OCCUPANCY_EPS) {
				unityAtoms.add(disorderedAtom);
				continue;
			}
			// counter to make sure the same atom isn't added to two different
			// lists
			int addedCount = 0;
			for (Iterator it=atomMap.entrySet().iterator(); it.hasNext(); ) {
				Map.Entry entry = (Map.Entry) it.next();
				if (Math.abs((Double)entry.getKey() - occupancy) < CrystalTool.OCCUPANCY_EPS) {
					((List)entry.getValue()).add(disorderedAtom);
					added = true;
					addedCount++;
					//LOG.debug("adding to current list of occupancy: "+(Double)entry.getKey());
				}
			}
			// if atom added to more than one list then throw exception
			if (addedCount > 1) {
				throw new RuntimeException("Ambiguous atom occupancy.  Cannot resolve disorder");
			}
			if (added) continue;
			List newList = new ArrayList();
			newList.add(disorderedAtom);
			atomMap.put(occupancy, newList);
		}
		boolean fixed = false;
		if (unityAtoms.size() == disorderedAtoms.size()) {
			// if all the disordered atoms actually have an occupancy
			// of 1.0 then set them to have no assembly or group
			for (CMLAtom atom : unityAtoms) {
				replaceAtomDisorderInformation(atom, ".", ".");
			}
			fixed = true;
		} else {
			// try to figure out the assembly/groupings
			fixed = reassignDisorderGroups(atomMap);
		}
		return fixed;
	}

	/**
	 * Takes a map of occupancies related to their atoms with that occupancy.  Tries
	 * to decipher the correct disorder assemblies and groups from the occupancies
	 * supplied.
	 *
	 * @param atomMap
	 * @return whether or not the disorder groups have been successfully reassigned
	 */
	private boolean reassignDisorderGroups(Map> atomMap) {
		// if only 1 or 0 occupancies left, then we cannot create a valid disorder assembly.
		if (atomMap.size() < 2) {
			throw new RuntimeException("Cannot resolve disorder. Only one non-unit occupancy value found in molecule.");
		}
		List occupancyList = new LinkedList(atomMap.keySet());
		List> partitionList = Partition.partition(atomMap.size());
		for (List partition : partitionList) {
			for (Integer in : partition) {
				String assemblyCode = "";
				int groupCode = 1;
				CombinationGenerator cg = new CombinationGenerator(atomMap.size(), in);
				// list to keep track of the occupancies used in this iteration
				List dList = new ArrayList();
				while (cg.hasMore()) {
					double occCount = 0.0;
					int[] indices = cg.getNext();
					for (int i : indices) {
						double occ = occupancyList.get(i);
						occCount += occ;
					}
					if (Math.abs(1.0 - occCount) < CrystalTool.OCCUPANCY_EPS) {
						assemblyCode += "z";
						for (int i : indices) {
							double occ = occupancyList.get(i);
							// check if already contained in dList
							boolean inDList = false;
							for (Double d : dList) {
								if (d.compareTo(occ) == 0) {
									inDList = true;
								}
							}
							if (!inDList) dList.add(occ);
							List atoms = atomMap.get(occ);
							for (CMLAtom atom : atoms) {
								replaceAtomDisorderInformation(atom, assemblyCode,
										String.valueOf(groupCode));
							}
							groupCode++;
						}
					}
				}
				// if the number of occupancies used is the same as the total
				// number then we have found a solution
				if (dList.size() == occupancyList.size()) {
					return true;
				}
			}
		}
		return false;
	}

	private void replaceAtomDisorderInformation(CMLAtom atom, String assemblyCode, String groupCode) {
		List assemblyNodes = CMLUtil.getQueryNodes(atom, ".//"+CMLScalar.NS+"[" +
				"contains(@dictRef,'"+CrystalTool.DISORDER_ASSEMBLY+"')]", CMLConstants.CML_XPATH);
		if (assemblyNodes.size() > 1) {
			throw new RuntimeException("Atom "+atom.getId()+" contains more than one"
					+" disorder assembly.");
		} else {
			// remove any old nodes
			for (int i = 0; i < assemblyNodes.size(); i++) {
				assemblyNodes.get(i).detach();
			}
			// add new node
			CMLElement scalar = new CMLElement(CMLScalar.TAG);
			atom.appendChild(scalar);
			scalar.addAttribute(new CMLAttribute("dictRef", CrystalTool.DISORDER_ASSEMBLY));
			scalar.addAttribute(new CMLAttribute("dataType", "xsd:string"));
			scalar.setStringContent(assemblyCode.toString());
		}
		List groupNodes = CMLUtil.getQueryNodes(atom, ".//"+CMLScalar.NS+"[" +
				"contains(@dictRef, '"+CrystalTool.DISORDER_GROUP+"')]", CMLConstants.CML_XPATH);
		if (groupNodes.size() > 1) {
			throw new RuntimeException("Atom "+atom.getId()+" contains more than one"
					+" disorder group.");
		} else {
			// remove any old nodes
			for (int i = 0; i < groupNodes.size(); i++) {
				groupNodes.get(i).detach();
			}
			CMLElement scalar = new CMLElement(CMLScalar.TAG);
			atom.appendChild(scalar);
			scalar.addAttribute(new CMLAttribute("dictRef", CrystalTool.DISORDER_GROUP));
			scalar.addAttribute(new CMLAttribute("dataType", "xsd:string"));
			scalar.setStringContent(String.valueOf(groupCode));
		}
	}

	/**
	 * adds metadata to molecule stating whether or not the disorder
	 * has been successfully processed.
	 *
	 * @param processed - whether or not the disorder has been successfully processed
	 */
	public void addDisorderMetadata(boolean processed) {
		if (processed) {
			disorderProcessed = true;
		} else {
			disorderProcessed = false;
		}
		List disorderedAtoms = DisorderAssembly.getDisorderedAtoms(molecule);
		for (CMLAtom atom : disorderedAtoms) {
			CMLScalar scalar = new CMLScalar();
			atom.appendChild(scalar);
			if (!processed) {	
				scalar.setDictRef(UNRESOLVED_DISORDER_DICTREF);
			} else if (processed) {
				scalar.setDictRef(RESOLVED_DISORDER_DICTREF);
			}
		}
	}

	/**
	 * Checks for CIF disorder flags on each atom, plus some other telling
	 * things like methyl groups with 6 H/Cl/F attached etc.
	 *
	 * @param molecule
	 * @return true if disordered
	 */
	public static boolean isDisordered(CMLMolecule molecule) {
		for (CMLAtom atom : molecule.getAtoms()) {
			int hCount = 0;
			int fCount = 0;
			int clCount = 0;
			if (AS.C.equals(atom.getElementType())) {
				for (CMLAtom ligand : atom.getLigandAtoms()) {
					if (AS.H.equals(ligand.getElementType())) hCount++;
					if (AS.F.equals(ligand.getElementType())) fCount++;
					if (AS.Cl.equals(ligand.getElementType())) clCount++;
				}
			}
			if (hCount == 6 || fCount == 6 || clCount == 6) {
				return true;
			}

			List nodes = CMLUtil.getQueryNodes(atom, ".//" + CMLScalar.NS
					+ "[@dictRef='" + CrystalTool.DISORDER_ASSEMBLY + "'] | "
					+ ".//" + CMLScalar.NS + "[@dictRef='"
					+ CrystalTool.DISORDER_GROUP + "']", CMLConstants.CML_XPATH);
			if (nodes.size() > 0) {
				return true;
			}
		}
		return false;
	}

	/**
	 * analyze atoms with disorder flags.
	 *
	 * This may need to be rafactored to CIF Converter
	 *
	 * CIF currently provides two flags atom_disorder_assembly identifies
	 * different independent groups atom_disorder_group identifies the different
	 * instances of atoms within a single group from the CIF dictionary:
	 *
	 * ===atom_site_disorder_assembly
	 *
	 * _name _atom_site_disorder_assembly _category atom_site _type char _list
	 * yes _list_reference _atom_site_label
	 *
	 * _example _detail A disordered methyl assembly with groups 1 and 2 B
	 * disordered sites related by a mirror S disordered sites independent of
	 * symmetry
	 *
	 * _definition A code which identifies a cluster of atoms that show
	 * long-range positional disorder but are locally ordered. Within each such
	 * cluster of atoms, _atom_site_disorder_group is used to identify the sites
	 * that are simultaneously occupied. This field is only needed if there is
	 * more than one cluster of disordered atoms showing independent local
	 * order.
	 *
	 * ===atom_site_disorder_group
	 *
	 *
	 * _name _atom_site_disorder_group _category atom_site _type char _list yes
	 * _list_reference _atom_site_label
	 *
	 * _example _detail 1 unique disordered site in group 1 2 unique disordered
	 * site in group 2 -1 symmetry-independent disordered site
	 *
	 * _definition A code which identifies a group of positionally disordered
	 * atom sites that are locally simultaneously occupied. Atoms that are
	 * positionally disordered over two or more sites (e.g. the hydrogen atoms
	 * of a methyl group that exists in two orientations) can be assigned to two
	 * or more groups. Sites belonging to the same group are simultaneously
	 * occupied, but those belonging to different groups are not. A minus prefix
	 * (e.g. "-1") is used to indicate sites disordered about a special
	 * position.
	 *
	 * we analyse this as follows: if there is no disorder_assembly we assume a
	 * single disorder_assembly containing all disordered atoms. if there are
	 * several disorder_assemblies, each is treated separately. within a
	 * disorderAssembly there must be 2 or more disorder_groups each group is a
	 * separate locally disordered syste, which we describe by an atomSet. The
	 * occupancies within in group must be identical. We also check for whether
	 * the groups contain the same number of atoms and whether the occupancies
	 * of the groups sum to 1.0
	 *
	 * At this stage it may not be known whether the assemblies are in different
	 * chemical molecules. It is unusual for disorder_groups to belong to
	 * different subMolecules though it could happen with partial proton
	 * transfer
	 *
	 * @exception Exception
	 * @return list of disorderAssembly
	 */
	private List getDisorderAssemblyList() {
		if (molecule == null) {
			throw new IllegalStateException(
			"Molecule is null, no operations possible");
		}
		List disorderedAtomList = DisorderAssembly
		.getDisorderedAtoms(molecule);
		List disorderAssemblyList = DisorderAssembly
		.getDisorderedAssemblyList(disorderedAtomList);
		return disorderAssemblyList;
	}
}
class CombinationGenerator {

	private int[] a;
	private int n;
	private int r;
	private BigInteger numLeft;
	private BigInteger total;

	//------------
	// Constructor
	//------------

	/** constructor.
	 * @param n
	 * @param r
	 */
	public CombinationGenerator (int n, int r) {
		if (r > n) {
			throw new IllegalArgumentException ();
		}
		if (n < 1) {
			throw new IllegalArgumentException ();
		}
		this.n = n;
		this.r = r;
		a = new int[r];
		BigInteger nFact = getFactorial (n);
		BigInteger rFact = getFactorial (r);
		BigInteger nminusrFact = getFactorial (n - r);
		total = nFact.divide (rFact.multiply (nminusrFact));
		reset ();
	}

	//------
	// Reset
	//------

	/** reset.
	 * 
	 */
	public void reset () {
		for (int i = 0; i < a.length; i++) {
			a[i] = i;
		}
		numLeft = new BigInteger (total.toString ());
	}

	//------------------------------------------------
	// Return number of combinations not yet generated
	//------------------------------------------------

	/** 
	 * @return big int
	 */
	public BigInteger getNumLeft () {
		return numLeft;
	}

	//-----------------------------
	// Are there more combinations?
	//-----------------------------

	/**
	 * @return has more
	 */
	public boolean hasMore () {
		return numLeft.compareTo (BigInteger.ZERO) == 1;
	}

	/**
	* Return total number of combinations
	 * @return big int
	 */
	public BigInteger getTotal () {
		return total;
	}

	//------------------
	// Compute factorial
	//------------------

	private static BigInteger getFactorial (int n) {
		BigInteger fact = BigInteger.ONE;
		for (int i = n; i > 1; i--) {
			fact = fact.multiply (new BigInteger (Integer.toString (i)));
		}
		return fact;
	}

	/** Generate next combination (algorithm from Rosen p. 286)
	 * 
	 * @return combinations
	 */

	public int[] getNext () {

		if (numLeft.equals (total)) {
			numLeft = numLeft.subtract (BigInteger.ONE);
			return a;
		}

		int i = r - 1;
		while (a[i] == n - r + i) {
			i--;
		}
		a[i] = a[i] + 1;
		for (int j = i + 1; j < r; j++) {
			a[j] = a[i] + j - i;
		}

		numLeft = numLeft.subtract (BigInteger.ONE);
		return a;

	}
}
class Partition {
	private static Logger LOG = Logger.getLogger(Partition.class);
	/**
	 * returns all the ways the integer provided can be divided
	 * into whole number parts.  This supplying 5 returns
	 * 
	 * 1 1 1 1 1
	 * 2 1 1 1
	 * 2 2 1
	 * 3 1 1
	 * 3 2
	 * 4 1
	 * 5
	 * 
	 * @param n
	 * @return list
	 */
	public static List> partition(int n) {
		List> partitionList = new ArrayList>();
		List partition = new ArrayList();
		return partition(n, n, "", partitionList, partition);
	}

	private static List> partition(int n, int max, String prefix,
			List> partitionList,
			List partition) {
		if (n == 0) {
			//LOG.debug(prefix);
			partitionList.add(new ArrayList(partition));
			partition.clear();
			return partitionList;
		}

		for (int i = Math.min(max, n); i >= 1; i--) {
			//LOG.debug(prefix + " " + i);
			partition.add(i);
			partition(n-i, i, prefix + " " + i, partitionList, partition);
		}
		return partitionList;
	}

	/** main.
	 * 
	 * @param args
	 */
	public static void main(String[] args) {
		List> partitionList = partition(5);
		for (List partition : partitionList) {
			for (Integer in : partition) {
				Util.println(" ");
			}
			LOG.debug("");
		}
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy