using a sheet in an excel user defined function

前端 未结 3 1785
夕颜
夕颜 2020-12-03 16:17

The VBA I\'m trying to write is fairly simple but Ive never written VBA and coming from the visual studio & C# world, this truly is hell!!

So I\'ll really be grat

相关标签:
3条回答
  • 2020-12-03 16:31

    Its usually best in cases like this to work within the Excel paradigm rather than convert to the C# way: Not sure why you need a UDF rather than the Excel formulae which your Finance guy has coded: why can't you/finance guy just code some fomulae to feed the inputs to the finance guy sheets, then write some more formulae to feed the results back? If you really need to do this with VBA then J Tolle has given you the basic architecture for a repetitive loop but I would recommend looking at doing it with formulae first.

    0 讨论(0)
  • 2020-12-03 16:42

    You might be making this more complicated than you need to. As a programmer, I'm sure you appreciate that when learning a new system, a little time with the help, Google, "Dummies" books, etc. is time well spent. Excel emphasizes a very different programming style than, say, C#.

    So, here are some options, starting with the simplest and most Excel-idiomatic:

    1) Use a formula in your result column. If the calculation you want to do is simple enough, this is the way to go. As a trivial example, say your calculation is just adding the two numbers. You'd put the formula

    =B2+C2
    

    into cell E2, and then drag it down the E column. Excel will know what to do and make forumlas for E3=B3+C3 and so on.

    You can do quite a lot with just formulas. Excel has many built-in worksheet functions, and you can find tons of material on the web and in the bookstore.

    2) Write a UDF in VBA and call it from the result column. A UDF is a function that doesn't modify anything else in the worksheet. It just calculates a result based on its input parameters. You would write a UDF when you have something too complex to do in a formula, or when you want to use the formula in multiple places. (Note that dragging a forumla across a Range is idiomatic Excel and counts as "one place" for DRY purposes).

    So, to write a UDF (let's use the addition example, but you'd write more complex logic), you'd put a function like this into a VBA code module:

    Public Function myUDF(current, oneYear)
        myUDF = current + oneYear
    End Function
    

    You'd then put the formula

    =myUDF(B2, C2)
    

    into cell E2 and drag that down the E column.

    Note that I've neglected stuff like error handling in this example. This is one reason why formulas are more "Excel" than UDFs. Now you have to start writing code that departs from Excel's declarative functional style, and it's a different mental model.

    EDIT: 2.5) Use an Excel Data Table - I'm embarrased that I forgot this before, but if you have exactly one or two cells that are your independent parameters, and exactly one cell that is your result, then Excel has a built-in "what-if" analysis tool that you can use. The name "data table" is regrettable since both of those words are so heavily overloaded, but that's what it's called. You tell Excel what your input(s) and output are, and it will do what you want. See:

    http://office.microsoft.com/en-us/excel-help/calculate-multiple-results-by-using-a-data-table-HP010072656.aspx

    Here is the little intro from that link:

    A data table is a range of cells that shows how changing one or two variables in your formulas (formula: A sequence of values, cell references, names, functions, or operators in a cell that together produce a new value. A formula always begins with an equal sign (=).) will affect the results of those formulas. Data tables provide a shortcut for calculating multiple results in one operation and a way to view and compare the results of all the different variations together on your worksheet.

    3) Write a macro that uses your calc sheet like it was a function. You might do this if you've got really complex logic on that sheet, and it's too hard to convert to VBA because it changes all the time or because it's being developed by an Excel user who doesn't know VBA and isn't familiar with imperative programming. I'm not going to supply code, but the pseudocode might look like this:

    Public Sub doCalc()
        'for each cell in the result column
            'copy the corresponding current and oneYear cells to the calc sheet
            'Excel will recalc the calc sheet
            'copy the result cell on the calc sheet back to the Range sheet in the right place
        'end
    End Sub
    

    This is now departing pretty far from the "Excel way", since you have to set up your data yourself, and then run the macro. If you go back and change some of the "inputs", you need to re-run the macro. (This can be automated to some extent, using Worksheet events.)

    Lot's of programmers who are new to Excel jump right to this step, relegating Excel to a giant grid control and neglecting the reason it exists in the first place - automatic recalculation.

    EDIT IN RESPONSE TO ADDITIONAL INFO IN QUESTION:

    O.K., it looks like case (3) is appropriate because you have a complex sheet not easily manually translated into VBA or C# code. Here is some (oversimplified, hardcoded, lacking EH, etc.) code to give you more of an idea of how to do this in VBA:

    First, there is the VBA macro that pretends it's a UDF. Note that it is a Sub, not a function, and you don't call it from a formula. You call subroutines in response to the user doing something like choosing a menu option or pressing a command button:

    Public Sub fakeFunct(input1, input2, outputRng As Range)
        CalcSheet.[b1] = input1
        CalcSheet.[b2] = input2
    
        'Excel recalcs what's in cell B3 if you have it set to automatic recalcualation
    
        outputRng = CalcSheet.[b3]
    End Sub
    

    In this example, 'CalcSheet' is the "code name" of your worksheet module. (For more about this, see Chip Pearson's excellent website: http://www.cpearson.com/excel/codemods.htm)

    You'd actually call the above macro from another one that loops through your "inputs":

    Public Sub loopFakeFunct()
        Dim i As Long
        For i = 2 To 2000
            Call fakeFunct(RangeSheet.Cells(i, 2), RangeSheet.Cells(i, 3), RangeSheet.Cells(i, 5))
        Next i
    End Sub
    

    This is the thing you call in response to user action, or from the Worksheet_Change event, etc. It will be slow and clunky, and I'm leaving a lot of practical VBA stuff out, but it should be enough to get you started. (I used 'RangeSheet' instead of 'Range' as a sheet code name to avoid conflicts with the Range class and the Application.Range method defined by Excel.)

    4) Use some third-party tool that converts your worksheet into a UDF that you can then call as in (2). There are commerical tools out there that will take your sheet, let you specifiy which cells are inputs and outputs, and then generate C++ code or whatever that can be compiled into an XLL add-in.

    5) Wait for Microsoft to do this:

    http://research.microsoft.com/en-us/um/people/simonpj/papers/excel/excel.htm

    6) Use Resolver One:

    http://www.resolversystems.com/products/resolver-one/

    It has a feature that allows you to use a whole workbook as a function in another one, or I think in any Python code.

    0 讨论(0)
  • 2020-12-03 16:43

    You recorded your macro in "relative" mode. R[-8]C[-10] means: the cell 8 rows above and 10 columns to the left of current cell.

    0 讨论(0)
提交回复
热议问题