-
Notifications
You must be signed in to change notification settings - Fork 0
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Mismatch increases in this case #35
Comments
Note that the mismatch does decrease when the grid size is made large in the z dimension (as it should be if one wants to correct this kind of jitter). I just discovered this by mistake, and it's concerning to me that it's choosing a solution that increases the mismatch. Even with a full-sized grid spacing in z it has trouble finding the right deformation, but it's not too far off. I'm looking into improving that now. |
It's pretty complicated to understand what's happening in that test case, so I've been looking for simpler ways to trigger strange behavior. I've found some surprising cases with 1-D registration. Again I'm creating the moving image by jittering the sampling locations in the one dimension. I find that
Here's an example script. If you run it a few times you'll notice that my guess is often better than the algorithm's, and the monitored mismatch is almost always negative. Could this be a problem with the way we're interpolating the mismatch? using BlockRegistration, BlockRegistrationScheduler, RegisterDeformation, RegisterMismatch
using ImageView2, Images
using FixedSizeArrays
using JLD
using Interpolations
#utility function for running Apertures registration on a pair of images
function _reg_def(fixed, moving, knots, mxshift; lambda = 0.01)
λ = lambda
algorithm = Apertures[Apertures(fixed, knots, mxshift, λ, identity; pid=myid(), correctbias=false) for i = 1:1]
mon = monitor(algorithm, (), Dict{Symbol,Any}(:u=>ArrayDecl(Array{Vec{1,Float64},1}, ([convert(Int,x.len) for x in knots]...))))
mon[1][:mismatch] = 0.0
fileout = tempname()
@time driver(fileout, algorithm, moving, mon)
u = JLD.load(fileout, "u")
mm = JLD.load(fileout, "mismatch")
return u, mm
end
#add jitter in sampling location, simulating inconsistencies in piezo position when using OCPI under certain conditions
function jitter{T}(img::Array{T,1}, npix::Float64)
@assert npix < 0.5 && npix >= 0.0 #don't tear space
etp = extrapolate(interpolate(img, BSpline(Linear()),OnGrid()), Flat())
out = zeros(eltype(img), size(img))
z_def = Float64[]
for i in 1:length(img)
r = (rand()*2*npix)-npix
push!(z_def, r)
out[i] = etp[i+r]
end
return out, z_def
end
fixed = zeros(6)
fixed[3:4] = 1.0
moving, z_def = jitter(fixed, 0.45);
mxshift = (1)
gridsize = (6)
knots = map(d->linspace(1,size(fixed,d),gridsize[d]), (1:ndims(fixed)...))
u, mm = _reg_def(fixed,moving,knots,mxshift;lambda=0.0);
@show mm #the monitored mismatch is negatve. red flag?
d = griddeformations(u, knots)
warped = warp(moving, d[1])
ideal_u = fill(0.0, size(u))
ideal_u[1,2,1] = -0.9
ideal_u[1,5,1] = 0.9
ideal_d = griddeformations(ideal_u, knots)
ideal_warped = warp(moving, ideal_d[1]);
@show ratio(mismatch0(moving,fixed), NaN)
@show ratio(mismatch0(warped,fixed), NaN) #this time it's not negative
@show ratio(mismatch0(ideal_warped,fixed), NaN) #our guessed answer is usually better |
Thanks for the very helpful issue reports. I've started with the last report, the 1d example. I can trace the negative mismatch to a simple source: In your example, the mismatch at certain grid points looks like z = NumDenom(0.0,1)
o = NumDenom(1.0,1)
mm = CenterIndexedArray([z, z, o]) and we can reproduce the negative penalty with a simple analog: julia> using Interpolations
julia> penalty = [0.0, 0.0, 1.0]
3-element Array{Float64,1}:
0.0
0.0
1.0
julia> itp = interpolate!(penalty, BSpline(Quadratic(InPlace())), OnCell())
3-element Interpolations.BSplineInterpolation{Float64,1,Array{Float64,1},Interpolations.BSpline{Interpolations.Quadratic{Interpolations.InPlace}},Interpolations.OnCell,0}:
0.0
-2.77556e-17
1.0
julia> itp[1.51]
-0.08792000000000001 You get the same thing from itp = CenterIndexedArray(interpolate!([z, z, o], BSpline(Quadratic(InPlace())), OnCell()))
RegisterPenalty.vecindex(itp, Vec(-0.49)) In a certain sense, the returned value is correct: if you interpolate a quadratic through those three points, the minimum will occur between the first and second point, and the value will be negative. However, it's also not hard to understand how this could be problematic. Linear interpolation wouldn't suffer from the same problem, but the issue there is that (for efficiency) we really need the gradient of the penalty during optimization, and you don't have a smooth gradient unless you use at least quadratic interpolation. Bottom line, I'm not quite sure what to do about this. I almost wonder if we should switch to a completely different optimization algorithm and focus only on integer-valued shifts. |
One option might be to do linear interpolation and use a subgradient method. I might be tempted to move so that the maximum step is 0.1 pixel for any grid point, and just stop when the resulting penalty increases. I'll try cooking something up along those lines. |
That would make sense to me! I also thought we would have to keep the step size small if we went with something like that. There's one other thing I should bring up before we proceed. I think there could be a fundamental problem with interpolating the mismatch. Consider So my question is, how hard would it be to move the interpolation upward into the mismatch calculation itself? I think I don't understand Fourier methods well enough to know if that makes sense. Alternatively I think it may make sense to upsample the images before running registration, though that may get expensive... |
Keep in mind that a linear interpolation of |
Oh yes you are right! So we wouldn't get a perfect answer like I said. But we would still get a lower mismatch value than at the integer locations. |
True. Interestingly, for that example, quadratic interpolation would suggest the right location for the minimum mismatch---the mismatch is symmetric around a half-grid point, so quadratic interpolation would place the minimum at 0.5. |
But it seems to me like the quadratic may be hurting us as often as it helps. I ran registration just to verify that it finds the right minimum in that case, and surprisingly it doesn't: using BlockRegistration, BlockRegistrationScheduler, RegisterDeformation, RegisterMismatch
using FixedSizeArrays
using JLD
using Interpolations
#utility function for running Apertures registration on a pair of images
function _reg_def(fixed, moving, knots, mxshift; lambda = 0.01)
�� = lambda
algorithm = Apertures[Apertures(fixed, knots, mxshift, ��, identity; pid=myid(), correctbias=false) for i = 1:1]
mon = monitor(algorithm, (), Dict{Symbol,Any}(:u=>ArrayDecl(Array{Vec{1,Float64},1}, (1,))))
mon[1][:mismatch] = 0.0
fileout = tempname()
@time driver(fileout, algorithm, moving, mon)
u = JLD.load(fileout, "u")
mm = JLD.load(fileout, "mismatch")
return u, mm
end
fixed = [0.0;0.0;1.0;1.0;0.0]
moving = [0.0;0.5;1.0;0.5;0.0]
mxshift = (1)
gridsize = (1)
knots = ([3],)
u, mm = _reg_def(fixed,moving,knots,mxshift;lambda=0.0);
@show u # == -0.336435 |
Sorry the last line should have been |
One issue I noted is that But I have one or more fixes coming (still testing...). |
Worth pointing out something else in this context. Setting julia> ϕ
RegisterDeformation.GridDeformation{Float64,1,Array{FixedSizeArrays.Vec{1,Float64},1},LinSpace{Float64}}(6-elementArray{FixedSizeArrays.Vec{1,Float64},1}:
Vec(0.0)
Vec(-1.0)
Vec(0.0)
Vec(-1.0)
Vec(1.0)
Vec(0.0)
,(linspace(1.0,6.0,6),))
julia> warped
6-element Array{Float64,1}:
0.0
NaN
0.898912
0.898912
0.0
0.0 Notice the |
Oops accidentally closed the issue via tab+enter
Ah interesting, that is good to know. One thing I haven't quite figured out is how to decide what is a small lambda for a given grid size. It seems to me that the two parameters are linked in a complex way. i.e. a |
I'd have to go through the code to remind myself what's being computed, but here's a guess. Our grid is a discrete approximation of a continuous penalty that looks something like
where
which implies is that That said, |
Given HolyLab/BlockRegistration.jl#46, I should post a script for testing things independently of BlockRegistrationScheduler: function reg_lowlevel(fixed, moving, knots, mxshift, λ = 0.01, order=Quadratic)
cs = coords_spatial(fixed)
gridsize = map(length, knots)
aperture_centers = aperture_grid(size(fixed, cs...), gridsize)
aperture_width = default_aperture_width(fixed, gridsize, zeros(Int, ndims(fixed)))
mms = mismatch_apertures(fixed, moving, aperture_centers, aperture_width, mxshift; normalization=:pixels)
E0 = zeros(size(mms))
cs = Array(Any, size(mms))
Qs = Array(Any, size(mms))
thresh = length(fixed)/prod(gridsize)/4
for i = 1:length(mms)
E0[i], cs[i], Qs[i] = qfit(mms[i], thresh; opt=false)
end
global mmc = deepcopy(mms) # for debugging
mmis = interpolate_mm!(mms, order)
ap = AffinePenalty{Float64,ndims(fixed)}(knots, λ)
ϕ, mismatch = RegisterOptimize.fixed_λ(cs, Qs, knots, ap, mmis)
end and you can replace a couple of lines of your script (the one that calls ϕ, mm = reg_lowlevel(fixed, moving, knots, mxshift, 0.001, Linear)
warped = warp(moving, ϕ) |
On your first script, if I allow 101 grid points along z, then I find that the best setting is something like ϕ, mm = reg_lowlevel(fixed, moving, knots, mxshift, 10, Quadratic); Using lower julia> @show ratio(mismatch0(warped,fixed), NaN)
ratio(mismatch0(warped,fixed),NaN) = 0.0023416889785358806
0.0023416889785358806
julia> @show ratio(mismatch0(moving,fixed), NaN)
ratio(mismatch0(moving,fixed),NaN) = 0.0016513571370021908
0.0016513571370021908 Keep in mind that our registration does not try to align the images; all the registration is performed on the (block-computed) mismatch data. So a parameter setting that decreases the interpolated mismatch may not decrease the mismatch with the warped image. This is obviously a weakness of our algorithm, but it's the key to its fast performance. If you have to warp and then compute the mismatch for every possible change in the deformation, that's going to be a big cost. We try to make things more efficient by predicting the impact that change will have by interpolation, but obviously it doesn't always work very well. I wonder if we should create a completely different algorithm for handling just the z component? It's much faster to interpolate along just one axis, which you may know you can do like this: julia> A = rand(5, 5, 7);
julia> itp = interpolate(A, (NoInterp(), NoInterp(), BSpline(Quadratic(Flat()))), OnCell()); This would involve 3 rather than 27 evaluation points per output point. |
It occurs to me that there may be a solution that allows us to have our cake and eat it too. The mismatch calculation looks like It seems possible that it would afford higher accuracy, because we'd have the actual gradient of the mismatch (at integer-pixel positions) rather than an estimate of the gradient calculated by finite differencing. However, before plunging in, considerable caution seems warranted. Implementing this would be a lot of work; not only would we have to implement the calculation of the gradient for integer shifts, but we'd have to develop the interpolation scheme from the math on up (I'm not aware of any such interpolation scheme in any language). I'd want a clear demonstration (in a simple situation, e.g., 1d) that the improvement in estimating the mismatch gradient was worth all the effort. An alternative is to add a stage at the end that iteratively warps and tests the result to achieve subpixel accuracy. |
It's worth pointing out that the scheme above still wouldn't accurately model the change in the deformation between grid points: any time you use the FFT, you're forced to hold the shift constant over each block. In contrast, when we go to warp the image, we interpolate the deformation. Consequently the above scheme still wouldn't represent a fully-accurate model of how the mismatch will depend on the deformation. |
Sorry for the delay in answering this!
Thanks, I will do that!
I was thinking about doing something like this as a preprocessing step. I had imagined just finding the linear combination of adjacent moving slices that matches each fixed slice the best. Do you think we will gain something from quadratic interpolation in that case? I can try it both ways.
I think this may be akin to what I (vaguely) suggested, to "move the interpolation upward into the mismatch calculation". I do recognize it's a lot of work, and I'm not sure I would trust myself to handle the FFT gradients properly without spending a long time. I expect that we could get better sub-pixel registration if we could pull it off. I can start by testing the iterative warping approach. If that improves accuracy a lot then maybe we can consider implementing this to improve performance (though I realize that this could still have accuracy shortcomings compared to iterative warping for the reason you mentioned). I'll be testing this stuff out today and tomorrow, thanks for the improvements! |
In addition to the new work underway in It's easiest to explain if you take the first stack as the fixed stack. Stack 2 is then the first "moving" stack. We perform a correction on stack two by examining pairs of adjacent moving slices in combination with the fixed slice. The idea is that each fixed slice was sampled "between" a pair of moving slices in physical space, so we can find the interpolated slice in the moving image that best matches each fixed slice. The set of best-matching interpolated slices then replaces the original moving image. We repeat the same procedure with stack 3, but using the corrected stack 2 as the fixed image. This is dirt simple really, and the optimal (linear) interpolation coefficient for each stack can be found analytically. (Before I realized the analytical solution was so easy I used an iterative approach with Optim, and it actually failed to find the optimal solution. I never tracked down why.) There is one caveat: if a fixed slice doesn't fall "between" a pair of moving slices then it fails. This can actually happen when the fixed slice happens to be sampled in a plane with a lot of peaks in intensity relative to surrounding space. In order to address this I also solve the inverse problem: solve again for the coefficients when the fixed and moving roles are swapped. When doing this we end up with complementary estimates of the displacement, and we can weight each one by the mismatch that it achieves. This can be viewed as a form or regularization, and in extreme cases the solution we get by combining both the forward and inverse solution can actually increase the mismatch even though the displacements it finds are correct. There are more details in the comments of my gist. I can clean up the gist, write tests, and push it to Github if you don't see any fundamental problems with the approach. In practice it's improving my images a lot by eye, and it's nice that it's so fast (about 1 second per stack to find the coefficients). Actually currently it takes 3x longer to resample the moving image than it does to find the solution. I'm not sure why that is; the resampling function looks fine via Here's the gist: https://gist.github.com/Cody-G/c36831daf2db7c469713db9507a05aea |
This is awesome. I love it because it's so directly targeted at the specific problem that is causing you the most trouble. I definitely see this as an excellent "preprocessing" step. If it obviates the need for HolyLab/BlockRegistration.jl#48 so much the better, although I won't be surprised if in the end we want both. But I'd suggest going piecemeal, and this sounds like the best piece to add first. Before you turn it into a PR, here's one thing I noticed. Let This might be considerably slower than your approach, because you only consider pairs, but it might be at least worth playing with. Those BLAS routines are fast. |
Formulating it as a single matrix equation is attractive. In that case adding a non-negativity constraint may be important, as I've seen negative solutions pop out with my pairwise approach. I can easily throw those solutions away now, so it would be nice to keep that ability. If we go with NMF are we guaranteed to find the best solution? I didn't think that was guaranteed with our HALS approach, but I'm not sure. Probably the more important concern is how to constrain entries of the weight matrix to be less than one. If an entry is greater than one it can no longer be interpreted as an interpolation (more like an extrapolation). Again I can deal with it now by just throwing away infeasible solutions. The only way I know how to solve the full matrix method is with a Lagrange multiplier for each slice, though I think there are a lot of methods I'm not familiar with. One could also argue that it's okay to let the coefficients exceed one to respect the reality that some slices could have a longer dwell time over the same region than others, but I don't think that difference is very significant. Even if it were I think we would also capture it in the pairwise "inverse" solution. I'll play around with this. The data may already constrain the problem enough that my concerns are for nothing... |
General NMF is not convex so there is no guarantee of getting the best answer, no matter what algorithm you use. However in this case it's a simpler problem, since one of the two factors is already specified, so this is really a case of nonnegative least squares, which is convex (https://en.wikipedia.org/wiki/Non-negative_least_squares). So you're guaranteed to find the optimum. I suspect that's true even if you add an upper-bound constraint, but I'm not sure. That's certainly true of the single-factor, single-coordinate updates that HALS is based on. Perhaps the most accessible treatment I know of is http://cmp.felk.cvut.cz/ftp/articles/franc/Franc-TR-2005-06.pdf, with the key point being covered in section 4. However, I also agree that the unconstrained problem might be good enough, if we're careful in how we use the solution. |
Here's an interesting view of the full matrix solution. I zoomed in on the diagonal. Notice that the dark spots are negative. That was a synthetic dataset. Here is the result from a pair of real stacks: I think it's actually pretty neat, and it smells a bit like deconvolution! I'm tempted to use all of the information around the diagonal, but I haven't fully convinced myself that this is fair in all circumstances. What do you think? In terms of performance the full solution takes about 5x as long as my pairwise solution. If I use the linear algebra solver to enact the pairwise solution (i.e. I'm thinking the only reason to change to BLAS is if we want to use the full, unconstrained matrix. Like I said, it's really tempting. I will see if I find any obvious problems in the resulting images. |
Hmm I have a linear algebra question. I solved for So of course when I try to use the inverse matrix it does a terrible job of reconstructing the |
...and if I go back and solve for It's a bit scary that the results are so different! I'll be more careful about using Unfortunately even with 64-bit precision the inverse matrix doesn't do a good job of reconstructing |
It's possible there's something ill-conditioned. Worth considering Q, R, p = qr(M, Val{true}) # pivoting should put your least-informative frames last
# For any diagonal elements of R that are almost zero, or even much smaller than the rest,
# set them to **Inf** here
Wp = R \ (Q' * F[:, p])
pinv = invperm(p)
W = Wp[pinv, pinv] # not sure I've got this right...
What does |
It might be a little slower, but perhaps it would be easiest to do all this with the USV = svdfact(M)
S = USV[:S]
thresh = sqrt(eps(eltype(S))) * maximum(S)
S[S .< thresh] = Inf
W = USV \ F That way you don't have to deal with pivoting. |
I tried all of your suggestions for calculating W, and I'm still getting a poor reconstruction of the moving image. The fixed image reconstruction looks great (i.e. m64 = sqrt.(Float64.(moving)) #convert from UInt16, square root transform
f64 = sqrt.(Float64.(fixed))
szxy = prod(size(m64)[1:2])
szz = size(m64,3)
M = reshape(m64, szxy, szz)
F = reshape(f64, szxy, szz)
W = M\F
F_reconstr = M*W #looks good
W_reconstr = F*inv(W) #very noisy image, can barely see anything
Not very close. Here's a plot of the (sorted) entries in the diagonal. There is one outlier, could that be a problem?
This does work for getting me a
Another scatter of the entries:
Same story as other methods for calculating W |
Can't really tell without log scaling on the y-axis, but some of those last singular values look extremely close to 0 (the last one in particular). If so, it's not at all surprising you're getting bad reconstruction; if some of the singular values are close to 0, you can never take Just use the same trick again: Wfact = svdfact(W)
S = Wfact[:S]
thresh = sqrt(eps(eltype(S))) * maximum(S)
S[S .< thresh] = Inf
Mrecon = Wfact \ F If you use this trick wherever you need it, I bet you'll be fine with |
I can't try your specific example because I don't have |
Your last line, I also started playing with the idea of removing jitter in all three dimensions simultaneously using this approach (we also get sub-pixel high frequency X-Y jitter from vibrations and the fish's heartbeat). I haven't yet figured out how to set up the linear algebra to solve for three dimensional recombination of slices. I started reading up a bit about tensors...does this sound like a useful direction to you? I'm thinking these sub-pixel corrections could make a big difference with segmentation. |
Just to expand on the last thing I said about removing jitter in multiple dimensions: I think for the 2D case we would need to solve for two weight matrices
That's one equation with two unknowns, which seems underconstrained. Intuitively it makes sense that it's undersconstrained because you can imagine a whole family of solutions that differ only by a scaling factor of I also noticed that any of the recombinations we are interested in can be reduced to one-dimensional under a coordinate transformation. But I'm not sure it's even possible to mix coordinate transformations with our current recombination approach--I haven't yet found a way. I could always just create a loss function and pass off the problem to |
That's a really interesting suggestion. I've spent some time playing with this in this gist. Because I found that the unconstrained problem settled on really whacky values for the "interpolation coefficients," I implemented it so that the sum-of-absolute-values of the "interpolation coefficients" is 1. (This is an L1 regularization, quite reminiscent of LASSO except that there isn't an unknown lagrange multiplier.) On a couple of simple tests, I've gotten very good results for shifts along a single axis, but as soon as I shift along two axes simultaneously I get junk. I don't see a way out, currently. Pity, because it's a really creative idea! |
Interesting. What if we give it a good initial guess? I think we could initialize the weight matrices with a simple transform that aligns the centroids of the two images. I think we could do that by shifting the two starting identity matrices by their respective coordinates of the centroid difference vector. I will give that a try after I figure out what's causing the gradient tests to fail in RegisterPixelwise |
When I use a good initial guess with your example in the gist, the quality of the reconstruction actually goes down after the first iteration and then oscillates all over the place. I didn't study very carefully how you're doing the coordinate descent, but maybe we could try a different method? |
Hmm, that's interesting. Theoretically it should result in guaranteed descent; if it's not doing that, it suggests a bug. As for switching to a different method, I'm all ears. But I don't see a simple alternative, do you? Coordinatewise descent is, as far as I know, a state-of-the-art method for LASSO (and simple, too). |
Okay I have good news! The less constrained method in your updated gist works! Also I've expanded it to 3D, and it also seems to work well there. I'll clean it up a bit and post a new gist. Some observations
|
That sounds fantastic! I'm really excited that this seems promising! With regards to the limitations, I think it's more general (and therefore harder to solve) than issues specifically with coordinate descent. The algorithm that constrains only
I suspect the biggest issue is that there are surely many minima and we're not guaranteed to find the best one. In 1d the problem is convex, but it's not in any higher dimensionality. Thus we are guaranteed to find the global minimum for a 1d problem, but there is no such guarantee for 2d or higher. A couple of your points do make me throw out one caution. Appearance of the image is not always the best guide to success; in particular, this approach does seem to run some risk of falling into the CURT trap. At some point we'll have to take the output of this algorithm and turn it into something that's in the spirit of a deformation. One of the things I like about this is that it can come up with things that aren't strictly interpolations (e.g., it can exploit negative coefficients), but in the end I suspect we'll need to impose some kind of locality. It would be really interesting to see what this algorithm does with a small-angle rotation. In particular it would be exciting if we can somehow make sense of the parameters to extract the angle(s). In contrast with the shift case, I can't easily picture what those matrices should look like. I even wonder if we'd need to consider multi-tensor solutions, i.e., instead of our current version which can be written as I wonder if we might need to use
|
I may have misidentified that as the issue, but our problem when doing alternating least squares feels analogous when you consider a test case like this:
If we think of the I also agree that it's very difficult to extract something that's in the spirit of a deformation, especially if we want to include rotations or, most generally, affine transforms. Here's one proposal that I think may be able to solve the original problem I wanted to solve, which was correcting for small differences between adjacent stacks in a timeseries (probably never more than 1 pixel of motion):
Here's an updated version of your gist that includes my 3D implementation and some test cases demonstrating the points I've made. https://gist.github.com/Cody-G/9d3c0e5c8a7fb20812234c1f8ad7457e I'm doing the tensor operations very inefficiently, calling |
I see your point. Certainly similar in spirit, if not exactly the same. It's essentially like a conventional deformation-based approach where you parametrize the deformation as a simple translation, and suppose you're aligning two images which are black (0) everywhere except for one bright blob. With conventional approaches, you move in the direction determined by the gradient, but if there's no overlap between the blobs, then the gradient at your starting point (no translation) is 0. So you might be able to reduce the mismatch, but the gradient doesn't help you figure out how to do it. This is part of the reason that our "forecasting" method aims to find the global minimum within the set of all allowable shifts. |
It's quite possible this is the most efficient approach. I definitely wouldn't worry about it until you know it's a problem. |
I see the analogy, but it's worth emphasizing that the alternating method can fail even when there is image overlap, as in the case below. If we were optimizing both matrices simultaneously I think the gradient would actually help us find the right solution in this case: fixed = [1.0 0 0;
0.0 1 0;
0.0 0 0]
moving = [0 0 0;
0 1 0
0 0 1.0] There's actually a lot of literature on solving similar problems. I think that PARAFAC might be the most similar to what we want to do, but I'm not sure. ALS seems like a popular algorithm but it's not the only one. I'm not sure how much time it's worth putting into this...I'll be back in the office soon, maybe we can discuss more in person. http://epubs.siam.org/doi/pdf/10.1137/07070111X |
I did think of one potential concern with this approach, even for the case of 1d registration, although things we're already incorporating may solve the problem effectively. Let's suppose you're looking at a field of view in which there is no movement, but in one particular frame every neuron gets brighter simultaneously. Then the matrix might be something like This makes it seem all the more important to constrain the sum over columns to be 1 (which circumvents that problem). |
I agree with your concern about global brightness changes. I just created a repository with my work on this so far. You can switch between three methods using a keyword argument: one is unconstrained, another is the sum-to-1 constraint, and another is the non-negative constraint. As we discussed, I'm thinking the nonnegative one may be helpful because with negativity it manages to find some creative (and physically impossible) ways to combine slices to cancel out shot noise. I do wonder whether we should just allow it to do this noise canceling. Images tend to get blurred a bit, but I don't think it's reducing the SNR (still need to quantify). In its current form the nonnegative method isn't working. I tried to use our |
Here's the issue over there: |
For some time I've been trying to get good registrations when there is inconsistency in sampling locations along the z-axis. This is common with high-speed OCPI recordings. We're doing a lot to handle this at the level of the microscope hardware and acquisition software, but it would be good if our registration algorithm could clean up the last bit of it. I wasn't satisfied with the results so i made a synthetic image volume and simulated the situation (script below). I'm seeing that the loss (aka mismatch) actually increases when registering these volumes. I didn't yet track down why that is, and this can probably be triggered by a simpler test case, but here's a demo that I know demonstrates the issue (note that it requires a utility package
BinaryVolumes
, another HolyLab repo.)The text was updated successfully, but these errors were encountered: