-
-
Notifications
You must be signed in to change notification settings - Fork 588
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
new implementation of method is_transitive
with linear memory space
#39221
new implementation of method is_transitive
with linear memory space
#39221
Conversation
Documentation preview for this PR (built with commit 3b7cc2e; changes) is ready! 🎉 |
Can you just reuse some existing implementation of dfs/bfs e.g. |
We could do that, but all my trials are way slower. That's why I decided to go this way. |
Of course it's not as much about complexity, but avoiding duplication of code. Where does the slowness come from? both are O(n ⋅ (n + m)) right? (function call overhead? → |
This new version is slightly simpler as it calls |
This seems to be good to go ? What do you think ? |
It looks good to me, but I'm the author so a little biased. It's a rather simple algorithm. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ok, then, green lights
Thank you. I'm setting the tag to positive review on your behalf. |
sagemathgh-39221: new implementation of method `is_transitive` with linear memory space The previous version of method `is_transitive` was building the distance matrix of the digraph and so had memory usage in $O(n^2 + m)$. We change that to a version with memory usage in $O(n + m)$. In addition, this new version is faster for non-transitive digraphs. Before ```sage sage: D = digraphs.Circuit(4) sage: D.is_transitive() False sage: %timeit D.is_transitive() 8.01 µs ± 35.7 ns per loop (mean ± std. dev. of 7 runs, 100,000 loops each) sage: D = digraphs.RandomDirectedGNP(30,.2) sage: D.dig6_string() ']AOGb?dA?pG?S?@?OPIGG???_?O?GC??W_H?BCjaJJA?gO@?A??_?VDHG_ACCiBU?O`HaUC ?kAAD@_AAJErW_G_ICCGa@?S@Oo?IC????DoCQ?Q?@?_@g?O?C?aAGK??o?cKO_W???A?G?G o?H`??Co' sage: D.is_transitive() False sage: %timeit D.is_transitive() 54.3 µs ± 167 ns per loop (mean ± std. dev. of 7 runs, 10,000 loops each) sage: D = digraphs.DeBruijn(5, 2) sage: D.is_transitive() False sage: %timeit D.is_transitive() 80.6 µs ± 116 ns per loop (mean ± std. dev. of 7 runs, 10,000 loops each) sage: D = digraphs.RandomDirectedGNP(20,.2).transitive_closure() sage: D.dig6_string() 'S^~~z~~~^~~z~~~^~~z~~~^~~z~~~^~~z~~~^~~z~~~^~~z~~~^~~z~~~^~~z~~~^~~w' sage: D.is_transitive() True sage: %timeit D.is_transitive() 56 µs ± 181 ns per loop (mean ± std. dev. of 7 runs, 10,000 loops each) ``` With this PR ```sage sage: D = digraphs.Circuit(4) sage: D.is_transitive() False sage: %timeit D.is_transitive() 6.63 µs ± 26.8 ns per loop (mean ± std. dev. of 7 runs, 100,000 loops each) sage: D = DiGraph(']AOGb?dA?pG?S?@?OPIGG???_?O?GC??W_H?BCjaJJA?gO@?A??_?VDHG_ACCiB U?O`HaUC?kAAD@_AAJErW_G_ICCGa@?S@Oo?IC????DoCQ?Q?@?_@g?O?C?aAGK??o?cKO_W ???A?G?Go?H`??Co') sage: D.is_transitive() False sage: %timeit D.is_transitive() 46.7 µs ± 88.4 ns per loop (mean ± std. dev. of 7 runs, 10,000 loops each) sage: D = digraphs.DeBruijn(5, 2) sage: D.is_transitive() False sage: %timeit D.is_transitive() 73.8 µs ± 116 ns per loop (mean ± std. dev. of 7 runs, 10,000 loops each) sage: D = DiGraph('S^~~z~~~^~~z~~~^~~z~~~^~~z~~~^~~z~~~^~~z~~~^~~z~~~^~~ z~~~^~~z~~~^~~w') sage: D.is_transitive() True sage: %timeit D.is_transitive() 58.6 µs ± 118 ns per loop (mean ± std. dev. of 7 runs, 10,000 loops each) ``` ### 📝 Checklist <!-- Put an `x` in all the boxes that apply. --> - [x] The title is concise and informative. - [x] The description explains in detail what this PR is about. - [ ] I have linked a relevant issue or discussion. - [ ] I have created tests covering the changes. - [ ] I have updated the documentation and checked the documentation preview. ### ⌛ Dependencies <!-- List all open PRs that this PR logically depends on. For example, --> <!-- - sagemath#12345: short description why this is a dependency --> <!-- - sagemath#34567: ... --> URL: sagemath#39221 Reported by: David Coudert Reviewer(s): Frédéric Chapoton
The previous version of method$O(n^2 + m)$ . We change that to a version with memory usage in $O(n + m)$ . In addition, this new version is faster for non-transitive digraphs.
is_transitive
was building the distance matrix of the digraph and so had memory usage inBefore
With this PR
📝 Checklist
⌛ Dependencies