diff --git a/.Rbuildignore b/.Rbuildignore index 271a8b9f0..2e6feb1c8 100644 --- a/.Rbuildignore +++ b/.Rbuildignore @@ -1,3 +1,4 @@ +^pkgdown$ ^appveyor\.yml$ ^.*\.Rproj$ ^\.Rproj\.user$ diff --git a/.travis.yml b/.travis.yml index 8dbf0a835..f40cfa9dc 100644 --- a/.travis.yml +++ b/.travis.yml @@ -13,6 +13,11 @@ matrix: - r: 3.5 after_success: - Rscript -e 'covr::codecov()' + before_deploy: Rscript -e 'remotes::install_cran("pkgdown")' + deploy: + provider: script + script: Rscript -e 'pkgdown::deploy_site_github()' + skip_cleanup: true - r: devel env: diff --git a/docs/CNAME b/docs/CNAME deleted file mode 100644 index acba4a539..000000000 --- a/docs/CNAME +++ /dev/null @@ -1 +0,0 @@ -purrr.tidyverse.org \ No newline at end of file diff --git a/docs/CONDUCT.html b/docs/CONDUCT.html deleted file mode 100644 index 4356e5e2f..000000000 --- a/docs/CONDUCT.html +++ /dev/null @@ -1,138 +0,0 @@ - - - -
- - - - -As contributors and maintainers of this project, we pledge to respect all people who contribute through reporting issues, posting feature requests, updating documentation, submitting pull requests or patches, and other activities.
-We are committed to making participation in this project a harassment-free experience for everyone, regardless of level of experience, gender, gender identity and expression, sexual orientation, disability, personal appearance, body size, race, ethnicity, age, or religion.
-Examples of unacceptable behavior by participants include the use of sexual language or imagery, derogatory comments or personal attacks, trolling, public or private harassment, insults, or other unprofessional conduct.
-Project maintainers have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct. Project maintainers who do not follow the Code of Conduct may be removed from the project team.
-Instances of abusive, harassing, or otherwise unacceptable behavior may be reported by opening an issue or contacting one or more of the project maintainers.
-This Code of Conduct is adapted from the Contributor Covenant (http://contributor-covenant.org), version 1.0.0, available at http://contributor-covenant.org/version/1/0/0/
-GNU GENERAL PUBLIC LICENSE - Version 3, 29 June 2007 - - Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/> - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - - Preamble - - The GNU General Public License is a free, copyleft license for -software and other kinds of works. - - The licenses for most software and other practical works are designed -to take away your freedom to share and change the works. By contrast, -the GNU General Public License is intended to guarantee your freedom to -share and change all versions of a program--to make sure it remains free -software for all its users. We, the Free Software Foundation, use the -GNU General Public License for most of our software; it applies also to -any other work released this way by its authors. You can apply it to -your programs, too. - - When we speak of free software, we are referring to freedom, not -price. Our General Public Licenses are designed to make sure that you -have the freedom to distribute copies of free software (and charge for -them if you wish), that you receive source code or can get it if you -want it, that you can change the software or use pieces of it in new -free programs, and that you know you can do these things. - - To protect your rights, we need to prevent others from denying you -these rights or asking you to surrender the rights. Therefore, you have -certain responsibilities if you distribute copies of the software, or if -you modify it: responsibilities to respect the freedom of others. - - For example, if you distribute copies of such a program, whether -gratis or for a fee, you must pass on to the recipients the same -freedoms that you received. You must make sure that they, too, receive -or can get the source code. And you must show them these terms so they -know their rights. - - Developers that use the GNU GPL protect your rights with two steps: -(1) assert copyright on the software, and (2) offer you this License -giving you legal permission to copy, distribute and/or modify it. - - For the developers' and authors' protection, the GPL clearly explains -that there is no warranty for this free software. For both users' and -authors' sake, the GPL requires that modified versions be marked as -changed, so that their problems will not be attributed erroneously to -authors of previous versions. - - Some devices are designed to deny users access to install or run -modified versions of the software inside them, although the manufacturer -can do so. This is fundamentally incompatible with the aim of -protecting users' freedom to change the software. The systematic -pattern of such abuse occurs in the area of products for individuals to -use, which is precisely where it is most unacceptable. Therefore, we -have designed this version of the GPL to prohibit the practice for those -products. If such problems arise substantially in other domains, we -stand ready to extend this provision to those domains in future versions -of the GPL, as needed to protect the freedom of users. - - Finally, every program is threatened constantly by software patents. -States should not allow patents to restrict development and use of -software on general-purpose computers, but in those that do, we wish to -avoid the special danger that patents applied to a free program could -make it effectively proprietary. To prevent this, the GPL assures that -patents cannot be used to render the program non-free. - - The precise terms and conditions for copying, distribution and -modification follow. - - TERMS AND CONDITIONS - - 0. Definitions. - - "This License" refers to version 3 of the GNU General Public License. - - "Copyright" also means copyright-like laws that apply to other kinds of -works, such as semiconductor masks. - - "The Program" refers to any copyrightable work licensed under this -License. Each licensee is addressed as "you". "Licensees" and -"recipients" may be individuals or organizations. - - To "modify" a work means to copy from or adapt all or part of the work -in a fashion requiring copyright permission, other than the making of an -exact copy. The resulting work is called a "modified version" of the -earlier work or a work "based on" the earlier work. - - A "covered work" means either the unmodified Program or a work based -on the Program. - - To "propagate" a work means to do anything with it that, without -permission, would make you directly or secondarily liable for -infringement under applicable copyright law, except executing it on a -computer or modifying a private copy. Propagation includes copying, -distribution (with or without modification), making available to the -public, and in some countries other activities as well. - - To "convey" a work means any kind of propagation that enables other -parties to make or receive copies. Mere interaction with a user through -a computer network, with no transfer of a copy, is not conveying. - - An interactive user interface displays "Appropriate Legal Notices" -to the extent that it includes a convenient and prominently visible -feature that (1) displays an appropriate copyright notice, and (2) -tells the user that there is no warranty for the work (except to the -extent that warranties are provided), that licensees may convey the -work under this License, and how to view a copy of this License. If -the interface presents a list of user commands or options, such as a -menu, a prominent item in the list meets this criterion. - - 1. Source Code. - - The "source code" for a work means the preferred form of the work -for making modifications to it. "Object code" means any non-source -form of a work. - - A "Standard Interface" means an interface that either is an official -standard defined by a recognized standards body, or, in the case of -interfaces specified for a particular programming language, one that -is widely used among developers working in that language. - - The "System Libraries" of an executable work include anything, other -than the work as a whole, that (a) is included in the normal form of -packaging a Major Component, but which is not part of that Major -Component, and (b) serves only to enable use of the work with that -Major Component, or to implement a Standard Interface for which an -implementation is available to the public in source code form. A -"Major Component", in this context, means a major essential component -(kernel, window system, and so on) of the specific operating system -(if any) on which the executable work runs, or a compiler used to -produce the work, or an object code interpreter used to run it. - - The "Corresponding Source" for a work in object code form means all -the source code needed to generate, install, and (for an executable -work) run the object code and to modify the work, including scripts to -control those activities. However, it does not include the work's -System Libraries, or general-purpose tools or generally available free -programs which are used unmodified in performing those activities but -which are not part of the work. For example, Corresponding Source -includes interface definition files associated with source files for -the work, and the source code for shared libraries and dynamically -linked subprograms that the work is specifically designed to require, -such as by intimate data communication or control flow between those -subprograms and other parts of the work. - - The Corresponding Source need not include anything that users -can regenerate automatically from other parts of the Corresponding -Source. - - The Corresponding Source for a work in source code form is that -same work. - - 2. Basic Permissions. - - All rights granted under this License are granted for the term of -copyright on the Program, and are irrevocable provided the stated -conditions are met. This License explicitly affirms your unlimited -permission to run the unmodified Program. The output from running a -covered work is covered by this License only if the output, given its -content, constitutes a covered work. This License acknowledges your -rights of fair use or other equivalent, as provided by copyright law. - - You may make, run and propagate covered works that you do not -convey, without conditions so long as your license otherwise remains -in force. You may convey covered works to others for the sole purpose -of having them make modifications exclusively for you, or provide you -with facilities for running those works, provided that you comply with -the terms of this License in conveying all material for which you do -not control copyright. Those thus making or running the covered works -for you must do so exclusively on your behalf, under your direction -and control, on terms that prohibit them from making any copies of -your copyrighted material outside their relationship with you. - - Conveying under any other circumstances is permitted solely under -the conditions stated below. Sublicensing is not allowed; section 10 -makes it unnecessary. - - 3. Protecting Users' Legal Rights From Anti-Circumvention Law. - - No covered work shall be deemed part of an effective technological -measure under any applicable law fulfilling obligations under article -11 of the WIPO copyright treaty adopted on 20 December 1996, or -similar laws prohibiting or restricting circumvention of such -measures. - - When you convey a covered work, you waive any legal power to forbid -circumvention of technological measures to the extent such circumvention -is effected by exercising rights under this License with respect to -the covered work, and you disclaim any intention to limit operation or -modification of the work as a means of enforcing, against the work's -users, your or third parties' legal rights to forbid circumvention of -technological measures. - - 4. Conveying Verbatim Copies. - - You may convey verbatim copies of the Program's source code as you -receive it, in any medium, provided that you conspicuously and -appropriately publish on each copy an appropriate copyright notice; -keep intact all notices stating that this License and any -non-permissive terms added in accord with section 7 apply to the code; -keep intact all notices of the absence of any warranty; and give all -recipients a copy of this License along with the Program. - - You may charge any price or no price for each copy that you convey, -and you may offer support or warranty protection for a fee. - - 5. Conveying Modified Source Versions. - - You may convey a work based on the Program, or the modifications to -produce it from the Program, in the form of source code under the -terms of section 4, provided that you also meet all of these conditions: - - a) The work must carry prominent notices stating that you modified - it, and giving a relevant date. - - b) The work must carry prominent notices stating that it is - released under this License and any conditions added under section - 7. This requirement modifies the requirement in section 4 to - "keep intact all notices". - - c) You must license the entire work, as a whole, under this - License to anyone who comes into possession of a copy. This - License will therefore apply, along with any applicable section 7 - additional terms, to the whole of the work, and all its parts, - regardless of how they are packaged. This License gives no - permission to license the work in any other way, but it does not - invalidate such permission if you have separately received it. - - d) If the work has interactive user interfaces, each must display - Appropriate Legal Notices; however, if the Program has interactive - interfaces that do not display Appropriate Legal Notices, your - work need not make them do so. - - A compilation of a covered work with other separate and independent -works, which are not by their nature extensions of the covered work, -and which are not combined with it such as to form a larger program, -in or on a volume of a storage or distribution medium, is called an -"aggregate" if the compilation and its resulting copyright are not -used to limit the access or legal rights of the compilation's users -beyond what the individual works permit. Inclusion of a covered work -in an aggregate does not cause this License to apply to the other -parts of the aggregate. - - 6. Conveying Non-Source Forms. - - You may convey a covered work in object code form under the terms -of sections 4 and 5, provided that you also convey the -machine-readable Corresponding Source under the terms of this License, -in one of these ways: - - a) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by the - Corresponding Source fixed on a durable physical medium - customarily used for software interchange. - - b) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by a - written offer, valid for at least three years and valid for as - long as you offer spare parts or customer support for that product - model, to give anyone who possesses the object code either (1) a - copy of the Corresponding Source for all the software in the - product that is covered by this License, on a durable physical - medium customarily used for software interchange, for a price no - more than your reasonable cost of physically performing this - conveying of source, or (2) access to copy the - Corresponding Source from a network server at no charge. - - c) Convey individual copies of the object code with a copy of the - written offer to provide the Corresponding Source. This - alternative is allowed only occasionally and noncommercially, and - only if you received the object code with such an offer, in accord - with subsection 6b. - - d) Convey the object code by offering access from a designated - place (gratis or for a charge), and offer equivalent access to the - Corresponding Source in the same way through the same place at no - further charge. You need not require recipients to copy the - Corresponding Source along with the object code. If the place to - copy the object code is a network server, the Corresponding Source - may be on a different server (operated by you or a third party) - that supports equivalent copying facilities, provided you maintain - clear directions next to the object code saying where to find the - Corresponding Source. Regardless of what server hosts the - Corresponding Source, you remain obligated to ensure that it is - available for as long as needed to satisfy these requirements. - - e) Convey the object code using peer-to-peer transmission, provided - you inform other peers where the object code and Corresponding - Source of the work are being offered to the general public at no - charge under subsection 6d. - - A separable portion of the object code, whose source code is excluded -from the Corresponding Source as a System Library, need not be -included in conveying the object code work. - - A "User Product" is either (1) a "consumer product", which means any -tangible personal property which is normally used for personal, family, -or household purposes, or (2) anything designed or sold for incorporation -into a dwelling. In determining whether a product is a consumer product, -doubtful cases shall be resolved in favor of coverage. For a particular -product received by a particular user, "normally used" refers to a -typical or common use of that class of product, regardless of the status -of the particular user or of the way in which the particular user -actually uses, or expects or is expected to use, the product. A product -is a consumer product regardless of whether the product has substantial -commercial, industrial or non-consumer uses, unless such uses represent -the only significant mode of use of the product. - - "Installation Information" for a User Product means any methods, -procedures, authorization keys, or other information required to install -and execute modified versions of a covered work in that User Product from -a modified version of its Corresponding Source. The information must -suffice to ensure that the continued functioning of the modified object -code is in no case prevented or interfered with solely because -modification has been made. - - If you convey an object code work under this section in, or with, or -specifically for use in, a User Product, and the conveying occurs as -part of a transaction in which the right of possession and use of the -User Product is transferred to the recipient in perpetuity or for a -fixed term (regardless of how the transaction is characterized), the -Corresponding Source conveyed under this section must be accompanied -by the Installation Information. But this requirement does not apply -if neither you nor any third party retains the ability to install -modified object code on the User Product (for example, the work has -been installed in ROM). - - The requirement to provide Installation Information does not include a -requirement to continue to provide support service, warranty, or updates -for a work that has been modified or installed by the recipient, or for -the User Product in which it has been modified or installed. Access to a -network may be denied when the modification itself materially and -adversely affects the operation of the network or violates the rules and -protocols for communication across the network. - - Corresponding Source conveyed, and Installation Information provided, -in accord with this section must be in a format that is publicly -documented (and with an implementation available to the public in -source code form), and must require no special password or key for -unpacking, reading or copying. - - 7. Additional Terms. - - "Additional permissions" are terms that supplement the terms of this -License by making exceptions from one or more of its conditions. -Additional permissions that are applicable to the entire Program shall -be treated as though they were included in this License, to the extent -that they are valid under applicable law. If additional permissions -apply only to part of the Program, that part may be used separately -under those permissions, but the entire Program remains governed by -this License without regard to the additional permissions. - - When you convey a copy of a covered work, you may at your option -remove any additional permissions from that copy, or from any part of -it. (Additional permissions may be written to require their own -removal in certain cases when you modify the work.) You may place -additional permissions on material, added by you to a covered work, -for which you have or can give appropriate copyright permission. - - Notwithstanding any other provision of this License, for material you -add to a covered work, you may (if authorized by the copyright holders of -that material) supplement the terms of this License with terms: - - a) Disclaiming warranty or limiting liability differently from the - terms of sections 15 and 16 of this License; or - - b) Requiring preservation of specified reasonable legal notices or - author attributions in that material or in the Appropriate Legal - Notices displayed by works containing it; or - - c) Prohibiting misrepresentation of the origin of that material, or - requiring that modified versions of such material be marked in - reasonable ways as different from the original version; or - - d) Limiting the use for publicity purposes of names of licensors or - authors of the material; or - - e) Declining to grant rights under trademark law for use of some - trade names, trademarks, or service marks; or - - f) Requiring indemnification of licensors and authors of that - material by anyone who conveys the material (or modified versions of - it) with contractual assumptions of liability to the recipient, for - any liability that these contractual assumptions directly impose on - those licensors and authors. - - All other non-permissive additional terms are considered "further -restrictions" within the meaning of section 10. If the Program as you -received it, or any part of it, contains a notice stating that it is -governed by this License along with a term that is a further -restriction, you may remove that term. If a license document contains -a further restriction but permits relicensing or conveying under this -License, you may add to a covered work material governed by the terms -of that license document, provided that the further restriction does -not survive such relicensing or conveying. - - If you add terms to a covered work in accord with this section, you -must place, in the relevant source files, a statement of the -additional terms that apply to those files, or a notice indicating -where to find the applicable terms. - - Additional terms, permissive or non-permissive, may be stated in the -form of a separately written license, or stated as exceptions; -the above requirements apply either way. - - 8. Termination. - - You may not propagate or modify a covered work except as expressly -provided under this License. Any attempt otherwise to propagate or -modify it is void, and will automatically terminate your rights under -this License (including any patent licenses granted under the third -paragraph of section 11). - - However, if you cease all violation of this License, then your -license from a particular copyright holder is reinstated (a) -provisionally, unless and until the copyright holder explicitly and -finally terminates your license, and (b) permanently, if the copyright -holder fails to notify you of the violation by some reasonable means -prior to 60 days after the cessation. - - Moreover, your license from a particular copyright holder is -reinstated permanently if the copyright holder notifies you of the -violation by some reasonable means, this is the first time you have -received notice of violation of this License (for any work) from that -copyright holder, and you cure the violation prior to 30 days after -your receipt of the notice. - - Termination of your rights under this section does not terminate the -licenses of parties who have received copies or rights from you under -this License. If your rights have been terminated and not permanently -reinstated, you do not qualify to receive new licenses for the same -material under section 10. - - 9. Acceptance Not Required for Having Copies. - - You are not required to accept this License in order to receive or -run a copy of the Program. Ancillary propagation of a covered work -occurring solely as a consequence of using peer-to-peer transmission -to receive a copy likewise does not require acceptance. However, -nothing other than this License grants you permission to propagate or -modify any covered work. These actions infringe copyright if you do -not accept this License. Therefore, by modifying or propagating a -covered work, you indicate your acceptance of this License to do so. - - 10. Automatic Licensing of Downstream Recipients. - - Each time you convey a covered work, the recipient automatically -receives a license from the original licensors, to run, modify and -propagate that work, subject to this License. You are not responsible -for enforcing compliance by third parties with this License. - - An "entity transaction" is a transaction transferring control of an -organization, or substantially all assets of one, or subdividing an -organization, or merging organizations. If propagation of a covered -work results from an entity transaction, each party to that -transaction who receives a copy of the work also receives whatever -licenses to the work the party's predecessor in interest had or could -give under the previous paragraph, plus a right to possession of the -Corresponding Source of the work from the predecessor in interest, if -the predecessor has it or can get it with reasonable efforts. - - You may not impose any further restrictions on the exercise of the -rights granted or affirmed under this License. For example, you may -not impose a license fee, royalty, or other charge for exercise of -rights granted under this License, and you may not initiate litigation -(including a cross-claim or counterclaim in a lawsuit) alleging that -any patent claim is infringed by making, using, selling, offering for -sale, or importing the Program or any portion of it. - - 11. Patents. - - A "contributor" is a copyright holder who authorizes use under this -License of the Program or a work on which the Program is based. The -work thus licensed is called the contributor's "contributor version". - - A contributor's "essential patent claims" are all patent claims -owned or controlled by the contributor, whether already acquired or -hereafter acquired, that would be infringed by some manner, permitted -by this License, of making, using, or selling its contributor version, -but do not include claims that would be infringed only as a -consequence of further modification of the contributor version. For -purposes of this definition, "control" includes the right to grant -patent sublicenses in a manner consistent with the requirements of -this License. - - Each contributor grants you a non-exclusive, worldwide, royalty-free -patent license under the contributor's essential patent claims, to -make, use, sell, offer for sale, import and otherwise run, modify and -propagate the contents of its contributor version. - - In the following three paragraphs, a "patent license" is any express -agreement or commitment, however denominated, not to enforce a patent -(such as an express permission to practice a patent or covenant not to -sue for patent infringement). To "grant" such a patent license to a -party means to make such an agreement or commitment not to enforce a -patent against the party. - - If you convey a covered work, knowingly relying on a patent license, -and the Corresponding Source of the work is not available for anyone -to copy, free of charge and under the terms of this License, through a -publicly available network server or other readily accessible means, -then you must either (1) cause the Corresponding Source to be so -available, or (2) arrange to deprive yourself of the benefit of the -patent license for this particular work, or (3) arrange, in a manner -consistent with the requirements of this License, to extend the patent -license to downstream recipients. "Knowingly relying" means you have -actual knowledge that, but for the patent license, your conveying the -covered work in a country, or your recipient's use of the covered work -in a country, would infringe one or more identifiable patents in that -country that you have reason to believe are valid. - - If, pursuant to or in connection with a single transaction or -arrangement, you convey, or propagate by procuring conveyance of, a -covered work, and grant a patent license to some of the parties -receiving the covered work authorizing them to use, propagate, modify -or convey a specific copy of the covered work, then the patent license -you grant is automatically extended to all recipients of the covered -work and works based on it. - - A patent license is "discriminatory" if it does not include within -the scope of its coverage, prohibits the exercise of, or is -conditioned on the non-exercise of one or more of the rights that are -specifically granted under this License. You may not convey a covered -work if you are a party to an arrangement with a third party that is -in the business of distributing software, under which you make payment -to the third party based on the extent of your activity of conveying -the work, and under which the third party grants, to any of the -parties who would receive the covered work from you, a discriminatory -patent license (a) in connection with copies of the covered work -conveyed by you (or copies made from those copies), or (b) primarily -for and in connection with specific products or compilations that -contain the covered work, unless you entered into that arrangement, -or that patent license was granted, prior to 28 March 2007. - - Nothing in this License shall be construed as excluding or limiting -any implied license or other defenses to infringement that may -otherwise be available to you under applicable patent law. - - 12. No Surrender of Others' Freedom. - - If conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License. If you cannot convey a -covered work so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you may -not convey it at all. For example, if you agree to terms that obligate you -to collect a royalty for further conveying from those to whom you convey -the Program, the only way you could satisfy both those terms and this -License would be to refrain entirely from conveying the Program. - - 13. Use with the GNU Affero General Public License. - - Notwithstanding any other provision of this License, you have -permission to link or combine any covered work with a work licensed -under version 3 of the GNU Affero General Public License into a single -combined work, and to convey the resulting work. The terms of this -License will continue to apply to the part which is the covered work, -but the special requirements of the GNU Affero General Public License, -section 13, concerning interaction through a network will apply to the -combination as such. - - 14. Revised Versions of this License. - - The Free Software Foundation may publish revised and/or new versions of -the GNU General Public License from time to time. Such new versions will -be similar in spirit to the present version, but may differ in detail to -address new problems or concerns. - - Each version is given a distinguishing version number. If the -Program specifies that a certain numbered version of the GNU General -Public License "or any later version" applies to it, you have the -option of following the terms and conditions either of that numbered -version or of any later version published by the Free Software -Foundation. If the Program does not specify a version number of the -GNU General Public License, you may choose any version ever published -by the Free Software Foundation. - - If the Program specifies that a proxy can decide which future -versions of the GNU General Public License can be used, that proxy's -public statement of acceptance of a version permanently authorizes you -to choose that version for the Program. - - Later license versions may give you additional or different -permissions. However, no additional obligations are imposed on any -author or copyright holder as a result of your choosing to follow a -later version. - - 15. Disclaimer of Warranty. - - THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY -APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT -HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY -OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, -THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM -IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF -ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - - 16. Limitation of Liability. - - IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING -WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS -THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY -GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE -USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF -DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD -PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), -EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF -SUCH DAMAGES. - - 17. Interpretation of Sections 15 and 16. - - If the disclaimer of warranty and limitation of liability provided -above cannot be given local legal effect according to their terms, -reviewing courts shall apply local law that most closely approximates -an absolute waiver of all civil liability in connection with the -Program, unless a warranty or assumption of liability accompanies a -copy of the Program in return for a fee. - - END OF TERMS AND CONDITIONS - - How to Apply These Terms to Your New Programs - - If you develop a new program, and you want it to be of the greatest -possible use to the public, the best way to achieve this is to make it -free software which everyone can redistribute and change under these terms. - - To do so, attach the following notices to the program. It is safest -to attach them to the start of each source file to most effectively -state the exclusion of warranty; and each file should have at least -the "copyright" line and a pointer to where the full notice is found. - - <one line to give the program's name and a brief idea of what it does.> - Copyright (C) <year> <name of author> - - This program is free software: you can redistribute it and/or modify - it under the terms of the GNU General Public License as published by - the Free Software Foundation, either version 3 of the License, or - (at your option) any later version. - - This program is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU General Public License for more details. - - You should have received a copy of the GNU General Public License - along with this program. If not, see <http://www.gnu.org/licenses/>. - -Also add information on how to contact you by electronic and paper mail. - - If the program does terminal interaction, make it output a short -notice like this when it starts in an interactive mode: - - <program> Copyright (C) <year> <name of author> - This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. - This is free software, and you are welcome to redistribute it - under certain conditions; type `show c' for details. - -The hypothetical commands `show w' and `show c' should show the appropriate -parts of the General Public License. Of course, your program's commands -might be different; for a GUI interface, you would use an "about box". - - You should also get your employer (if you work as a programmer) or school, -if any, to sign a "copyright disclaimer" for the program, if necessary. -For more information on this, and how to apply and follow the GNU GPL, see -<http://www.gnu.org/licenses/>. - - The GNU General Public License does not permit incorporating your program -into proprietary programs. If your program is a subroutine library, you -may consider it more useful to permit linking proprietary applications with -the library. If this is what you want to do, use the GNU Lesser General -Public License instead of this License. But first, please read -<http://www.gnu.org/philosophy/why-not-lgpl.html>. -- -
vignettes/other-langs.Rmd
- other-langs.Rmd
purrr draws inspiration from many related tools:
-List operations defined in the Haskell prelude
Scala’s list methods.
Functional programming librarys for javascript: underscore.js, lodash and lazy.js.
rlist, another R package to support working with lists. Similar goals but somewhat different philosophy.
However, the goal of purrr is not to try and simulate a purer functional programming language in R; we don’t want to implement a second-class version of Haskell in R. The goal is to give you similar expressiveness to an FP language, while allowing you to write code that looks and works like R:
-Instead of point free (tacit) style, we use the pipe, %>%
, to write code that can be read from left to right.
Instead of currying, we use ...
to pass in extra arguments.
Anonymous functions are verbose in R, so we provide two convenient shorthands. For unary functions, ~ .x + 1
is equivalent to function(.x) .x + 1
. For chains of transformations functions, . %>% f() %>% g()
is equivalent to function(.) . %>% f() %>% g()
(this shortcut is provided by magrittr).
R is weakly typed, so we need map
variants that describe the output type (like map_int()
, map_dbl()
, etc) because we don’t know the return type of .f
.
R has named arguments, so instead of providing different functions for minor variations (e.g. detect()
and detectLast()
) we use a named argument, .right
. Type-stable functions are easy to reason about so additional arguments will never change the type of the output.
purrr enhances R’s functional programming (FP) toolkit by providing a complete and consistent set of tools for working with functions and vectors. If you’ve never heard of FP before, the best place to start is the family of map()
functions which allow you to replace many for loops with code that is both more succinct and easier to read. The best place to learn about the map()
functions is the iteration chapter in R for data science.
The following example uses purrr to solve a fairly realistic problem: split a data frame into pieces, fit a model to each piece, compute the summary, then extract the R2.
-library(purrr)
-
-mtcars %>%
- split(.$cyl) %>% # from base R
- map(~ lm(mpg ~ wt, data = .)) %>%
- map(summary) %>%
- map_dbl("r.squared")
-#> 4 6 8
-#> 0.5086326 0.4645102 0.4229655
This example illustrates some of the advantages of purrr functions over the equivalents in base R:
-The first argument is always the data, so purrr works naturally with the pipe.
All purrr functions are type-stable. They always return the advertised output type (map()
returns lists; map_dbl()
returns double vectors), or they throw an errror.
All map()
functions either accept function, formulas (used for succinctly generating anonymous functions), a character vector (used to extract components by name), or a numeric vector (used to extract by position).
Please note that this project is released with a Contributor Code of Conduct. By participating in this project you agree to abide by its terms.
-NEWS.md
- accumulate()
and accumulate_right()
now inherit names from their first input (@AshesITR, #446)
attr_getter()
no longer uses partial matching. For example, if an x
object has a labels
attribute but no label
attribute, attr_getter("label")(x)
will no longer extract the labels
attribute (#460, @huftis).
flatten_dfr()
and flatten_dfc()
now aborts if dplyr is not installed. (#454)
imap_dfr()
now works with .id
argument is provided (#429)
map_at()
, modify_at()
and lmap_at()
accept negative values for .at
, ignoring elements at those positions.
map()
and modify()
now work with calls and pairlists (#412).
modify()
, modify_if()
and modify_at()
now preserve the class of atomic vectors instead of promoting them to lists. New S3 methods are provided for character, logical, double, and integer classes (@t-kalinowski, #417).
list_modify()
, update_list()
and list_merge()
now handle duplicate duplicate argument names correctly (#441, @mgirlich).
map_raw
, imap_raw
, flatten_raw
, invoke_map_raw
, map2_raw
and pmap_raw
added to support raw vectors. (#455, @romainfrancois)
We noticed the following issues during reverse dependencies checks:
-If reduce()
fails with this message: Error: `.x` is empty, and no `.init` supplied
, this is because reduce()
now returns .init
when .x
is empty. Fix the problem by supplying an appropriate argument to .init
, or by providing special behaviour when .x
has length 0.
The type predicates have been migrated to rlang. Consequently the bare-type-predicates
documentation topic is no longer in purrr, which might cause a warning if you cross-reference it.
purrr no longer depends on lazyeval or Rcpp (or dplyr, as of the previous version). This makes the dependency graph of the tidyverse simpler, and makes purrr more suitable as a dependency of lower-level packages.
-There have also been two changes to eliminate name conflicts between purrr and dplyr:
-order_by()
, sort_by()
and split_by()
have been removed. order_by()
conflicted with dplyr::order_by()
and the complete family doesn’t feel that useful. Use tibbles instead (#217).
contains()
has been renamed to has_element()
to avoid conflicts with dplyr (#217).
The plucking mechanism used for indexing into data structures with map()
has been extracted into the function pluck()
. Plucking is often more readable to extract an element buried in a deep data structure. Compare this syntax-heavy extraction which reads non-linearly:
accessor(x[[1]])$foo
-to the equivalent pluck:
-x %>% pluck(1, accessor, "foo")
-as_function()
is now as_mapper()
because it is a tranformation that makes sense primarily for mapping functions, not in general (#298). .null
has been renamed to .default
to better reflect its intent (#298). .default
is returned whenever an element is absent or empty (#231, #254).
as_mapper()
sanitises primitive functions by transforming them to closures with standardised argument names (using rlang::as_closure()
). For instance +
is transformed to function(.x, .y) .x + .y
. This results in proper argument matching so that map(1:10, partial(
-, .x = 5))
produces list(5 - 1, 5 - 2, ...)
.
Recursive indexing can now extract objects out of environments (#213) and S4 objects (#200), as well as lists.
attr_getter()
makes it possible to extract from attributes like map(list(iris, mtcars), attr_getter("row.names"))
.
The argument list for formula-functions has been tweaked so that you can refer to arguments by position with ..1
, ..2
, and so on. This makes it possible to use the formula shorthand for functions with more than two arguments (#289).
possibly()
, safely()
and friends no longer capture interrupts: this means that you can now terminate a mapper using one of these with Escape or Ctrl + C (#314)
All map functions now treat NULL
the same way as an empty vector (#199), and return an empty vector if any input is an empty vector.
All map()
functions now force their arguments in the same way that base R does for lapply()
(#191). This makes map()
etc easier to use when generating functions.
A new family of “indexed” map functions, imap()
, imap_lgl()
etc, provide a short-hand for map2(x, names(x))
or map2(x, seq_along(x))
(#240).
The data frame suffix _df
has been (soft) deprecated in favour of _dfr
to more clearly indicate that it’s a row-bind. All variants now also have a _dfc
for column binding (#167). (These will not be terribly useful until dplyr::bind_rows()
/dplyr::bind_cols()
have better semantics for vectors.)
A new modify()
family returns the same output of the type as the input .x
. This is in contrast to the map()
family which always returns a list, regardless of the input type.
The modify functions are S3 generics. However their default methods should be sufficient for most classes since they rely on the semantics of [<-
. modify.default()
is thus a shorthand for x[] <- map(x, f)
.
at_depth()
has been renamed to modify_depth()
.
modify_depth()
gains new .ragged
argument, and negative depths are now computed relative to the deepest component of the list (#236).
auto_browse(f)
returns a new function that automatically calls browser()
if f
throws an error (#281).
vec_depth()
computes the depth (i.e. the number of levels of indexing) or a vector (#243).
reduce2()
and reduce2_right()
make it possible to reduce with a 3 argument function where the first argument is the accumulated value, the second argument is .x
, and the third argument is .y
(#163).
list_modify()
extends stats::modifyList()
to replace by position if the list is not named.(#201). list_merge()
operates similarly to list_modify()
but combines instead of replacing (#322).
The legacy function update_list()
is basically a version of list_modify
that evaluates formulas within the list. It is likely to be deprecated in the future in favour of a tidyeval interface such as a list method for dplyr::mutate()
.
Thanks to @dchiu911, the unit test coverage of purrr is now much greater.
All predicate functions are re-exported from rlang (#124).
compact()
now works with standard mapper conventions (#282).
cross_n()
has been renamed to cross()
. The _n
suffix was removed for consistency with pmap()
(originally called map_n()
at the start of the project) and transpose()
(originally called zip_n()
). Similarly, cross_d()
has been renamed to cross_df()
for consistency with map_df()
.
every()
and some()
now return NA
if present in the input (#174).
invoke()
uses a more robust approach to generate the argument list (#249) It no longer uses lazyeval to figure out which enviroment a character f
comes from.
is_numeric()
and is_scalar_numeric()
are deprecated because they don’t test for what you might expect at first sight.
reduce()
now throws an error if .x
is empty and .init
is not supplied.
Deprecated functions flatmap()
, map3()
, map_n()
, walk3()
, walk_n()
, zip2()
, zip3()
, zip_n()
have been removed.
pmap()
coerces data frames to lists to avoid the expensive [.data.frame
which provides security that is unneeded here (#220).
set_names()
can now take a function to tranform the names programmatically (#276), and you can supply names in ...
to reduce typing even more more (#316). set_names()
is now powered by rlang::set_names()
.
transpose()
now matches by name if available (#164). You can override the default choice with the new .names
argument.
The function argument of detect()
and detect_index()
have been renamed from .p
to .f
. This is because they have mapper semantics rather than predicate semantics.
This is a compatibility release with dplyr 0.6.0.
-dmap()
, dmap_at()
, dmap_if()
, invoke_rows()
, slice_rows()
, map_rows()
, by_slice()
, by_row()
, and unslice()
have been moved to purrrlyr. This is a bit of an aggresive change but it allows us to make the dependencies much lighter.Fix for dev tibble support.
as_function()
now supports list arguments which allow recursive indexing using either names or positions. They now always stop when encountering the first NULL (#173).
accumulate
and reduce
correctly pass extra arguments to the worker function.
as_function()
gains a .null
argument that for character and numeric values allows you to specify what to return for null/absent elements (#110). This can be used with any map function, e.g. map_int(x, 1, .null = NA)
as_function()
is now generic.
New is_function()
that returns TRUE
only for regular functions.
Fix crash on GCC triggered by invoke_rows()
.
There are two handy infix functions:
- -accumulate()
has been added to handle recursive folding. It is shortand for Reduce(f, .x, accumulate = TRUE)
and follows a similar syntax to reduce()
(#145). A right-hand version accumulate_right()
was also added.
map_df()
row-binds output together. It’s the equivalent of plyr::ldply()
(#127)
flatten()
is now type-stable and always returns a list. To return a simpler vector, use flatten_lgl()
, flatten_int()
, flatten_dbl()
, flatten_chr()
, or flatten_df()
.
invoke()
has been overhauled to be more useful: it now works similarly to map_call()
when .x
is NULL, and hence map_call()
has been deprecated. invoke_map()
is a vectorised complement to invoke()
(#125), and comes with typed variants invoke_map_lgl()
, invoke_map_int()
, invoke_map_dbl()
, invoke_map_chr()
, and invoke_map_df()
.
transpose()
replaces zip2()
, zip3()
, and zip_n()
(#128). The name more clearly reflects the intent (transposing the first and second levels of list). It no longer has fields argument or the .simplify
argument; instead use the new simplify_all()
function.
safely()
, quietly()
, and possibly()
are experimental functions for working with functions with side-effects (e.g. printed output, messages, warnings, and errors) (#120). safely()
is a version of try()
that modifies a function (rather than an expression), and always returns a list with two components, result
and error
.
list_along()
and rep_along()
generalise the idea of seq_along()
. (#122).
is_null()
is the snake-case version of is.null()
.
pmap()
(parallel map) replaces map_n()
(#132), and has typed-variants suffixed pmap_lgl()
, pmap_int()
, pmap_dbl()
, pmap_chr()
, and pmap_df()
.
set_names()
is a snake-case alternative to setNames()
with stricter equality checking, and more convenient defaults for pipes: x %>% set_names()
is equivalent to setNames(x, x)
(#119).
We are still figuring out what belongs in dplyr and what belongs in purrr. Expect much experimentation and many changes with these functions.
-map()
now always returns a list. Data frame support has been moved to map_df()
and dmap()
. The latter supports sliced data frames as a shortcut for the combination of by_slice()
and dmap()
: x %>% by_slice(dmap, fun, .collate = "rows")
. The conditional variants dmap_at()
and dmap_if()
also support sliced data frames and will recycle scalar results to the slice size.
map_rows()
has been renamed to invoke_rows()
. As other rows-based functionals, it collates results inside lists by default, but with column collation this function is equivalent to plyr::mdply()
.
The rows-based functionals gain a .to
option to name the output column as well as a .collate
argument. The latter allows to collate the output in lists (by default), on columns or on rows. This makes these functions more flexible and more predictable.
as_function()
, which converts formulas etc to functions, is now exported (#123).
update_list()
can now modify an element called x
(#98).
map*()
now use custom C code, rather than relying on lapply()
, mapply()
etc. The performance characteristcs are very similar, but it allows us greater control over the output (#118).
accumulate
applies a function recursively over a list from the left, while
-accumulate_right
applies the function from the right. Unlike reduce
-both functions keep the intermediate results.
accumulate(.x, .f, ..., .init) - -accumulate_right(.x, .f, ..., .init)- -
.x | -A list or atomic vector. |
-
---|---|
.f | -For For |
-
... | -Additional arguments passed on to |
-
.init | -If supplied, will be used as the first value to start
-the accumulation, rather than using |
-
A vector the same length of .x
with the same names as .x
-1:3 %>% accumulate(`+`)#> [1] 1 3 61:10 %>% accumulate_right(`*`)#> [1] 3628800 3628800 1814400 604800 151200 30240 5040 720 90 -#> [10] 10-# From Haskell's scanl documentation -1:10 %>% accumulate(max, .init = 5)#> [1] 5 5 5 5 5 5 6 7 8 9 10-# Understanding the arguments .x and .y when .f -# is a lambda function -# .x is the accumulating value -1:10 %>% accumulate(~ .x)#> [1] 1 1 1 1 1 1 1 1 1 1# .y is element in the list -1:10 %>% accumulate(~ .y)#> [1] 1 2 3 4 5 6 7 8 9 10-# Simulating stochastic processes with drift -# NOT RUN { -library(dplyr) -library(ggplot2) - -rerun(5, rnorm(100)) %>% - set_names(paste0("sim", 1:5)) %>% - map(~ accumulate(., ~ .05 + .x + .y)) %>% - map_dfr(~ data_frame(value = .x, step = 1:100), .id = "simulation") %>% - ggplot(aes(x = step, y = value)) + - geom_line(aes(color = simulation)) + - ggtitle("Simulations of a random walk with drift") -# }
These functions take the idea of seq_along()
and generalise
-it to creating lists (list_along
) and repeating values
-(rep_along
).
list_along(x) - -rep_along(x, y)- -
x | -A vector. |
-
---|---|
y | -Values to repeat. |
-
A vector of the same length as x
.
-x <- 1:5 -rep_along(x, 1:2)#> [1] 1 2 1 2 1rep_along(x, 1)#> [1] 1 1 1 1 1list_along(x)#> [[1]] -#> NULL -#> -#> [[2]] -#> NULL -#> -#> [[3]] -#> NULL -#> -#> [[4]] -#> NULL -#> -#> [[5]] -#> NULL -#>
array_branch()
and array_tree()
enable arrays to be
-used with purrr's functionals by turning them into lists. The
-details of the coercion are controlled by the margin
-argument. array_tree()
creates an hierarchical list (a tree)
-that has as many levels as dimensions specified in margin
,
-while array_branch()
creates a flat list (by analogy, a
-branch) along all mentioned dimensions.
array_branch(array, margin = NULL) - -array_tree(array, margin = NULL)- -
array | -An array to coerce into a list. |
-
---|---|
margin | -A numeric vector indicating the positions of the
-indices to be to be enlisted. If |
-
When no margin is specified, all dimensions are used by
-default. When margin
is a numeric vector of length zero, the
-whole array is wrapped in a list.
-# We create an array with 3 dimensions -x <- array(1:12, c(2, 2, 3)) - -# A full margin for such an array would be the vector 1:3. This is -# the default if you don't specify a margin - -# Creating a branch along the full margin is equivalent to -# as.list(array) and produces a list of size length(x): -array_branch(x) %>% str()#> List of 12 -#> $ : int 1 -#> $ : int 2 -#> $ : int 3 -#> $ : int 4 -#> $ : int 5 -#> $ : int 6 -#> $ : int 7 -#> $ : int 8 -#> $ : int 9 -#> $ : int 10 -#> $ : int 11 -#> $ : int 12-# A branch along the first dimension yields a list of length 2 -# with each element containing a 2x3 array: -array_branch(x, 1) %>% str()#> List of 2 -#> $ : int [1:2, 1:3] 1 3 5 7 9 11 -#> $ : int [1:2, 1:3] 2 4 6 8 10 12-# A branch along the first and third dimensions yields a list of -# length 2x3 whose elements contain a vector of length 2: -array_branch(x, c(1, 3)) %>% str()#> List of 6 -#> $ : int [1:2] 1 3 -#> $ : int [1:2] 2 4 -#> $ : int [1:2] 5 7 -#> $ : int [1:2] 6 8 -#> $ : int [1:2] 9 11 -#> $ : int [1:2] 10 12-# Creating a tree from the full margin creates a list of lists of -# lists: -array_tree(x) %>% str()#> List of 2 -#> $ :List of 2 -#> ..$ :List of 3 -#> .. ..$ : int 1 -#> .. ..$ : int 5 -#> .. ..$ : int 9 -#> ..$ :List of 3 -#> .. ..$ : int 3 -#> .. ..$ : int 7 -#> .. ..$ : int 11 -#> $ :List of 2 -#> ..$ :List of 3 -#> .. ..$ : int 2 -#> .. ..$ : int 6 -#> .. ..$ : int 10 -#> ..$ :List of 3 -#> .. ..$ : int 4 -#> .. ..$ : int 8 -#> .. ..$ : int 12-# The ordering and the depth of the tree are controlled by the -# margin argument: -array_tree(x, c(3, 1)) %>% str()#> List of 3 -#> $ :List of 2 -#> ..$ : int [1:2] 1 3 -#> ..$ : int [1:2] 2 4 -#> $ :List of 2 -#> ..$ : int [1:2] 5 7 -#> ..$ : int [1:2] 6 8 -#> $ :List of 2 -#> ..$ : int [1:2] 9 11 -#> ..$ : int [1:2] 10 12
as_mapper
is the powerhouse behind the varied function
-specifications that most purrr functions allow. It is an S3
-generic. The default method forwards its arguments to
-rlang::as_function()
.
as_mapper(.f, ...) - -# S3 method for character -as_mapper(.f, ..., .null, .default = NULL) - -# S3 method for numeric -as_mapper(.f, ..., .null, .default = NULL) - -# S3 method for list -as_mapper(.f, ..., .null, .default = NULL)- -
.f | -A function, formula, or atomic vector. -If a function, it is used as is. -If a formula, e.g.
This syntax allows you to create very compact anonymous functions. -If character vector, numeric vector, or list, it
-is converted to an extractor function. Character vectors index by name
-and numeric vectors index by position; use a list to index by position
-and name at different levels. Within a list, wrap strings in |
-
---|---|
... | -Additional arguments passed on to methods. |
-
.default, .null | -Optional additional argument for extractor functions
-(i.e. when |
-
-as_mapper(~ . + 1)#> function (..., .x = ..1, .y = ..2, . = ..1) -#> . + 1 -#> <environment: 0x7f9d525cf5c8>as_mapper(1)#> function (x, ...) -#> pluck(x, list(1), .default = NULL) -#> <environment: 0x7f9d52109cd0>-as_mapper(c("a", "b", "c"))#> function (x, ...) -#> pluck(x, list("a", "b", "c"), .default = NULL) -#> <environment: 0x7f9d55aa0698># Equivalent to function(x) x[["a"]][["b"]][["c"]] - -as_mapper(list(1, "a", 2))#> function (x, ...) -#> pluck(x, list(1, "a", 2), .default = NULL) -#> <environment: 0x7f9d559eb990>#> function (x, ...) -#> pluck(x, list(1, function (x) -#> attr(x, attr, exact = TRUE)), .default = NULL) -#> <environment: 0x7f9d5384ef90># Equivalent to function(x) attr(x[[1]], "a") - -as_mapper(c("a", "b", "c"), .null = NA)#> function (x, ...) -#> pluck(x, list("a", "b", "c"), .default = NA) -#> <environment: 0x7f9d532a7d80>
as_vector()
collapses a list of vectors into one vector. It
-checks that the type of each vector is consistent with
-.type
. If the list can not be simplified, it throws an error.
-simplify
will simplify a vector if possible; simplify_all
-will apply simplify
to every element of a list.
as_vector(.x, .type = NULL) - -simplify(.x, .type = NULL) - -simplify_all(.x, .type = NULL)- -
.x | -A list of vectors |
-
---|---|
.type | -A vector mold or a string describing the type of the
-input vectors. The latter can be any of the types returned by
- |
-
.type
can be a vector mold specifying both the type and the
-length of the vectors to be concatenated, such as numeric(1)
-or integer(4)
. Alternatively, it can be a string describing
-the type, one of: "logical", "integer", "double", "complex",
-"character" or "raw".
-# Supply the type either with a string: -as.list(letters) %>% as_vector("character")#> [1] "a" "b" "c" "d" "e" "f" "g" "h" "i" "j" "k" "l" "m" "n" "o" "p" "q" "r" "s" -#> [20] "t" "u" "v" "w" "x" "y" "z"-# Or with a vector mold: -as.list(letters) %>% as_vector(character(1))#> [1] "a" "b" "c" "d" "e" "f" "g" "h" "i" "j" "k" "l" "m" "n" "o" "p" "q" "r" "s" -#> [20] "t" "u" "v" "w" "x" "y" "z"-# Vector molds are more flexible because they also specify the -# length of the concatenated vectors: -list(1:2, 3:4, 5:6) %>% as_vector(integer(2))#> [1] 1 2 3 4 5 6-# Note that unlike vapply(), as_vector() never adds dimension -# attributes. So when you specify a vector mold of size > 1, you -# always get a vector and not a matrix -
Compose multiple functions
- -compose(...)- -
... | -n functions to apply in order from right to left. |
-
---|
A function
- - --not_null <- compose(`!`, is.null) -not_null(4)#> [1] TRUEnot_null(NULL)#> [1] FALSE-add1 <- function(x) x + 1 -compose(add1, add1)(8)#> [1] 10
cross2()
returns the product set of the elements of
-.x
and .y
. cross3()
takes an additional
-.z
argument. cross()
takes a list .l
and
-returns the cartesian product of all its elements in a list, with
-one combination by element. cross_df()
is like
-cross()
but returns a data frame, with one combination by
-row.
cross(.l, .filter = NULL) - -cross2(.x, .y, .filter = NULL) - -cross3(.x, .y, .z, .filter = NULL) - -cross_df(.l, .filter = NULL)- -
.l | -A list of lists or atomic vectors. Alternatively, a data
-frame. |
-
---|---|
.filter | -A predicate function that takes the same number of -arguments as the number of variables to be combined. |
-
.x, .y, .z | -Lists or atomic vectors. |
-
cross2()
, cross3()
and cross()
-always return a list. cross_df()
always returns a data
-frame. cross()
returns a list where each element is one
-combination so that the list can be directly mapped
-over. cross_df()
returns a data frame where each row is one
-combination.
cross()
, cross2()
and cross3()
return the
-cartesian product is returned in wide format. This makes it more
-amenable to mapping operations. cross_df()
returns the output
-in long format just as expand.grid()
does. This is adapted
-to rowwise operations.
When the number of combinations is large and the individual
-elements are heavy memory-wise, it is often useful to filter
-unwanted combinations on the fly with .filter
. It must be
-a predicate function that takes the same number of arguments as the
-number of crossed objects (2 for cross2()
, 3 for
-cross3()
, length(.l)
for cross()
) and
-returns TRUE
or FALSE
. The combinations where the
-predicate function returns TRUE
will be removed from the
-result.
expand.grid()
-# We build all combinations of names, greetings and separators from our -# list of data and pass each one to paste() -data <- list( - id = c("John", "Jane"), - greeting = c("Hello.", "Bonjour."), - sep = c("! ", "... ") -) - -data %>% - cross() %>% - map(lift(paste))#> [[1]] -#> [1] "John! Hello." -#> -#> [[2]] -#> [1] "Jane! Hello." -#> -#> [[3]] -#> [1] "John! Bonjour." -#> -#> [[4]] -#> [1] "Jane! Bonjour." -#> -#> [[5]] -#> [1] "John... Hello." -#> -#> [[6]] -#> [1] "Jane... Hello." -#> -#> [[7]] -#> [1] "John... Bonjour." -#> -#> [[8]] -#> [1] "Jane... Bonjour." -#>-# cross() returns the combinations in long format: many elements, -# each representing one combination. With cross_df() we'll get a -# data frame in long format: crossing three objects produces a data -# frame of three columns with each row being a particular -# combination. This is the same format that expand.grid() returns. -args <- data %>% cross_df() - -# In case you need a list in long format (and not a data frame) -# just run as.list() after cross_df() -args %>% as.list()#> $id -#> [1] "John" "Jane" "John" "Jane" "John" "Jane" "John" "Jane" -#> -#> $greeting -#> [1] "Hello." "Hello." "Bonjour." "Bonjour." "Hello." "Hello." "Bonjour." -#> [8] "Bonjour." -#> -#> $sep -#> [1] "! " "! " "! " "! " "... " "... " "... " "... " -#>-# This format is often less pratical for functional programming -# because applying a function to the combinations requires a loop -out <- vector("list", length = nrow(args)) -for (i in seq_along(out)) - out[[i]] <- map(args, i) %>% invoke(paste, .) -out#> [[1]] -#> [1] "John! Hello." -#> -#> [[2]] -#> [1] "Jane! Hello." -#> -#> [[3]] -#> [1] "John! Bonjour." -#> -#> [[4]] -#> [1] "Jane! Bonjour." -#> -#> [[5]] -#> [1] "John... Hello." -#> -#> [[6]] -#> [1] "Jane... Hello." -#> -#> [[7]] -#> [1] "John... Bonjour." -#> -#> [[8]] -#> [1] "Jane... Bonjour." -#>-# It's easier to transpose and then use invoke_map() -args %>% transpose() %>% map_chr(~ invoke(paste, .))#> [1] "John! Hello." "Jane! Hello." "John! Bonjour." "Jane! Bonjour." -#> [5] "John... Hello." "Jane... Hello." "John... Bonjour." "Jane... Bonjour."-# Unwanted combinations can be filtered out with a predicate function -filter <- function(x, y) x >= y -cross2(1:5, 1:5, .filter = filter) %>% str()#> List of 10 -#> $ :List of 2 -#> ..$ : int 1 -#> ..$ : int 2 -#> $ :List of 2 -#> ..$ : int 1 -#> ..$ : int 3 -#> $ :List of 2 -#> ..$ : int 2 -#> ..$ : int 3 -#> $ :List of 2 -#> ..$ : int 1 -#> ..$ : int 4 -#> $ :List of 2 -#> ..$ : int 2 -#> ..$ : int 4 -#> $ :List of 2 -#> ..$ : int 3 -#> ..$ : int 4 -#> $ :List of 2 -#> ..$ : int 1 -#> ..$ : int 5 -#> $ :List of 2 -#> ..$ : int 2 -#> ..$ : int 5 -#> $ :List of 2 -#> ..$ : int 3 -#> ..$ : int 5 -#> $ :List of 2 -#> ..$ : int 4 -#> ..$ : int 5-# To give names to the components of the combinations, we map -# setNames() on the product: -seq_len(3) %>% - cross2(., ., .filter = `==`) %>% - map(setNames, c("x", "y"))#> [[1]] -#> [[1]]$x -#> [1] 2 -#> -#> [[1]]$y -#> [1] 1 -#> -#> -#> [[2]] -#> [[2]]$x -#> [1] 3 -#> -#> [[2]]$y -#> [1] 1 -#> -#> -#> [[3]] -#> [[3]]$x -#> [1] 1 -#> -#> [[3]]$y -#> [1] 2 -#> -#> -#> [[4]] -#> [[4]]$x -#> [1] 3 -#> -#> [[4]]$y -#> [1] 2 -#> -#> -#> [[5]] -#> [[5]]$x -#> [1] 1 -#> -#> [[5]]$y -#> [1] 3 -#> -#> -#> [[6]] -#> [[6]]$x -#> [1] 2 -#> -#> [[6]]$y -#> [1] 3 -#> -#>-# Alternatively we can encapsulate the arguments in a named list -# before crossing to get named components: -seq_len(3) %>% - list(x = ., y = .) %>% - cross(.filter = `==`)#> [[1]] -#> [[1]]$x -#> [1] 2 -#> -#> [[1]]$y -#> [1] 1 -#> -#> -#> [[2]] -#> [[2]]$x -#> [1] 3 -#> -#> [[2]]$y -#> [1] 1 -#> -#> -#> [[3]] -#> [[3]]$x -#> [1] 1 -#> -#> [[3]]$y -#> [1] 2 -#> -#> -#> [[4]] -#> [[4]]$x -#> [1] 3 -#> -#> [[4]]$y -#> [1] 2 -#> -#> -#> [[5]] -#> [[5]]$x -#> [1] 1 -#> -#> [[5]]$y -#> [1] 3 -#> -#> -#> [[6]] -#> [[6]]$x -#> [1] 2 -#> -#> [[6]]$y -#> [1] 3 -#> -#>
The depth of a vector is basically how many levels that you can index -into it.
- - -depth(x)- -
x | -A vector |
-
---|
An integer.
- - --x <- list( - list(), - list(list()), - list(list(list(1))) -) -depth(x)#> [1] 5#> [1] 1 2 4
Find the value or position of the first match.
- -detect(.x, .f, ..., .right = FALSE, .p) - -detect_index(.x, .f, ..., .right = FALSE, .p)- -
.x | -A list or atomic vector. |
-
---|---|
.f | -A function, formula, or atomic vector. -If a function, it is used as is. -If a formula, e.g.
This syntax allows you to create very compact anonymous functions. -If character vector, numeric vector, or list, it
-is converted to an extractor function. Character vectors index by name
-and numeric vectors index by position; use a list to index by position
-and name at different levels. Within a list, wrap strings in |
-
... | -Additional arguments passed on to |
-
.right | -If |
-
.p | -A single predicate function, a formula describing such a
-predicate function, or a logical vector of the same length as |
-
detect
the value of the first item that matches the
-predicate; detect_index
the position of the matching item.
-If not found, detect
returns NULL
and detect_index
-returns 0.
-is_even <- function(x) x %% 2 == 0 - -3:10 %>% detect(is_even)#> [1] 43:10 %>% detect_index(is_even)#> [1] 2-3:10 %>% detect(is_even, .right = TRUE)#> [1] 103:10 %>% detect_index(is_even, .right = TRUE)#> [1] 8- -# Since `.f` is passed to as_mapper(), you can supply a -# lambda-formula or a pluck object: -x <- list( - list(1, foo = FALSE), - list(2, foo = TRUE), - list(3, foo = TRUE) -) - -detect(x, "foo")#> [[1]] -#> [1] 2 -#> -#> $foo -#> [1] TRUE -#>detect_index(x, "foo")#> [1] 2
Do every or some elements of a list satisfy a predicate?
- -every(.x, .p, ...) - -some(.x, .p, ...)- -
.x | -A list or atomic vector. |
-
---|---|
.p | -A single predicate function, a formula describing such a
-predicate function, or a logical vector of the same length as |
-
... | -Additional arguments passed on to |
-
A logical vector of length 1.
- - --x <- list(0, 1, TRUE) -x %>% every(identity)#> [1] TRUEx %>% some(identity)#> [1] TRUE-y <- list(0:10, 5.5) -y %>% every(is.numeric)#> [1] TRUEy %>% every(is.integer)#> [1] FALSE
Extract an element from a vector or environment
- - -extract(x, index, default = NULL)- -
x | -A vector or environment |
-
---|---|
index | -A list indexing into the object |
-
default | -Value to use if target is empty or absent. |
-
These functions remove a level hierarchy from a list. They are similar to
-unlist()
, but they only ever remove a single layer of hierarchy and they
-are type-stable, so you always know what the type of the output is.
flatten(.x) - -flatten_lgl(.x) - -flatten_int(.x) - -flatten_dbl(.x) - -flatten_chr(.x) - -flatten_raw(.x) - -flatten_dfr(.x, .id = NULL) - -flatten_dfc(.x)- -
.x | -A list of flatten. The contents of the list can be anything for
- |
-
---|---|
.id | -Either a string or Only applies to |
-
flatten()
returns a list, flatten_lgl()
a logical
-vector, flatten_int()
an integer vector, flatten_dbl()
a
-double vector, and flatten_chr()
a character vector.
flatten_dfr()
and flatten_dfc()
return data frames created by
-row-binding and column-binding respectively. They require dplyr to
-be installed.
-#> [[1]] -#> [1] 1 3 2 4 -#> -#> [[2]] -#> [1] 1 2 4 3 -#>x %>% flatten()#> [[1]] -#> [1] 1 -#> -#> [[2]] -#> [1] 3 -#> -#> [[3]] -#> [1] 2 -#> -#> [[4]] -#> [1] 4 -#> -#> [[5]] -#> [1] 1 -#> -#> [[6]] -#> [1] 2 -#> -#> [[7]] -#> [1] 4 -#> -#> [[8]] -#> [1] 3 -#>x %>% flatten_int()#> [1] 1 3 2 4 1 2 4 3#> [1] 1 1#> [1] 1 1
Find head/tail that all satisfies a predicate.
- -head_while(.x, .p, ...) - -tail_while(.x, .p, ...)- -
.x | -A list or atomic vector. |
-
---|---|
.p | -A single predicate function, a formula describing such a
-predicate function, or a logical vector of the same length as |
-
... | -Additional arguments passed on to |
-
A vector the same type as .x
.
-pos <- function(x) x >= 0 -head_while(5:-5, pos)#> [1] 5 4 3 2 1 0#> [1] -1 -2 -3 -4 -5-big <- function(x) x > 100 -head_while(0:10, big)#> integer(0)tail_while(0:10, big)#> integer(0)
imap_xxx(x, ...)
, an indexed map, is short hand for
-map2(x, names(x), ...)
if x
has names, or map2(x, seq_along(x), ...)
-if it does not. This is useful if you need to compute on both the value
-and the position of an element.
imap(.x, .f, ...) - -imap_lgl(.x, .f, ...) - -imap_chr(.x, .f, ...) - -imap_int(.x, .f, ...) - -imap_dbl(.x, .f, ...) - -imap_raw(.x, .f, ...) - -imap_dfr(.x, .f, ..., .id = NULL) - -imap_dfc(.x, .f, ...) - -iwalk(.x, .f, ...)- -
.x | -A list or atomic vector. |
-
---|---|
.f | -A function, formula, or atomic vector. -If a function, it is used as is. -If a formula, e.g.
This syntax allows you to create very compact anonymous functions. -If character vector, numeric vector, or list, it
-is converted to an extractor function. Character vectors index by name
-and numeric vectors index by position; use a list to index by position
-and name at different levels. Within a list, wrap strings in |
-
... | -Additional arguments passed on to |
-
.id | -Either a string or Only applies to |
-
A vector the same length as .x
.
-# Note that when using the formula shortcut, the first argument -# is the value, and the second is the position -imap_chr(sample(10), ~ paste0(.y, ": ", .x))#> [1] "1: 8" "2: 7" "3: 9" "4: 2" "5: 1" "6: 10" "7: 5" "8: 6" "9: 3" -#> [10] "10: 4"iwalk(mtcars, ~ cat(.y, ": ", median(.x), "\n", sep = ""))#> mpg: 19.2 -#> cyl: 6 -#> disp: 196.3 -#> hp: 123 -#> drat: 3.695 -#> wt: 3.325 -#> qsec: 17.71 -#> vs: 0 -#> am: 0 -#> gear: 4 -#> carb: 2
- Map family-- The As well as functions, |
- |
---|---|
-
|
- Apply a function to each element of a vector |
-
- - | -Convert an object into a mapper function |
-
- Map variants-A rich set of variants builds on the basic idea of |
- |
-
|
- Map over multiple inputs simultaneously. |
-
- - | -Modify elements selectively |
-
-
|
- Apply a function to each element of a vector, and its index |
-
- - | -Apply a function to list-elements of a list |
-
-
|
- Invoke functions. |
-
- Predicate functionals-A predicate function is a function that either returns |
- |
- - | -Find the value or position of the first match. |
-
- - | -Do every or some elements of a list satisfy a predicate? |
-
- - | -Does a list contain an object? |
-
- - | -Find head/tail that all satisfies a predicate. |
-
- - | -Keep or discard elements using a predicate function. |
-
- - | -Negate a predicate function. |
-
- - | -Prepend a vector |
-
- Other vector transforms-A grab bag of useful tools for manipulating vectors. - |
- |
- - | -Accumulate recursive folds across a list |
-
- - | -Produce all combinations of list elements |
-
-
|
- Flatten a list of lists into a simple vector. |
-
- - | -Modify a list |
-
- - | -Reduce a list to a single value by iteratively applying a binary function. |
-
- - | -Splice objects and lists of objects into a list |
-
- - | -Transpose a list. |
-
- Adverbs-Adverbs modify the action of a function; taking a function as input and returning a function with modified action as output. - |
- |
- - | -Compose multiple functions |
-
-
|
- Lift the domain of a function |
-
- - | -Negate a predicate function. |
-
- - | -Partial apply a function, filling in some arguments. |
-
- - | -Capture side effects. |
-
- Misc- - |
- |
- - | -Default value for |
-
- - | -Infix attribute accessor |
-
- - | -Coerce array to list |
-
- - | -Coerce a list to a vector |
-
- - | -Generate random sample from a Bernoulli distribution |
-
- - | -Generate random sample from a discrete uniform distribution |
-
- - | -Re-run expressions multiple times. |
-
- - | -Set names in a vector |
-
- - | -Compute the depth of a vector |
-
This pair of functions make it easier to combine a function and list
-of parameters to get a result. invoke
is a wrapper around
-do.call
that makes it easy to use in a pipe. invoke_map
-makes it easier to call lists of functions with lists of parameters.
invoke(.f, .x = NULL, ..., .env = NULL) - -invoke_map(.f, .x = list(NULL), ..., .env = NULL) - -invoke_map_lgl(.f, .x = list(NULL), ..., .env = NULL) - -invoke_map_int(.f, .x = list(NULL), ..., .env = NULL) - -invoke_map_dbl(.f, .x = list(NULL), ..., .env = NULL) - -invoke_map_chr(.f, .x = list(NULL), ..., .env = NULL) - -invoke_map_raw(.f, .x = list(NULL), ..., .env = NULL) - -invoke_map_dfr(.f, .x = list(NULL), ..., .env = NULL) - -invoke_map_dfc(.f, .x = list(NULL), ..., .env = NULL)- -
.f | -For |
-
---|---|
.x | -For |
-
... | -Additional arguments passed to each function. |
-
.env | -Environment in which |
-
-# Invoke a function with a list of arguments -invoke(runif, list(n = 10))#> [1] 0.38870131 0.97554784 0.28989230 0.67838043 0.73531960 0.19595673 -#> [7] 0.98053967 0.74152153 0.05144628 0.53021246# Invoke a function with named arguments -invoke(runif, n = 10)#> [1] 0.69582388 0.68855600 0.03123033 0.22556253 0.30083081 0.63646561 -#> [7] 0.47902455 0.43217126 0.70643384 0.94857658-# Combine the two: -invoke(paste, list("01a", "01b"), sep = "-")#> [1] "01a-01b"# That's more natural as part of a pipeline: -list("01a", "01b") %>% - invoke(paste, ., sep = "-")#> [1] "01a-01b"-# Invoke a list of functions, each with different arguments -invoke_map(list(runif, rnorm), list(list(n = 10), list(n = 5)))#> [[1]] -#> [1] 0.18033877 0.21689988 0.68016292 0.49884561 0.64167935 0.66028435 -#> [7] 0.09602416 0.76560016 0.76967480 0.99071231 -#> -#> [[2]] -#> [1] 1.88850493 -0.09744510 -0.93584735 -0.01595031 -0.82678895 -#># Or with the same inputs: -invoke_map(list(runif, rnorm), list(list(n = 5)))#> [[1]] -#> [1] 0.06521611 0.35420680 0.82519942 0.27381825 0.57004495 -#> -#> [[2]] -#> [1] -0.4241750 -0.8723159 0.1066846 -0.5870140 -0.3278536 -#>invoke_map(list(runif, rnorm), n = 5)#> [[1]] -#> [1] 0.46598719 0.39003139 0.02006522 0.37697093 0.55991284 -#> -#> [[2]] -#> [1] 1.06730788 0.07003485 -0.63912332 -0.04996490 -0.25148344 -#># Or the same function with different inputs: -invoke_map("runif", list(list(n = 5), list(n = 10)))#> [[1]] -#> [1] 0.67176682 0.05861411 0.99706914 0.14903547 0.51855664 -#> -#> [[2]] -#> [1] 0.84612005 0.71826972 0.24131402 0.54704337 0.83480182 0.02795603 -#> [7] 0.46938430 0.80568003 0.81405131 0.40391100 -#>-# Or as a pipeline -list(m1 = mean, m2 = median) %>% invoke_map(x = rcauchy(100))#> $m1 -#> [1] -17.1555 -#> -#> $m2 -#> [1] -0.2478571 -#>list(m1 = mean, m2 = median) %>% invoke_map_dbl(x = rcauchy(100))#> m1 m2 -#> -1.0554082 -0.1491601-# Note that you can also match by position by explicitly omitting `.x`. -# This can be useful when the argument names of the functions are not -# identical -list(m1 = mean, m2 = median) %>% - invoke_map(, rcauchy(100))#> $m1 -#> [1] -0.634837 -#> -#> $m2 -#> [1] -0.1482912 -#>-# If you have pairs of function name and arguments, it's natural -# to store them in a data frame. Here we use a tibble because -# it has better support for list-columns -df <- tibble::tibble( - f = c("runif", "rpois", "rnorm"), - params = list( - list(n = 10), - list(n = 5, lambda = 10), - list(n = 10, mean = -3, sd = 10) - ) -) -df#> # A tibble: 3 x 2 -#> f params -#> <chr> <list> -#> 1 runif <list [1]> -#> 2 rpois <list [2]> -#> 3 rnorm <list [3]>invoke_map(df$f, df$params)#> [[1]] -#> [1] 0.204329063 0.471755611 0.385183100 0.392751929 0.218338430 0.007208941 -#> [7] 0.363712143 0.526106730 0.844163393 0.135991171 -#> -#> [[2]] -#> [1] 9 9 5 6 9 -#> -#> [[3]] -#> [1] -3.240840 12.844975 -2.943244 -9.482045 15.586632 5.283032 -#> [7] 8.990353 3.000512 -2.412399 -15.897635 -#>
Numeric is used in three different ways in base R:
as an alias for double (as in as.numeric()
)
to mean either integer or double (as in mode()
)
for something representable as numeric (as in as.numeric()
)
-This function tests for the second, which is often not what you want
-so these functions are deprecated.
is_numeric(x) - -is_scalar_numeric(x)- - -
keep()
and discard()
are opposites. compact()
is a handy
-wrapper that removes all empty elements.
keep(.x, .p, ...) - -discard(.x, .p, ...) - -compact(.x, .p = identity)- -
.x | -A list or vector. |
-
---|---|
.p | -A single predicate function, a formula describing such a
-predicate function, or a logical vector of the same length as |
-
... | -Additional arguments passed on to |
-
These are usually called select
or filter
and reject
or
-drop
, but those names are already taken. keep()
is similar to
-Filter()
, but the argument order is more convenient, and the
-evaluation of the predicate function .p
is stricter.
-#> [[1]] -#> [1] 9 6 10 2 7 -#> -#> [[2]] -#> [1] 2 9 8 10 6 -#> -#> [[3]] -#> [1] 1 8 4 10 9 -#> -#> [[4]] -#> [1] 10 8 5 9 3 -#>#> list()-# Using a string instead of a function will select all list elements -# where that subelement is TRUE -x <- rerun(5, a = rbernoulli(1), b = sample(10)) -x#> [[1]] -#> [[1]]$a -#> [1] TRUE -#> -#> [[1]]$b -#> [1] 6 4 5 1 10 9 8 2 3 7 -#> -#> -#> [[2]] -#> [[2]]$a -#> [1] TRUE -#> -#> [[2]]$b -#> [1] 7 4 3 6 8 1 9 10 5 2 -#> -#> -#> [[3]] -#> [[3]]$a -#> [1] TRUE -#> -#> [[3]]$b -#> [1] 7 10 9 2 8 1 3 4 6 5 -#> -#> -#> [[4]] -#> [[4]]$a -#> [1] FALSE -#> -#> [[4]]$b -#> [1] 1 4 6 3 9 10 5 7 2 8 -#> -#> -#> [[5]] -#> [[5]]$a -#> [1] TRUE -#> -#> [[5]]$b -#> [1] 1 7 10 8 5 3 6 2 4 9 -#> -#>x %>% keep("a")#> [[1]] -#> [[1]]$a -#> [1] TRUE -#> -#> [[1]]$b -#> [1] 6 4 5 1 10 9 8 2 3 7 -#> -#> -#> [[2]] -#> [[2]]$a -#> [1] TRUE -#> -#> [[2]]$b -#> [1] 7 4 3 6 8 1 9 10 5 2 -#> -#> -#> [[3]] -#> [[3]]$a -#> [1] TRUE -#> -#> [[3]]$b -#> [1] 7 10 9 2 8 1 3 4 6 5 -#> -#> -#> [[4]] -#> [[4]]$a -#> [1] TRUE -#> -#> [[4]]$b -#> [1] 1 7 10 8 5 3 6 2 4 9 -#> -#>x %>% discard("a")#> [[1]] -#> [[1]]$a -#> [1] FALSE -#> -#> [[1]]$b -#> [1] 1 4 6 3 9 10 5 7 2 8 -#> -#>-# compact() discards elements that are NULL or that have length zero -list(a = "a", b = NULL, c = integer(0), d = NA, e = list()) %>% - compact()#> $a -#> [1] "a" -#> -#> $d -#> [1] NA -#>
lift_xy()
is a composition helper. It helps you compose
-functions by lifting their domain from a kind of input to another
-kind. The domain can be changed from and to a list (l), a vector
-(v) and dots (d). For example, lift_ld(fun)
transforms a
-function taking a list to a function taking dots.
lift(..f, ..., .unnamed = FALSE) - -lift_dl(..f, ..., .unnamed = FALSE) - -lift_dv(..f, ..., .unnamed = FALSE) - -lift_vl(..f, ..., .type) - -lift_vd(..f, ..., .type) - -lift_ld(..f, ...) - -lift_lv(..f, ...)- -
..f | -A function to lift. |
-
---|---|
... | -Default arguments for |
-
.unnamed | -If |
-
.type | -A vector mold or a string describing the type of the
-input vectors. The latter can be any of the types returned by
- |
-
A function.
- -The most important of those helpers is probably lift_dl()
-because it allows you to transform a regular function to one that
-takes a list. This is often essential for composition with purrr
-functional tools. Since this is such a common function,
-lift()
is provided as an alias for that operation.
list(...)
or c(...)
Here dots should be taken here in a figurative way. The lifted
-functions does not need to take dots per se. The function is
-simply wrapped a function in do.call()
, so instead
-of taking multiple arguments, it takes a single named list or
-vector which will be interpreted as its arguments. This is
-particularly useful when you want to pass a row of a data frame
-or a list to a function and don't want to manually pull it apart
-in your function.
c(...)
to list(...)
or ...
These factories allow a function taking a vector to take a list
-or dots instead. The lifted function internally transforms its
-inputs back to an atomic vector. purrr does not obey the usual R
-casting rules (e.g., c(1, "2")
produces a character
-vector) and will produce an error if the types are not
-compatible. Additionally, you can enforce a particular vector
-type by supplying .type
.
lift_ld()
turns a function that takes a list into a
-function that takes dots. lift_vd()
does the same with a
-function that takes an atomic vector. These factory functions are
-the inverse operations of lift_dl()
and lift_dv()
.
lift_vd()
internally coerces the inputs of ..f
to
-an atomic vector. The details of this coercion can be controlled
-with .type
.
-### Lifting from ... to list(...) or c(...) - -x <- list(x = c(1:100, NA, 1000), na.rm = TRUE, trim = 0.9) -lift_dl(mean)(x)#> [1] 51#> [1] 51-# You can also use the lift() alias for this common operation: -lift(mean)(x)#> [1] 51-# Default arguments can also be specified directly in lift_dl() -list(c(1:100, NA, 1000)) %>% lift_dl(mean, na.rm = TRUE)()#> [1] 59.90099-# lift_dl() and lift_ld() are inverse of each other. -# Here we transform sum() so that it takes a list -fun <- sum %>% lift_dl() -fun(list(3, NA, 4, na.rm = TRUE))#> [1] 7-# Now we transform it back to a variadic function -fun2 <- fun %>% lift_ld() -fun2(3, NA, 4, na.rm = TRUE)#> [1] 7-# It can sometimes be useful to make sure the lifted function's -# signature has no named parameters, as would be the case for a -# function taking only dots. The lifted function will take a list -# or vector but will not match its arguments to the names of the -# input. For instance, if you give a data frame as input to your -# lifted function, the names of the columns are probably not -# related to the function signature and should be discarded. -lifted_identical <- lift_dl(identical, .unnamed = TRUE) -mtcars[c(1, 1)] %>% lifted_identical()#> [1] TRUEmtcars[c(1, 2)] %>% lifted_identical()#> [1] FALSE# - - -### Lifting from c(...) to list(...) or ... - -# In other situations we need the vector-valued function to take a -# variable number of arguments as with pmap(). This is a job for -# lift_vd(): -pmap(mtcars, lift_vd(mean))#> [[1]] -#> [1] 29.90727 -#> -#> [[2]] -#> [1] 29.98136 -#> -#> [[3]] -#> [1] 23.59818 -#> -#> [[4]] -#> [1] 38.73955 -#> -#> [[5]] -#> [1] 53.66455 -#> -#> [[6]] -#> [1] 35.04909 -#> -#> [[7]] -#> [1] 59.72 -#> -#> [[8]] -#> [1] 24.63455 -#> -#> [[9]] -#> [1] 27.23364 -#> -#> [[10]] -#> [1] 31.86 -#> -#> [[11]] -#> [1] 31.78727 -#> -#> [[12]] -#> [1] 46.43091 -#> -#> [[13]] -#> [1] 46.5 -#> -#> [[14]] -#> [1] 46.35 -#> -#> [[15]] -#> [1] 66.23273 -#> -#> [[16]] -#> [1] 66.05855 -#> -#> [[17]] -#> [1] 65.97227 -#> -#> [[18]] -#> [1] 19.44091 -#> -#> [[19]] -#> [1] 17.74227 -#> -#> [[20]] -#> [1] 18.81409 -#> -#> [[21]] -#> [1] 24.88864 -#> -#> [[22]] -#> [1] 47.24091 -#> -#> [[23]] -#> [1] 46.00773 -#> -#> [[24]] -#> [1] 58.75273 -#> -#> [[25]] -#> [1] 57.37955 -#> -#> [[26]] -#> [1] 18.92864 -#> -#> [[27]] -#> [1] 24.77909 -#> -#> [[28]] -#> [1] 24.88027 -#> -#> [[29]] -#> [1] 60.97182 -#> -#> [[30]] -#> [1] 34.50818 -#> -#> [[31]] -#> [1] 63.15545 -#> -#> [[32]] -#> [1] 26.26273 -#>-# lift_vd() will collect the arguments and concatenate them to a -# vector before passing them to ..f. You can add a check to assert -# the type of vector you expect: -lift_vd(tolower, .type = character(1))("this", "is", "ok")#> [1] "this" "is" "ok"# - - -### Lifting from list(...) to c(...) or ... - -# cross() normally takes a list of elements and returns their -# cartesian product. By lifting it you can supply the arguments as -# if it was a function taking dots: -cross_dots <- lift_ld(cross) -out1 <- cross(list(a = 1:2, b = c("a", "b", "c"))) -out2 <- cross_dots(a = 1:2, b = c("a", "b", "c")) -identical(out1, out2)#> [1] TRUE-# This kind of lifting is sometimes needed for function -# composition. An example would be to use pmap() with a function -# that takes a list. In the following, we use some() on each row of -# a data frame to check they each contain at least one element -# satisfying a condition: -mtcars %>% pmap(lift_ld(some, partial(`<`, 200)))#> [[1]] -#> [1] FALSE -#> -#> [[2]] -#> [1] FALSE -#> -#> [[3]] -#> [1] FALSE -#> -#> [[4]] -#> [1] TRUE -#> -#> [[5]] -#> [1] TRUE -#> -#> [[6]] -#> [1] TRUE -#> -#> [[7]] -#> [1] TRUE -#> -#> [[8]] -#> [1] FALSE -#> -#> [[9]] -#> [1] FALSE -#> -#> [[10]] -#> [1] FALSE -#> -#> [[11]] -#> [1] FALSE -#> -#> [[12]] -#> [1] TRUE -#> -#> [[13]] -#> [1] TRUE -#> -#> [[14]] -#> [1] TRUE -#> -#> [[15]] -#> [1] TRUE -#> -#> [[16]] -#> [1] TRUE -#> -#> [[17]] -#> [1] TRUE -#> -#> [[18]] -#> [1] FALSE -#> -#> [[19]] -#> [1] FALSE -#> -#> [[20]] -#> [1] FALSE -#> -#> [[21]] -#> [1] FALSE -#> -#> [[22]] -#> [1] TRUE -#> -#> [[23]] -#> [1] TRUE -#> -#> [[24]] -#> [1] TRUE -#> -#> [[25]] -#> [1] TRUE -#> -#> [[26]] -#> [1] FALSE -#> -#> [[27]] -#> [1] FALSE -#> -#> [[28]] -#> [1] FALSE -#> -#> [[29]] -#> [1] TRUE -#> -#> [[30]] -#> [1] FALSE -#> -#> [[31]] -#> [1] TRUE -#> -#> [[32]] -#> [1] FALSE -#>-# Default arguments for ..f can be specified in the call to -# lift_ld() -lift_ld(cross, .filter = `==`)(1:3, 1:3) %>% str()#> List of 6 -#> $ :List of 2 -#> ..$ : int 2 -#> ..$ : int 1 -#> $ :List of 2 -#> ..$ : int 3 -#> ..$ : int 1 -#> $ :List of 2 -#> ..$ : int 1 -#> ..$ : int 2 -#> $ :List of 2 -#> ..$ : int 3 -#> ..$ : int 2 -#> $ :List of 2 -#> ..$ : int 1 -#> ..$ : int 3 -#> $ :List of 2 -#> ..$ : int 2 -#> ..$ : int 3- -# Here is another function taking a list and that we can update to -# take a vector: -glue <- function(l) { - if (!is.list(l)) stop("not a list") - l %>% invoke(paste, .) -}# NOT RUN { -letters %>% glue() # fails because glue() expects a list -# }-letters %>% lift_lv(glue)() # succeeds#> [1] "a b c d e f g h i j k l m n o p q r s t u v w x y z"
list_modify()
and list_merge()
recursively combine two lists, matching
-elements either by name or position. If a sub-element is present in
-both lists list_modify()
takes the value from y
, and list_merge()
-concatenates the values together.
update_list()
handles formulas and quosures that can refer to
-values existing within the input list. Note that this function
-might be deprecated in the future in favour of a dplyr::mutate()
-method for lists.
list_modify(.x, ...) - -list_merge(.x, ...)- -
.x | -List to modify. |
-
---|---|
... | -New values of a list. Use |
-
-x <- list(x = 1:10, y = 4, z = list(a = 1, b = 2)) -str(x)#> List of 3 -#> $ x: int [1:10] 1 2 3 4 5 6 7 8 9 10 -#> $ y: num 4 -#> $ z:List of 2 -#> ..$ a: num 1 -#> ..$ b: num 2-# Update values -str(list_modify(x, a = 1))#> List of 4 -#> $ x: int [1:10] 1 2 3 4 5 6 7 8 9 10 -#> $ y: num 4 -#> $ z:List of 2 -#> ..$ a: num 1 -#> ..$ b: num 2 -#> $ a: num 1# Replace values -str(list_modify(x, z = 5))#> List of 3 -#> $ x: int [1:10] 1 2 3 4 5 6 7 8 9 10 -#> $ y: num 4 -#> $ z: num 5str(list_modify(x, z = list(a = 1:5)))#> List of 3 -#> $ x: int [1:10] 1 2 3 4 5 6 7 8 9 10 -#> $ y: num 4 -#> $ z:List of 2 -#> ..$ a: int [1:5] 1 2 3 4 5 -#> ..$ b: num 2# Remove values -str(list_modify(x, z = NULL))#> List of 2 -#> $ x: int [1:10] 1 2 3 4 5 6 7 8 9 10 -#> $ y: num 4-# Combine values -str(list_merge(x, x = 11, z = list(a = 2:5, c = 3)))#> List of 3 -#> $ x: num [1:11] 1 2 3 4 5 6 7 8 9 10 ... -#> $ y: num 4 -#> $ z:List of 3 -#> ..$ a: num [1:5] 1 2 3 4 5 -#> ..$ b: num 2 -#> ..$ c: num 3- -# All these functions take dots with splicing. Use !!! or UQS() to -# splice a list of arguments: -l <- list(new = 1, y = NULL, z = 5) -str(list_modify(x, !!! l))#> List of 3 -#> $ x : int [1:10] 1 2 3 4 5 6 7 8 9 10 -#> $ z : num 5 -#> $ new: num 1-# In update_list() you can also use quosures and formulas to -# compute new values. This function is likely to be deprecated in -# the future -str(update_list(x, z1 = ~z[[1]]))#> List of 4 -#> $ x : int [1:10] 1 2 3 4 5 6 7 8 9 10 -#> $ y : num 4 -#> $ z :List of 2 -#> ..$ a: num 1 -#> ..$ b: num 2 -#> $ z1: num 1#> List of 3 -#> $ x: int [1:10] 1 2 3 4 5 6 7 8 9 10 -#> $ y: num 4 -#> $ z: num [1:10] 5 6 7 8 9 10 11 12 13 14
lmap()
, lmap_at()
and lmap_if()
are similar to
-map()
, map_at()
and map_if()
, with the
-difference that they operate exclusively on functions that take
-and return a list (or data frame). Thus, instead of mapping
-the elements of a list (as in .x[[i]]
), they apply a
-function .f
to each subset of size 1 of that list (as in
-.x[i]
). We call those elements list-elements
).
lmap(.x, .f, ...) - -lmap_if(.x, .p, .f, ...) - -lmap_at(.x, .at, .f, ...)- -
.x | -A list or data frame. |
-
---|---|
.f | -A function that takes and returns a list or data frame. |
-
... | -Additional arguments passed on to |
-
.p | -A single predicate function, a formula describing such a
-predicate function, or a logical vector of the same length as |
-
.at | -A character vector of names, positive numeric vector of
-positions to include, or a negative numeric vector of positions to
-exlude. Only those elements corresponding to |
-
If .x
is a list, a list. If .x
is a data
-frame, a data frame.
Mapping the list-elements .x[i]
has several advantages. It
-makes it possible to work with functions that exclusively take a
-list or data frame. It enables .f
to access the attributes
-of the encapsulating list, like the name of the components it
-receives. It also enables .f
to return a larger list than
-the list-element of size 1 it got as input. Conversely, .f
-can also return empty lists. In these cases, the output list is
-reshaped with a different size than the input list .x
.
-# Let's write a function that returns a larger list or an empty list -# depending on some condition. This function also uses the names -# metadata available in the attributes of the list-element -maybe_rep <- function(x) { - n <- rpois(1, 2) - out <- rep_len(x, n) - if (length(out) > 0) { - names(out) <- paste0(names(x), seq_len(n)) - } - out -} - -# The output size varies each time we map f() -x <- list(a = 1:4, b = letters[5:7], c = 8:9, d = letters[10]) -x %>% lmap(maybe_rep)#> $a1 -#> [1] 1 2 3 4 -#> -#> $b1 -#> [1] "e" "f" "g" -#> -#> $b2 -#> [1] "e" "f" "g" -#> -#> $c1 -#> [1] 8 9 -#> -#> $d1 -#> [1] "j" -#> -#> $d2 -#> [1] "j" -#> -#> $d3 -#> [1] "j" -#>-# We can apply f() on a selected subset of x -x %>% lmap_at(c("a", "d"), maybe_rep)#> $a1 -#> [1] 1 2 3 4 -#> -#> $a2 -#> [1] 1 2 3 4 -#> -#> $a3 -#> [1] 1 2 3 4 -#> -#> $a4 -#> [1] 1 2 3 4 -#> -#> $b -#> [1] "e" "f" "g" -#> -#> $c -#> [1] 8 9 -#> -#> $d1 -#> [1] "j" -#> -#> $d2 -#> [1] "j" -#> -#> $d3 -#> [1] "j" -#>-# Or only where a condition is satisfied -x %>% lmap_if(is.character, maybe_rep)#> $a -#> [1] 1 2 3 4 -#> -#> $b1 -#> [1] "e" "f" "g" -#> -#> $b2 -#> [1] "e" "f" "g" -#> -#> $b3 -#> [1] "e" "f" "g" -#> -#> $b4 -#> [1] "e" "f" "g" -#> -#> $c -#> [1] 8 9 -#> -#> $d1 -#> [1] "j" -#>- -# A more realistic example would be a function that takes discrete -# variables in a dataset and turns them into disjunctive tables, a -# form that is amenable to fitting some types of models. - -# A disjunctive table contains only 0 and 1 but has as many columns -# as unique values in the original variable. Ideally, we want to -# combine the names of each level with the name of the discrete -# variable in order to identify them. Given these requirements, it -# makes sense to have a function that takes a data frame of size 1 -# and returns a data frame of variable size. -disjoin <- function(x, sep = "_") { - name <- names(x) - x <- as.factor(x[[1]]) - - out <- lapply(levels(x), function(level) { - as.numeric(x == level) - }) - - names(out) <- paste(name, levels(x), sep = sep) - tibble::as_tibble(out) -} - -# Now, we are ready to map disjoin() on each categorical variable of a -# data frame: -iris %>% lmap_if(is.factor, disjoin)#> # A tibble: 150 x 7 -#> Sepal.Length Sepal.Width Petal.Length Petal.Width Species_setosa -#> <dbl> <dbl> <dbl> <dbl> <dbl> -#> 1 5.1 3.5 1.4 0.2 1 -#> 2 4.9 3 1.4 0.2 1 -#> 3 4.7 3.2 1.3 0.2 1 -#> 4 4.6 3.1 1.5 0.2 1 -#> 5 5 3.6 1.4 0.2 1 -#> 6 5.4 3.9 1.7 0.4 1 -#> 7 4.6 3.4 1.4 0.3 1 -#> 8 5 3.4 1.5 0.2 1 -#> 9 4.4 2.9 1.4 0.2 1 -#> 10 4.9 3.1 1.5 0.1 1 -#> # ... with 140 more rows, and 2 more variables: Species_versicolor <dbl>, -#> # Species_virginica <dbl>mtcars %>% lmap_at(c("cyl", "vs", "am"), disjoin)#> # A tibble: 32 x 15 -#> mpg cyl_4 cyl_6 cyl_8 disp hp drat wt qsec vs_0 vs_1 am_0 -#> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> -#> 1 21 0 1 0 160 110 3.9 2.62 16.5 1 0 0 -#> 2 21 0 1 0 160 110 3.9 2.88 17.0 1 0 0 -#> 3 22.8 1 0 0 108 93 3.85 2.32 18.6 0 1 0 -#> 4 21.4 0 1 0 258 110 3.08 3.22 19.4 0 1 1 -#> 5 18.7 0 0 1 360 175 3.15 3.44 17.0 1 0 1 -#> 6 18.1 0 1 0 225 105 2.76 3.46 20.2 0 1 1 -#> 7 14.3 0 0 1 360 245 3.21 3.57 15.8 1 0 1 -#> 8 24.4 1 0 0 147. 62 3.69 3.19 20 0 1 1 -#> 9 22.8 1 0 0 141. 95 3.92 3.15 22.9 0 1 1 -#> 10 19.2 0 1 0 168. 123 3.92 3.44 18.3 0 1 1 -#> # ... with 22 more rows, and 3 more variables: am_1 <dbl>, gear <dbl>, -#> # carb <dbl>
The map functions transform their input by applying a function to -each element and returning a vector the same length as the input.
map()
, map_if()
and map_at()
always return a list. See the
-modify()
family for versions that return an object of the same
-type as the input.
The _if
and _at
variants take a predicate function .p
that
-determines which elements of .x
are transformed with .f
.
map_lgl()
, map_int()
, map_dbl()
and map_chr()
return
-vectors of the corresponding type (or die trying).
map_dfr()
and map_dfc()
return data frames created by
-row-binding and column-binding respectively. They require dplyr
-to be installed.
walk()
calls .f
for its side-effect and returns the input .x
.
map(.x, .f, ...) - -map_if(.x, .p, .f, ...) - -map_at(.x, .at, .f, ...) - -map_lgl(.x, .f, ...) - -map_chr(.x, .f, ...) - -map_int(.x, .f, ...) - -map_dbl(.x, .f, ...) - -map_raw(.x, .f, ...) - -map_dfr(.x, .f, ..., .id = NULL) - -map_dfc(.x, .f, ...) - -walk(.x, .f, ...)- -
.x | -A list or atomic vector. |
-
---|---|
.f | -A function, formula, or atomic vector. -If a function, it is used as is. -If a formula, e.g.
This syntax allows you to create very compact anonymous functions. -If character vector, numeric vector, or list, it
-is converted to an extractor function. Character vectors index by name
-and numeric vectors index by position; use a list to index by position
-and name at different levels. Within a list, wrap strings in |
-
... | -Additional arguments passed on to |
-
.p | -A single predicate function, a formula describing such a
-predicate function, or a logical vector of the same length as |
-
.at | -A character vector of names, positive numeric vector of
-positions to include, or a negative numeric vector of positions to
-exlude. Only those elements corresponding to |
-
.id | -Either a string or Only applies to |
-
All functions return a vector the same length as .x
.
map()
returns a list, map_lgl()
a logical vector, map_int()
an
-integer vector, map_dbl()
a double vector, and map_chr()
a character
-vector. The output of .f
will be automatically typed upwards,
-e.g. logical -> integer -> double -> character.
walk()
returns the input .x
(invisibly). This makes it easy to
-use in pipe.
-1:10 %>% - map(rnorm, n = 10) %>% - map_dbl(mean)#> [1] 0.9714714 2.3044667 3.3578648 4.0606456 5.2247875 6.6328310 -#> [7] 6.6803245 8.1763451 9.2655661 10.2225539-# Or use an anonymous function -1:10 %>% - map(function(x) rnorm(10, x))#> [[1]] -#> [1] -0.4960537 -0.1848187 1.6302344 3.1012525 0.3862632 -0.6346383 -#> [7] 0.9895589 0.3434939 0.3304666 0.5214110 -#> -#> [[2]] -#> [1] 3.3194563 2.6365628 2.5143278 0.2486249 2.8935975 2.2230384 2.5808166 -#> [8] 1.8221786 2.7409667 1.0025569 -#> -#> [[3]] -#> [1] 0.06102244 3.71901566 2.30199496 1.10587416 3.07629925 3.87530850 -#> [7] 3.45382739 2.14928309 3.56620161 4.15221195 -#> -#> [[4]] -#> [1] 3.243803 3.510742 2.833948 3.520331 4.115348 2.231952 2.592361 4.709178 -#> [9] 2.759157 3.631673 -#> -#> [[5]] -#> [1] 5.462080 4.677167 3.712785 3.969960 6.514089 5.346904 6.779442 5.386631 -#> [9] 4.081305 3.415664 -#> -#> [[6]] -#> [1] 5.915941 3.914929 6.003568 5.644229 7.146360 5.778812 7.018179 5.736281 -#> [9] 7.658542 5.225913 -#> -#> [[7]] -#> [1] 6.076062 6.724467 6.406600 6.877714 8.179784 7.641037 6.370411 6.192265 -#> [9] 6.139510 4.830761 -#> -#> [[8]] -#> [1] 6.624163 7.506868 7.418348 7.832771 8.485993 6.666604 7.738034 8.652386 -#> [9] 8.748855 8.896560 -#> -#> [[9]] -#> [1] 10.489300 8.340597 9.537283 9.746803 10.896317 6.939929 9.064544 -#> [8] 8.734853 8.552655 7.589299 -#> -#> [[10]] -#> [1] 9.493581 9.730238 8.914845 10.362159 9.664328 11.363804 9.288476 -#> [8] 10.662179 10.291130 10.197958 -#>-# Or a formula -1:10 %>% - map(~ rnorm(10, .x))#> [[1]] -#> [1] -0.2035661 0.9601830 1.6869825 1.7052670 1.9914417 2.1442490 -#> [7] -0.2389102 3.6548983 0.8430828 0.5765099 -#> -#> [[2]] -#> [1] 1.8016129 1.1051976 2.9042691 2.0796492 0.7411728 3.0256851 1.2692214 -#> [8] 1.8098545 2.5288647 2.5502105 -#> -#> [[3]] -#> [1] 3.5496843 2.3404576 3.0574217 0.1919895 2.0877402 2.2176208 2.3358951 -#> [8] 3.6263098 2.4927518 3.2703613 -#> -#> [[4]] -#> [1] 4.467477 4.723995 4.613837 3.382131 4.220725 5.127927 5.813454 3.916174 -#> [9] 5.367707 3.372565 -#> -#> [[5]] -#> [1] 4.783371 4.316286 4.555297 5.606490 5.624183 4.304569 4.216361 4.046876 -#> [9] 6.792756 5.348977 -#> -#> [[6]] -#> [1] 6.259104 5.194048 6.105665 5.666400 7.641848 5.356094 6.587021 5.849597 -#> [9] 4.289178 7.431033 -#> -#> [[7]] -#> [1] 4.354788 5.967543 6.292534 6.299440 7.537885 6.683668 6.160377 5.645072 -#> [9] 6.182432 6.365600 -#> -#> [[8]] -#> [1] 8.815949 8.302796 9.807087 7.105973 7.953572 7.528821 7.473307 7.904865 -#> [9] 5.504635 8.166889 -#> -#> [[9]] -#> [1] 9.350492 10.433701 9.765907 10.167521 8.863057 8.485098 10.519744 -#> [8] 8.671508 8.946328 8.436475 -#> -#> [[10]] -#> [1] 9.256091 9.890958 9.439171 10.188002 10.748851 8.083462 10.236096 -#> [8] 10.628953 10.417926 11.976758 -#>-# Extract by name or position -# .default specifies value for elements that are missing or NULL -l1 <- list(list(a = 1L), list(a = NULL, b = 2L), list(b = 3L)) -l1 %>% map("a", .default = "???")#> [[1]] -#> [1] 1 -#> -#> [[2]] -#> [1] "???" -#> -#> [[3]] -#> [1] "???" -#>l1 %>% map_int("b", .default = NA)#> [1] NA 2 3l1 %>% map_int(2, .default = NA)#> [1] NA 2 NA-# Supply multiple values to index deeply into a list -l2 <- list( - list(num = 1:3, letters[1:3]), - list(num = 101:103, letters[4:6]), - list() -) -l2 %>% map(c(2, 2))#> [[1]] -#> [1] "b" -#> -#> [[2]] -#> [1] "e" -#> -#> [[3]] -#> NULL -#>-# Use a list to build an extractor that mixes numeric indices and names, -# and .default to provide a default value if the element does not exist -l2 %>% map(list("num", 3))#> [[1]] -#> [1] 3 -#> -#> [[2]] -#> [1] 103 -#> -#> [[3]] -#> NULL -#>l2 %>% map_int(list("num", 3), .default = NA)#> [1] 3 103 NA-# A more realistic example: split a data frame into pieces, fit a -# model to each piece, summarise and extract R^2 -mtcars %>% - split(.$cyl) %>% - map(~ lm(mpg ~ wt, data = .x)) %>% - map(summary) %>% - map_dbl("r.squared")#> 4 6 8 -#> 0.5086326 0.4645102 0.4229655-# Use map_lgl(), map_dbl(), etc to reduce to a vector. -# * list -mtcars %>% map(sum)#> $mpg -#> [1] 642.9 -#> -#> $cyl -#> [1] 198 -#> -#> $disp -#> [1] 7383.1 -#> -#> $hp -#> [1] 4694 -#> -#> $drat -#> [1] 115.09 -#> -#> $wt -#> [1] 102.952 -#> -#> $qsec -#> [1] 571.16 -#> -#> $vs -#> [1] 14 -#> -#> $am -#> [1] 13 -#> -#> $gear -#> [1] 118 -#> -#> $carb -#> [1] 90 -#># * vector -mtcars %>% map_dbl(sum)#> mpg cyl disp hp drat wt qsec vs -#> 642.900 198.000 7383.100 4694.000 115.090 102.952 571.160 14.000 -#> am gear carb -#> 13.000 118.000 90.000-# If each element of the output is a data frame, use -# map_dfr to row-bind them together: -mtcars %>% - split(.$cyl) %>% - map(~ lm(mpg ~ wt, data = .x)) %>% - map_dfr(~ as.data.frame(t(as.matrix(coef(.)))))#> (Intercept) wt -#> 1 39.57120 -5.647025 -#> 2 28.40884 -2.780106 -#> 3 23.86803 -2.192438# (if you also want to preserve the variable names see -# the broom package) -
These functions are variants of map()
that iterate over multiple arguments
-simultaneously. They are parallel in the sense that each input is processed
-in parallel with the others, not in the sense of multicore computing. They
-share the same notion of "parallel" as base::pmax()
and base::pmin()
.
-map2()
and walk2()
are specialised for the two argument case; pmap()
-and pwalk()
allow you to provide any number of arguments in a list.
map2(.x, .y, .f, ...) - -map2_lgl(.x, .y, .f, ...) - -map2_int(.x, .y, .f, ...) - -map2_dbl(.x, .y, .f, ...) - -map2_chr(.x, .y, .f, ...) - -map2_raw(.x, .y, .f, ...) - -map2_dfr(.x, .y, .f, ..., .id = NULL) - -map2_dfc(.x, .y, .f, ...) - -walk2(.x, .y, .f, ...) - -pmap(.l, .f, ...) - -pmap_lgl(.l, .f, ...) - -pmap_int(.l, .f, ...) - -pmap_dbl(.l, .f, ...) - -pmap_chr(.l, .f, ...) - -pmap_raw(.l, .f, ...) - -pmap_dfr(.l, .f, ..., .id = NULL) - -pmap_dfc(.l, .f, ...) - -pwalk(.l, .f, ...)- -
.x, .y | -Vectors of the same length. A vector of length 1 will -be recycled. |
-
---|---|
.f | -A function, formula, or atomic vector. -If a function, it is used as is. -If a formula, e.g.
This syntax allows you to create very compact anonymous functions. -If character vector, numeric vector, or list, it
-is converted to an extractor function. Character vectors index by name
-and numeric vectors index by position; use a list to index by position
-and name at different levels. Within a list, wrap strings in |
-
... | -Additional arguments passed on to |
-
.id | -Either a string or Only applies to |
-
.l | -A list of lists. The length of |
-
An atomic vector, list, or data frame, depending on the suffix.
-Atomic vectors and lists will be named if .x
or the first
-element of .l
is named.
If all input is length 0, the output will be length 0. If any -input is length 1, it will be recycled to the length of the longest.
- -Note that arguments to be vectorised over come before the .f
,
-and arguments that are supplied to every call come after .f
.
-x <- list(1, 10, 100) -y <- list(1, 2, 3) -z <- list(5, 50, 500) - -map2(x, y, ~ .x + .y)#> [[1]] -#> [1] 2 -#> -#> [[2]] -#> [1] 12 -#> -#> [[3]] -#> [1] 103 -#># Or just -map2(x, y, `+`)#> [[1]] -#> [1] 2 -#> -#> [[2]] -#> [1] 12 -#> -#> [[3]] -#> [1] 103 -#>-pmap(list(x, y, z), sum)#> [[1]] -#> [1] 7 -#> -#> [[2]] -#> [1] 62 -#> -#> [[3]] -#> [1] 603 -#>-# Matching arguments by position -pmap(list(x, y, z), function(a, b, c) a / (b + c))#> [[1]] -#> [1] 0.1666667 -#> -#> [[2]] -#> [1] 0.1923077 -#> -#> [[3]] -#> [1] 0.1988072 -#>-# Matching arguments by name -l <- list(a = x, b = y, c = z) -pmap(l, function(c, b, a) a / (b + c))#> [[1]] -#> [1] 0.1666667 -#> -#> [[2]] -#> [1] 0.1923077 -#> -#> [[3]] -#> [1] 0.1988072 -#>-# Split into pieces, fit model to each piece, then predict -by_cyl <- mtcars %>% split(.$cyl) -mods <- by_cyl %>% map(~ lm(mpg ~ wt, data = .)) -map2(mods, by_cyl, predict)#> $`4` -#> Datsun 710 Merc 240D Merc 230 Fiat 128 Honda Civic -#> 26.47010 21.55719 21.78307 27.14774 30.45125 -#> Toyota Corolla Toyota Corona Fiat X1-9 Porsche 914-2 Lotus Europa -#> 29.20890 25.65128 28.64420 27.48656 31.02725 -#> Volvo 142E -#> 23.87247 -#> -#> $`6` -#> Mazda RX4 Mazda RX4 Wag Hornet 4 Drive Valiant Merc 280 -#> 21.12497 20.41604 19.47080 18.78968 18.84528 -#> Merc 280C Ferrari Dino -#> 18.84528 20.70795 -#> -#> $`8` -#> Hornet Sportabout Duster 360 Merc 450SE Merc 450SL -#> 16.32604 16.04103 14.94481 15.69024 -#> Merc 450SLC Cadillac Fleetwood Lincoln Continental Chrysler Imperial -#> 15.58061 12.35773 11.97625 12.14945 -#> Dodge Challenger AMC Javelin Camaro Z28 Pontiac Firebird -#> 16.15065 16.33700 15.44907 15.43811 -#> Ford Pantera L Maserati Bora -#> 16.91800 16.04103 -#>-# Vectorizing a function over multiple arguments -df <- data.frame( - x = c("apple", "banana", "cherry"), - pattern = c("p", "n", "h"), - replacement = c("x", "f", "q"), - stringsAsFactors = FALSE - ) -pmap(df, gsub)#> [[1]] -#> [1] "axxle" -#> -#> [[2]] -#> [1] "bafafa" -#> -#> [[3]] -#> [1] "cqerry" -#>pmap_chr(df, gsub)#> [1] "axxle" "bafafa" "cqerry"-# Use `...` to absorb unused components of input list .l -df <- data.frame( - x = 1:3 + 0.1, - y = 3:1 - 0.1, - z = letters[1:3] -) -plus <- function(x, y) x + y# NOT RUN { -# this won't work -pmap(df, plus) -# }# but this will -plus2 <- function(x, y, ...) x + y -pmap_dbl(df, plus2)#> [1] 4 4 4-# The "p" for "parallel" in pmap() is the same as in base::pmin() -# and base::pmax() -df <- data.frame( - x = c(1, 2, 5), - y = c(5, 4, 8) -) -# all produce the same result -pmin(df$x, df$y)#> [1] 1 2 5map2_dbl(df$x, df$y, min)#> [1] 1 2 5pmap_dbl(df, min)#> [1] 1 2 5
modify()
is a short-cut for x[] <- map(x, .f); return(x)
. modify_if()
-only modifies the elements of x
that satisfy a predicate and leaves the
-others unchanged. modify_at()
only modifies elements given by names or
-positions. modify_depth()
only modifies elements at a given level of a
-nested data structure.
modify(.x, .f, ...) - -# S3 method for default -modify(.x, .f, ...) - -modify_if(.x, .p, .f, ...) - -# S3 method for default -modify_if(.x, .p, .f, ...) - -modify_at(.x, .at, .f, ...) - -# S3 method for default -modify_at(.x, .at, .f, ...) - -modify_depth(.x, .depth, .f, ..., .ragged = .depth < 0) - -# S3 method for default -modify_depth(.x, .depth, .f, ..., .ragged = .depth < 0)- -
.x | -A list or atomic vector. |
-
---|---|
.f | -A function, formula, or atomic vector. -If a function, it is used as is. -If a formula, e.g.
This syntax allows you to create very compact anonymous functions. -If character vector, numeric vector, or list, it
-is converted to an extractor function. Character vectors index by name
-and numeric vectors index by position; use a list to index by position
-and name at different levels. Within a list, wrap strings in |
-
... | -Additional arguments passed on to |
-
.p | -A single predicate function, a formula describing such a
-predicate function, or a logical vector of the same length as |
-
.at | -A character vector of names, positive numeric vector of
-positions to include, or a negative numeric vector of positions to
-exlude. Only those elements corresponding to |
-
.depth | -Level of
|
-
.ragged | -If |
-
An object the same class as .x
Since the transformation can alter the structure of the input; it's
-your responsibility to ensure that the transformation produces a
-valid output. For example, if you're modifying a data frame, .f
-must preserve the length of the input.
All these functions are S3 generic. However, the default method is
-sufficient in many cases. It should be suitable for any data type
-that implements the subset-assignment method [<-
. Methods are provided
-for character, logical, integer and double classes (counterparts to map_chr
,
-map_lgl
, map_int
, and map_dbl
)
In some cases it may make sense to provide a custom implementation
-with a method suited to your S3 class. For example, a grouped_df
-method might take into account the grouped nature of a data frame.
-# Convert factors to characters -iris %>% - modify_if(is.factor, as.character) %>% - str()#> 'data.frame': 150 obs. of 5 variables: -#> $ Sepal.Length: num 5.1 4.9 4.7 4.6 5 5.4 4.6 5 4.4 4.9 ... -#> $ Sepal.Width : num 3.5 3 3.2 3.1 3.6 3.9 3.4 3.4 2.9 3.1 ... -#> $ Petal.Length: num 1.4 1.4 1.3 1.5 1.4 1.7 1.4 1.5 1.4 1.5 ... -#> $ Petal.Width : num 0.2 0.2 0.2 0.2 0.2 0.4 0.3 0.2 0.2 0.1 ... -#> $ Species : chr "setosa" "setosa" "setosa" "setosa" ...-# Specify which columns to map with a numeric vector of positions: -mtcars %>% modify_at(c(1, 4, 5), as.character) %>% str()#> 'data.frame': 32 obs. of 11 variables: -#> $ mpg : chr "21" "21" "22.8" "21.4" ... -#> $ cyl : num 6 6 4 6 8 6 8 4 4 6 ... -#> $ disp: num 160 160 108 258 360 ... -#> $ hp : chr "110" "110" "93" "110" ... -#> $ drat: chr "3.9" "3.9" "3.85" "3.08" ... -#> $ wt : num 2.62 2.88 2.32 3.21 3.44 ... -#> $ qsec: num 16.5 17 18.6 19.4 17 ... -#> $ vs : num 0 0 1 1 0 1 0 1 1 1 ... -#> $ am : num 1 1 1 0 0 0 0 0 0 0 ... -#> $ gear: num 4 4 4 3 3 3 3 4 4 4 ... -#> $ carb: num 4 4 1 1 2 1 4 2 2 4 ...-# Or with a vector of names: -mtcars %>% modify_at(c("cyl", "am"), as.character) %>% str()#> 'data.frame': 32 obs. of 11 variables: -#> $ mpg : num 21 21 22.8 21.4 18.7 18.1 14.3 24.4 22.8 19.2 ... -#> $ cyl : chr "6" "6" "4" "6" ... -#> $ disp: num 160 160 108 258 360 ... -#> $ hp : num 110 110 93 110 175 105 245 62 95 123 ... -#> $ drat: num 3.9 3.9 3.85 3.08 3.15 2.76 3.21 3.69 3.92 3.92 ... -#> $ wt : num 2.62 2.88 2.32 3.21 3.44 ... -#> $ qsec: num 16.5 17 18.6 19.4 17 ... -#> $ vs : num 0 0 1 1 0 1 0 1 1 1 ... -#> $ am : chr "1" "1" "1" "0" ... -#> $ gear: num 4 4 4 3 3 3 3 4 4 4 ... -#> $ carb: num 4 4 1 1 2 1 4 2 2 4 ...-list(x = rbernoulli(100), y = 1:100) %>% - transpose() %>% - modify_if("x", ~ update_list(., y = ~ y * 100)) %>% - transpose() %>% - simplify_all()#> $x -#> [1] FALSE TRUE FALSE FALSE FALSE TRUE TRUE TRUE FALSE TRUE TRUE TRUE -#> [13] FALSE FALSE FALSE TRUE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE -#> [25] TRUE FALSE FALSE TRUE TRUE TRUE FALSE TRUE TRUE TRUE FALSE TRUE -#> [37] TRUE TRUE TRUE FALSE FALSE FALSE FALSE TRUE FALSE FALSE TRUE FALSE -#> [49] FALSE FALSE TRUE FALSE FALSE FALSE FALSE TRUE FALSE TRUE TRUE FALSE -#> [61] TRUE TRUE FALSE TRUE FALSE TRUE FALSE FALSE FALSE FALSE TRUE FALSE -#> [73] TRUE FALSE FALSE TRUE TRUE TRUE FALSE FALSE TRUE FALSE TRUE FALSE -#> [85] TRUE TRUE FALSE TRUE FALSE TRUE TRUE TRUE FALSE TRUE FALSE FALSE -#> [97] TRUE FALSE FALSE TRUE -#> -#> $y -#> [1] 1 200 3 4 5 600 700 800 9 1000 1100 1200 -#> [13] 13 14 15 1600 17 18 19 20 21 22 23 24 -#> [25] 2500 26 27 2800 2900 3000 31 3200 3300 3400 35 3600 -#> [37] 3700 3800 3900 40 41 42 43 4400 45 46 4700 48 -#> [49] 49 50 5100 52 53 54 55 5600 57 5800 5900 60 -#> [61] 6100 6200 63 6400 65 6600 67 68 69 70 7100 72 -#> [73] 7300 74 75 7600 7700 7800 79 80 8100 82 8300 84 -#> [85] 8500 8600 87 8800 89 9000 9100 9200 93 9400 95 96 -#> [97] 9700 98 99 10000 -#>-# Modify at specified depth --------------------------- -l1 <- list( - obj1 = list( - prop1 = list(param1 = 1:2, param2 = 3:4), - prop2 = list(param1 = 5:6, param2 = 7:8) - ), - obj2 = list( - prop1 = list(param1 = 9:10, param2 = 11:12), - prop2 = list(param1 = 12:14, param2 = 15:17) - ) -) - -# In the above list, "obj" is level 1, "prop" is level 2 and "param" -# is level 3. To apply sum() on all params, we map it at depth 3: -l1 %>% modify_depth(3, sum) %>% str()#> List of 2 -#> $ obj1:List of 2 -#> ..$ prop1:List of 2 -#> .. ..$ param1: int 3 -#> .. ..$ param2: int 7 -#> ..$ prop2:List of 2 -#> .. ..$ param1: int 11 -#> .. ..$ param2: int 15 -#> $ obj2:List of 2 -#> ..$ prop1:List of 2 -#> .. ..$ param1: int 19 -#> .. ..$ param2: int 23 -#> ..$ prop2:List of 2 -#> .. ..$ param1: int 39 -#> .. ..$ param2: int 48-# modify() lets us pluck the elements prop1/param2 in obj1 and obj2: -l1 %>% modify(c("prop1", "param2")) %>% str()#> List of 2 -#> $ obj1: int [1:2] 3 4 -#> $ obj2: int [1:2] 11 12-# But what if we want to pluck all param2 elements? Then we need to -# act at a lower level: -l1 %>% modify_depth(2, "param2") %>% str()#> List of 2 -#> $ obj1:List of 2 -#> ..$ prop1: int [1:2] 3 4 -#> ..$ prop2: int [1:2] 7 8 -#> $ obj2:List of 2 -#> ..$ prop1: int [1:2] 11 12 -#> ..$ prop2: int [1:3] 15 16 17-# modify_depth() can be with other purrr functions to make them operate at -# a lower level. Here we ask pmap() to map paste() simultaneously over all -# elements of the objects at the second level. paste() is effectively -# mapped at level 3. -l1 %>% modify_depth(2, ~ pmap(., paste, sep = " / ")) %>% str()#> List of 2 -#> $ obj1:List of 2 -#> ..$ prop1:List of 2 -#> .. ..$ : chr "1 / 3" -#> .. ..$ : chr "2 / 4" -#> ..$ prop2:List of 2 -#> .. ..$ : chr "5 / 7" -#> .. ..$ : chr "6 / 8" -#> $ obj2:List of 2 -#> ..$ prop1:List of 2 -#> .. ..$ : chr "9 / 11" -#> .. ..$ : chr "10 / 12" -#> ..$ prop2:List of 3 -#> .. ..$ : chr "12 / 15" -#> .. ..$ : chr "13 / 16" -#> .. ..$ : chr "14 / 17"
Negate a predicate function.
- -negate(.p, .default = FALSE)- -
.p | -A single predicate function, a formula describing such a
-predicate function, or a logical vector of the same length as |
-
---|---|
.default | -Optional additional argument for extractor functions
-(i.e. when |
-
A new predicate function.
- - --negate("x")#> function (x, ...) -#> { -#> !pluck(x, list("x"), .default = NULL) -#> } -#> <environment: 0x7f9d53c00600>negate(is.null)#> function (x) -#> { -#> !is.null(x = x) -#> } -#> <environment: base>negate(~ .x > 0)#> function (..., .x = ..1, .y = ..2, . = ..1) -#> { -#> !.x > 0 -#> } -#> <environment: 0x7f9d53fcf190>#> [1] 5#> [1] 5#> [1] 5
NULL
. — null-default • purrrThis infix function makes it easy to replace NULL
s with a
-default value. It's inspired by the way that Ruby's or operation (||
)
-works.
x %||% y- -
x, y | -If |
-
---|
-1 %||% 2#> [1] 1NULL %||% 2#> [1] 2
Partial function application allows you to modify a function by pre-filling -some of the arguments. It is particularly useful in conjunction with -functionals and other function operators.
- -partial(...f, ..., .env = parent.frame(), .lazy = TRUE, .first = TRUE)- -
...f | -a function. For the output source to read well, this should be a -named function. |
-
---|---|
... | -named arguments to |
-
.env | -the environment of the created function. Defaults to
- |
-
.lazy | -If |
-
.first | -If |
-
There are many ways to implement partial function application in R.
-(see e.g. dots
in https://github.com/crowding/ptools for another
-approach.) This implementation is based on creating functions that are as
-similar as possible to the anonymous functions that you'd create by hand,
-if you weren't using partial
.
-# Partial is designed to replace the use of anonymous functions for -# filling in function arguments. Instead of: -compact1 <- function(x) discard(x, is.null) - -# we can write: -compact2 <- partial(discard, .p = is.null) - -# and the generated source code is very similar to what we made by hand -compact1#> function(x) discard(x, is.null) -#> <environment: 0x7f9d52624630>compact2#> function (...) -#> discard(.p = is.null, ...) -#> <environment: 0x7f9d52624630>-# Note that the evaluation occurs "lazily" so that arguments will be -# repeatedly evaluated -f <- partial(runif, n = rpois(1, 5)) -f#> function (...) -#> runif(n = rpois(1, 5), ...) -#> <environment: 0x7f9d52624630>f()#> [1] 0.17635011 0.11075900 0.93834241 0.84647114 0.57114358 0.67909341 0.08932224f()#> [1] 0.2269574 0.4481785 0.1612276 0.1761117 0.1982281 0.3576113 0.1813332-# You can override this by saying .lazy = FALSE -f <- partial(runif, n = rpois(1, 5), .lazy = FALSE) -f#> function (...) -#> runif(n = 5L, ...) -#> <environment: 0x7f9d52624630>f()#> [1] 0.659083010 0.660253409 0.002400788 0.993446034 0.627489231f()#> [1] 0.01463034 0.20517822 0.66307666 0.46374403 0.36034816-# This also means that partial works fine with functions that do -# non-standard evaluation -my_long_variable <- 1:10 -plot2 <- partial(plot, my_long_variable) -plot2()plot2(runif(10), type = "l")
This is a generalised form of [[
which allows you to index deeply
-and flexibly into data structures. It supports R standard accessors
-like integer positions and string names, and also accepts arbitrary
-accessor functions, i.e. functions that take an object and return
-some internal piece.
pluck()
is often more readable than a mix of operators and
-accessors because it reads linearly and is free of syntactic
-cruft. Compare: accessor(x[[1]])$foo
to pluck(x, 1, accessor, "foo")
.
Furthermore, pluck()
never partial-matches unlike $
which will
-select the disp
object if you write mtcars$di
.
attr_getter()
generates an attribute accessor function;
-i.e., it generates a function for extracting an attribute with
-a given name. Unlike the base R attr()
function with default
-options, it doesn't use partial matching.
pluck(.x, ..., .default = NULL) - -attr_getter(attr)- -
.x | -A vector or environment |
-
---|---|
... | -A list of accessors for indexing into the object. Can be -an integer position, a string name, or an accessor function. If -the object being indexed is an S4 object, accessing it by name -will return the corresponding slot. -These dots splice lists automatically. This -means you can supply arguments and lists of arguments -indistinctly. |
-
.default | -Value to use if target is empty or absent. |
-
attr | -An attribute name as string. |
-
Since it handles arbitrary accessor functions, pluck()
is a type
-of composition operator. However, it is indexing-oriented thanks to
-its handling of strings and integers. By the same token is also
-explicit regarding the intent of the composition (e.g. extraction).
-# pluck() supports integer positions, string names, and functions. -# Using functions, you can easily extend pluck(). Let's create a -# list of data structures: -obj1 <- list("a", list(1, elt = "foobar")) -obj2 <- list("b", list(2, elt = "foobaz")) -x <- list(obj1, obj2) - -# And now an accessor for these complex data structures: -my_element <- function(x) x[[2]]$elt - -# The accessor can then be passed to pluck: -pluck(x, 1, my_element)#> [1] "foobar"pluck(x, 2, my_element)#> [1] "foobaz"-# Even for this simple data structure, this is more readable than -# the alternative form because it requires you to read both from -# right-to-left and from left-to-right in different parts of the -# expression: -my_element(x[[1]])#> [1] "foobar"- -# This technique is used for plucking into attributes with -# attr_getter(). It takes an attribute name and returns a function -# to access the attribute: -obj1 <- structure("obj", obj_attr = "foo") -obj2 <- structure("obj", obj_attr = "bar") -x <- list(obj1, obj2) - -# pluck() is handy for extracting deeply into a data structure. -# Here we'll first extract by position, then by attribute: -pluck(x, 1, attr_getter("obj_attr")) # From first object#> [1] "foo"pluck(x, 2, attr_getter("obj_attr")) # From second object#> [1] "bar"- -# pluck() splices lists of arguments automatically. The following -# pluck is equivalent to the one above: -idx <- list(1, attr_getter("obj_attr")) -pluck(x, idx)#> [1] "foo"
This is a companion to append()
to help merging two
-lists or atomic vectors. prepend()
is a clearer semantic
-signal than c()
that a vector is to be merged at the beginning of
-another, especially in a pipe chain.
prepend(x, values, before = 1)- -
x | -the vector to be modified. |
-
---|---|
values | -to be included in the modified vector. |
-
before | -a subscript, before which the values are to be appended. |
-
A merged vector.
- - --x <- as.list(1:3) - -x %>% append("a")#> [[1]] -#> [1] 1 -#> -#> [[2]] -#> [1] 2 -#> -#> [[3]] -#> [1] 3 -#> -#> [[4]] -#> [1] "a" -#>x %>% prepend("a")#> [[1]] -#> [1] "a" -#> -#> [[2]] -#> [1] 1 -#> -#> [[3]] -#> [1] 2 -#> -#> [[4]] -#> [1] 3 -#>x %>% prepend(list("a", "b"), before = 3)#> [[1]] -#> [1] 1 -#> -#> [[2]] -#> [1] 2 -#> -#> [[3]] -#> [1] "a" -#> -#> [[4]] -#> [1] "b" -#> -#> [[5]] -#> [1] 3 -#>
Generate random sample from a Bernoulli distribution
- -rbernoulli(n, p = 0.5)- -
n | -Number of samples |
-
---|---|
p | -Probability of getting |
-
A logical vector
- - --rbernoulli(10)#> [1] TRUE FALSE FALSE FALSE TRUE FALSE FALSE FALSE FALSE FALSErbernoulli(100, 0.1)#> [1] FALSE FALSE FALSE FALSE FALSE TRUE TRUE TRUE FALSE FALSE TRUE FALSE -#> [13] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE -#> [25] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE -#> [37] FALSE FALSE FALSE FALSE TRUE FALSE FALSE FALSE FALSE FALSE FALSE FALSE -#> [49] FALSE FALSE FALSE FALSE FALSE FALSE FALSE TRUE FALSE FALSE FALSE FALSE -#> [61] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE -#> [73] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE TRUE FALSE FALSE -#> [85] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE -#> [97] FALSE FALSE FALSE FALSE
Generate random sample from a discrete uniform distribution
- -rdunif(n, b, a = 1)- -
n | -Number of samples to draw. |
-
---|---|
a, b | -Range of the distribution (inclusive). |
-
-table(rdunif(1e3, 10))#> -#> 1 2 3 4 5 6 7 8 9 10 -#> 109 86 106 105 103 89 109 93 95 105table(rdunif(1e3, 10, -5))#> -#> -5 -4 -3 -2 -1 0 1 2 3 4 5 6 7 8 9 10 -#> 50 63 69 60 67 54 66 66 69 54 81 67 61 55 72 46
R/reduce.R
- reduce.Rd
reduce()
combines from the left, reduce_right()
combines from
-the right. reduce(list(x1, x2, x3), f)
is equivalent to
-f(f(x1, x2), x3)
; reduce_right(list(x1, x2, x3), f)
is equivalent to
-f(f(x3, x2), x1)
.
reduce(.x, .f, ..., .init) - -reduce_right(.x, .f, ..., .init) - -reduce2(.x, .y, .f, ..., .init) - -reduce2_right(.x, .y, .f, ..., .init)- -
.x | -A list or atomic vector. |
-
---|---|
.f | -For For |
-
... | -Additional arguments passed on to |
-
.init | -If supplied, will be used as the first value to start
-the accumulation, rather than using |
-
.y | -For |
-
-1:3 %>% reduce(`+`)#> [1] 61:10 %>% reduce(`*`)#> [1] 3628800-paste2 <- function(x, y, sep = ".") paste(x, y, sep = sep) -letters[1:4] %>% reduce(paste2)#> [1] "a.b.c.d"letters[1:4] %>% reduce2(c("-", ".", "-"), paste2)#> [1] "a-b.c-d"#> [[1]] -#> [1] 5 8 6 10 9 -#> -#> [[2]] -#> [1] 8 10 6 4 1 -#>reduce(samples, union)#> [1] 5 8 6 10 9 4 1reduce(samples, intersect)#> [1] 8 6 10-x <- list(c(0, 1), c(2, 3), c(4, 5)) -x %>% reduce(c)#> [1] 0 1 2 3 4 5x %>% reduce_right(c)#> [1] 4 5 2 3 0 1# Equivalent to: -x %>% rev() %>% reduce(c)#> [1] 4 5 2 3 0 1
These objects are imported from other packages. Follow the links -below to see their documentation.
-is_bare_list
, is_bare_atomic
, is_bare_vector
, is_bare_double
, is_bare_integer
, is_bare_numeric
, is_bare_character
, is_bare_logical
, is_list
, is_atomic
, is_vector
, is_integer
, is_double
, is_character
, is_logical
, is_null
, is_function
, is_scalar_list
, is_scalar_atomic
, is_scalar_vector
, is_scalar_double
, is_scalar_character
, is_scalar_logical
, is_scalar_integer
, is_empty
, is_formula
This is a convenient way of generating sample data. It works similarly to
-replicate(..., simplify = FALSE)
.
rerun(.n, ...)- -
.n | -Number of times to run expressions |
-
---|---|
... | -Expressions to re-run. |
-
A list of length .n
. Each element of ...
will be
-re-run once for each .n
.
There is one special case: if there's a single unnamed input, the second
-level list will be dropped. In this case, rerun(n, x)
behaves like
-replicate(n, x, simplify = FALSE)
.
-10 %>% rerun(rnorm(5))#> [[1]] -#> [1] 0.5802748 -0.5941795 -0.3718553 -1.4230352 -1.1404838 -#> -#> [[2]] -#> [1] 1.2685047 -0.2801044 0.9110540 2.0043945 0.2471404 -#> -#> [[3]] -#> [1] -0.4290761 -2.1837551 -0.1236349 0.2873839 -0.3715993 -#> -#> [[4]] -#> [1] 1.6265623 0.3099083 -1.5640407 -1.2306235 -1.3701975 -#> -#> [[5]] -#> [1] 0.1065558 -0.3988255 -0.3534061 0.2345011 1.7697899 -#> -#> [[6]] -#> [1] 0.45787638 1.14375811 -1.30772262 -0.94254960 -0.09496125 -#> -#> [[7]] -#> [1] -1.130047 -1.456208 -1.698718 -1.240827 -1.058299 -#> -#> [[8]] -#> [1] -0.75437857 -0.58510694 0.55814208 -0.01900658 -0.16935602 -#> -#> [[9]] -#> [1] 1.295331 -1.426060 1.754104 -2.115575 1.009702 -#> -#> [[10]] -#> [1] 0.2545351 -0.3775541 0.3724693 -1.4626811 0.9037093 -#>#> [1] 0.3884734 -0.2797106 -0.1499915 -0.1213636 0.1875155 -0.1596498 -#> [7] 0.1293135 0.5927749 0.7689696 -0.4564721
These functions wrap functions so that instead of generating side effects -through printed output, messages, warnings, and errors, they return enhanced -output. They are all adverbs because they modify the action of a verb (a -function).
- -safely(.f, otherwise = NULL, quiet = TRUE) - -quietly(.f) - -possibly(.f, otherwise, quiet = TRUE) - -auto_browse(.f)- -
.f | -A function, formula, or atomic vector. -If a function, it is used as is. -If a formula, e.g.
This syntax allows you to create very compact anonymous functions. -If character vector, numeric vector, or list, it
-is converted to an extractor function. Character vectors index by name
-and numeric vectors index by position; use a list to index by position
-and name at different levels. Within a list, wrap strings in |
-
---|---|
otherwise | -Default value to use when an error occurs. |
-
quiet | -Hide errors ( |
-
safely
: wrapped function instead returns a list with
-components result
and error
. One value is always NULL
.
quietly
: wrapped function instead returns a list with components
-result
, output
, messages
and warnings
.
possibly
: wrapped function uses a default value (otherwise
)
-whenever an error occurs.
-safe_log <- safely(log) -safe_log(10)#> $result -#> [1] 2.302585 -#> -#> $error -#> NULL -#>safe_log("a")#> $result -#> NULL -#> -#> $error -#> <simpleError in log(x = x, base = base): non-numeric argument to mathematical function> -#>#> $result -#> $result[[1]] -#> NULL -#> -#> $result[[2]] -#> [1] 2.302585 -#> -#> $result[[3]] -#> [1] 4.60517 -#> -#> -#> $error -#> $error[[1]] -#> <simpleError in log(x = x, base = base): non-numeric argument to mathematical function> -#> -#> $error[[2]] -#> NULL -#> -#> $error[[3]] -#> NULL -#> -#>-# This is a bit easier to work with if you supply a default value -# of the same type and use the simplify argument to transpose(): -safe_log <- safely(log, otherwise = NA_real_) -list("a", 10, 100) %>% - map(safe_log) %>% - transpose() %>% - simplify_all()#> $result -#> [1] NA 2.302585 4.605170 -#> -#> $error -#> $error[[1]] -#> <simpleError in log(x = x, base = base): non-numeric argument to mathematical function> -#> -#> $error[[2]] -#> NULL -#> -#> $error[[3]] -#> NULL -#> -#>-# To replace errors with a default value, use possibly(). -list("a", 10, 100) %>% - map_dbl(possibly(log, NA_real_))#> [1] NA 2.302585 4.605170-# For interactive usage, auto_browse() is useful because it automatically -# starts a browser() in the right place. -f <- function(x) { - y <- 20 - if (x > 5) { - stop("!") - } else { - x - } -} -if (interactive()) { - map(1:6, auto_browse(f)) -}#> Called from: eval_bare(quote(browser()), env = frame$env)#> Error in .f(...): !-# It doesn't make sense to use auto_browse with primitive functions, -# because they are implemented in C so there's no useful environment -# for you to interact with. -
These objects are imported from other packages. Follow the links -below to see their documentation.
-set_names(x, nm = x, ...)- -
x | -Vector to name |
-
---|---|
nm, ... | -Vector of names, the same length as You can specify names in three ways:
|
-
.x
with the names attribute set.
This is a snake case wrapper for stats::setNames()
, with
-tweaked defaults, and stricter argument checking.
-set_names(1:4, c("a", "b", "c", "d"))#> a b c d -#> 1 2 3 4set_names(1:4, letters[1:4])#> a b c d -#> 1 2 3 4set_names(1:4, "a", "b", "c", "d")#> a b c d -#> 1 2 3 4-# If the second argument is ommitted a vector is named with itself -set_names(letters[1:5])#> a b c d e -#> "a" "b" "c" "d" "e"-# Alternatively you can supply a function -set_names(1:10, ~ letters[seq_along(.)])#> a b c d e f g h i j -#> 1 2 3 4 5 6 7 8 9 10set_names(head(mtcars), toupper)#> MPG CYL DISP HP DRAT WT QSEC VS AM GEAR CARB -#> Mazda RX4 21.0 6 160 110 3.90 2.620 16.46 0 1 4 4 -#> Mazda RX4 Wag 21.0 6 160 110 3.90 2.875 17.02 0 1 4 4 -#> Datsun 710 22.8 4 108 93 3.85 2.320 18.61 1 1 4 1 -#> Hornet 4 Drive 21.4 6 258 110 3.08 3.215 19.44 1 0 3 1 -#> Hornet Sportabout 18.7 8 360 175 3.15 3.440 17.02 0 0 3 2 -#> Valiant 18.1 6 225 105 2.76 3.460 20.22 1 0 3 1
This splices all arguments into a list. Non-list objects and lists -with a S3 class are encapsulated in a list before concatenation.
- -splice(...)- -
... | -Objects to concatenate. |
-
---|
A list.
- - --inputs <- list(arg1 = "a", arg2 = "b") - -# splice() concatenates the elements of inputs with arg3 -splice(inputs, arg3 = c("c1", "c2")) %>% str()#> List of 3 -#> $ arg1: chr "a" -#> $ arg2: chr "b" -#> $ arg3: chr [1:2] "c1" "c2"list(inputs, arg3 = c("c1", "c2")) %>% str()#> List of 2 -#> $ :List of 2 -#> ..$ arg1: chr "a" -#> ..$ arg2: chr "b" -#> $ arg3: chr [1:2] "c1" "c2"c(inputs, arg3 = c("c1", "c2")) %>% str()#> List of 4 -#> $ arg1 : chr "a" -#> $ arg2 : chr "b" -#> $ arg31: chr "c1" -#> $ arg32: chr "c2"
Transpose turns a list-of-lists "inside-out"; it turns a pair of lists into a
-list of pairs, or a list of pairs into pair of lists. For example,
-if you had a list of length n where each component had values a
and
-b
, transpose()
would make a list with elements a
and
-b
that contained lists of length n. It's called transpose because
-x[[1]][[2]]
is equivalent to transpose(x)[[2]][[1]]
.
transpose(.l, .names = NULL)- -
.l | -A list of vectors to transpose. The first element is used as the -template; you'll get a warning if a subsequent element has a different -length. |
-
---|---|
.names | -For efficiency, |
-
A list with indexing transposed compared to .l
.
Note that transpose()
is its own inverse, much like the
-transpose operation on a matrix. You can get back the original
-input by transposing it twice.
-#> List of 5 -#> $ :List of 2 -#> ..$ x: num 0.475 -#> ..$ y: num [1:5] 0.206 0.707 0.449 0.714 0.358 -#> $ :List of 2 -#> ..$ x: num 0.17 -#> ..$ y: num [1:5] 0.47 0.26 0.136 0.633 0.833 -#> $ :List of 2 -#> ..$ x: num 0.375 -#> ..$ y: num [1:5] 0.8264 0.1672 0.0548 0.1037 0.5826 -#> $ :List of 2 -#> ..$ x: num 0.944 -#> ..$ y: num [1:5] 0.869 0.426 0.257 0.557 0.138 -#> $ :List of 2 -#> ..$ x: num 0.42 -#> ..$ y: num [1:5] 0.2382 0.7582 0.0865 0.0568 0.3083x %>% transpose() %>% str()#> List of 2 -#> $ x:List of 5 -#> ..$ : num 0.475 -#> ..$ : num 0.17 -#> ..$ : num 0.375 -#> ..$ : num 0.944 -#> ..$ : num 0.42 -#> $ y:List of 5 -#> ..$ : num [1:5] 0.206 0.707 0.449 0.714 0.358 -#> ..$ : num [1:5] 0.47 0.26 0.136 0.633 0.833 -#> ..$ : num [1:5] 0.8264 0.1672 0.0548 0.1037 0.5826 -#> ..$ : num [1:5] 0.869 0.426 0.257 0.557 0.138 -#> ..$ : num [1:5] 0.2382 0.7582 0.0865 0.0568 0.3083# Back to where we started -x %>% transpose() %>% transpose() %>% str()#> List of 5 -#> $ :List of 2 -#> ..$ x: num 0.475 -#> ..$ y: num [1:5] 0.206 0.707 0.449 0.714 0.358 -#> $ :List of 2 -#> ..$ x: num 0.17 -#> ..$ y: num [1:5] 0.47 0.26 0.136 0.633 0.833 -#> $ :List of 2 -#> ..$ x: num 0.375 -#> ..$ y: num [1:5] 0.8264 0.1672 0.0548 0.1037 0.5826 -#> $ :List of 2 -#> ..$ x: num 0.944 -#> ..$ y: num [1:5] 0.869 0.426 0.257 0.557 0.138 -#> $ :List of 2 -#> ..$ x: num 0.42 -#> ..$ y: num [1:5] 0.2382 0.7582 0.0865 0.0568 0.3083-# transpose() is useful in conjunction with safely() & quietly() -x <- list("a", 1, 2) -y <- x %>% map(safely(log)) -y %>% str()#> List of 3 -#> $ :List of 2 -#> ..$ result: NULL -#> ..$ error :List of 2 -#> .. ..$ message: chr "non-numeric argument to mathematical function" -#> .. ..$ call : language log(x = x, base = base) -#> .. ..- attr(*, "class")= chr [1:3] "simpleError" "error" "condition" -#> $ :List of 2 -#> ..$ result: num 0 -#> ..$ error : NULL -#> $ :List of 2 -#> ..$ result: num 0.693 -#> ..$ error : NULLy %>% transpose() %>% str()#> List of 2 -#> $ result:List of 3 -#> ..$ : NULL -#> ..$ : num 0 -#> ..$ : num 0.693 -#> $ error :List of 3 -#> ..$ :List of 2 -#> .. ..$ message: chr "non-numeric argument to mathematical function" -#> .. ..$ call : language log(x = x, base = base) -#> .. ..- attr(*, "class")= chr [1:3] "simpleError" "error" "condition" -#> ..$ : NULL -#> ..$ : NULL-# Use simplify_all() to reduce to atomic vectors where possible -x <- list(list(a = 1, b = 2), list(a = 3, b = 4), list(a = 5, b = 6)) -x %>% transpose()#> $a -#> $a[[1]] -#> [1] 1 -#> -#> $a[[2]] -#> [1] 3 -#> -#> $a[[3]] -#> [1] 5 -#> -#> -#> $b -#> $b[[1]] -#> [1] 2 -#> -#> $b[[2]] -#> [1] 4 -#> -#> $b[[3]] -#> [1] 6 -#> -#>#> $a -#> [1] 1 3 5 -#> -#> $b -#> [1] 2 4 6 -#>-# Provide explicit component names to prevent loss of those that don't -# appear in first component -ll <- list( - list(x = 1, y = "one"), - list(z = "deux", x = 2) -) -ll %>% transpose()#> $x -#> $x[[1]] -#> [1] 1 -#> -#> $x[[2]] -#> [1] 2 -#> -#> -#> $y -#> $y[[1]] -#> [1] "one" -#> -#> $y[[2]] -#> NULL -#> -#>#> $x -#> $x[[1]] -#> [1] 1 -#> -#> $x[[2]] -#> [1] 2 -#> -#> -#> $y -#> $y[[1]] -#> [1] "one" -#> -#> $y[[2]] -#> NULL -#> -#> -#> $z -#> $z[[1]] -#> NULL -#> -#> $z[[2]] -#> [1] "deux" -#> -#>
R/when.R
- when.Rd
when
is a flavour of pattern matching (or an if-else abstraction) in
-which a value is matched against a sequence of condition-action sets. When a
-valid match/condition is found the action is executed and the result of the
-action is returned.
when(., ...)- -
. | -the value to match against |
-
---|---|
... | -formulas; each containing a condition as LHS and an action as RHS. -named arguments will define additional values. |
-
The value resulting from the action of the first valid -match/condition is returned. If no matches are found, and no default is -given, NULL will be returned.
-Validity of the conditions are tested with isTRUE
, or equivalently
-with identical(condition, TRUE)
.
-In other words conditions resulting in more than one logical will never
-be valid. Note that the input value is always treated as a single object,
-as opposed to the ifelse
function.
-1:10 %>% - when( - sum(.) <= 50 ~ sum(.), - sum(.) <= 100 ~ sum(.)/2, - ~ 0 - )#> [1] 27.5-1:10 %>% - when( - sum(.) <= x ~ sum(.), - sum(.) <= 2*x ~ sum(.)/2, - ~ 0, - x = 60 - )#> [1] 55-iris %>% - subset(Sepal.Length > 10) %>% - when( - nrow(.) > 0 ~ ., - ~ iris %>% head(10) - )#> Sepal.Length Sepal.Width Petal.Length Petal.Width Species -#> 1 5.1 3.5 1.4 0.2 setosa -#> 2 4.9 3.0 1.4 0.2 setosa -#> 3 4.7 3.2 1.3 0.2 setosa -#> 4 4.6 3.1 1.5 0.2 setosa -#> 5 5.0 3.6 1.4 0.2 setosa -#> 6 5.4 3.9 1.7 0.4 setosa -#> 7 4.6 3.4 1.4 0.3 setosa -#> 8 5.0 3.4 1.5 0.2 setosa -#> 9 4.4 2.9 1.4 0.2 setosa -#> 10 4.9 3.1 1.5 0.1 setosa-iris %>% - head %>% - when(nrow(.) < 10 ~ ., - ~ stop("Expected fewer than 10 rows."))#> Sepal.Length Sepal.Width Petal.Length Petal.Width Species -#> 1 5.1 3.5 1.4 0.2 setosa -#> 2 4.9 3.0 1.4 0.2 setosa -#> 3 4.7 3.2 1.3 0.2 setosa -#> 4 4.6 3.1 1.5 0.2 setosa -#> 5 5.0 3.6 1.4 0.2 setosa -#> 6 5.4 3.9 1.7 0.4 setosa