In the article of Maxim Shatsky et al. entitled "A Method for the Alignment of Heterogeneous Macromolecules from Electron Microscopy" (J Struct Biol. 2009, 166(1): 67–78. doi:10.1016/j.jsb.2008.12.008.), the authors show that pure noise images aligned to a reference image give an average image resembling the reference. This is a perfect example of the model bias occurring during the 2D classification step. Let's do it!!!
1- Pre-processing
In the article, they chose the iconic image of Einstein by Arthur Sasse. I just crop and resize this image in 128x128. Additionally, I convert it in 32-bits (Image > Type > 32-bits) and rescale the image in order to have a zero mean and a sigma equal to 1.0 (Process > Math > Macro with a formula v=(v-mean)/stdev ). To get the values of mean and stdev, the simplest way is to calculate the histogram (Analyze > Histogram) (Fig.1).
![]() |
Fig.1: Reference image inspired by those used in the article of Shatsky et al. This image is copyrighted and I did not granted permission, so this image will be removed upon request. |
2. 2D alignment by cross-correlation
3. The source code
The script below is composed of three parts:
- (i) Initialization
- (ii) Creation of the stack of gaussian noise slices
- (iii) 2D alignment of noise to the reference by cross-correlation
The computation of the average image from the "aligned" stack is not included in the script. You can easily do that in (Image > Stacks > Z-Project and choose Average Intensity).
+++ IJ JavaScript snippet +++
+++End of JavaScript snippet +++
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/* | |
Model Bias by aligning from a reference | |
Jean-Christophe Taveau | |
2022/07/19 | |
*/ | |
IJ.log('\\Clear'); | |
// Einstein | |
const ref = IJ.getImage(); | |
const refName = ref.getTitle(); | |
const size = ref.getWidth(); | |
const half = ref.getWidth()/2.0; | |
// Create stack of N images of pure gaussian noise | |
const N=300; | |
const noise = IJ.createImage("Noise", "32-bit black", size, size, N); | |
const stack = noise.getImageStack(); | |
for (let i=0; i < stack.getSize();i++) { | |
stack.getProcessor(i+1).noise(1.0); | |
} | |
noise.show(); | |
// Align to a reference | |
for (let i=0; i < stack.getSize();i++) { | |
let tmp = new ImagePlus("NoiseFrame",stack.getProcessor(i+1)); | |
tmp.show(); | |
IJ.run(tmp, "FD Math...", `image1=${refName} operation=Correlate image2=NoiseFrame result=Result_${i} do`); | |
let corrmap = WindowManager.getImage(`Result_${i}`); | |
const ce = new ContrastEnhancer(); | |
ce.setNormalize(true); | |
ce.stretchHistogram(corrmap,0.0); | |
// Force update | |
corrmap.updateAndDraw(); | |
IJ.log(`${i}: ${corrmap.getTitle()}`); | |
let maxFinder = new MaximumFinder(); | |
const list = maxFinder.getMaxima(corrmap.getProcessor(), 0.8, 0); | |
// IJ.log(`${i+1}: ${list.xpoints.length},${list.ypoints.length},${list.xpoints[0]},${list.ypoints[0]}`); | |
const tx = half - list.xpoints[0]; | |
const ty = half - list.ypoints[0]; | |
noise.setSlice(i+1); | |
IJ.run(noise, "Translate...", `x=${-tx} y=${-ty} interpolation=Bilinear slice`); | |
tmp.close(); | |
corrmap.close(); | |
} | |
No comments:
Post a Comment